1*6dbdd20aSAndroid Build Coastguard Worker// Copyright (C) 2024 The Android Open Source Project 2*6dbdd20aSAndroid Build Coastguard Worker// 3*6dbdd20aSAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*6dbdd20aSAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*6dbdd20aSAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*6dbdd20aSAndroid Build Coastguard Worker// 7*6dbdd20aSAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*6dbdd20aSAndroid Build Coastguard Worker// 9*6dbdd20aSAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*6dbdd20aSAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*6dbdd20aSAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*6dbdd20aSAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*6dbdd20aSAndroid Build Coastguard Worker// limitations under the License. 14*6dbdd20aSAndroid Build Coastguard Worker 15*6dbdd20aSAndroid Build Coastguard Workertype Reducer = () => unknown; 16*6dbdd20aSAndroid Build Coastguard Workertype Callback = () => void; 17*6dbdd20aSAndroid Build Coastguard Worker 18*6dbdd20aSAndroid Build Coastguard Worker/** 19*6dbdd20aSAndroid Build Coastguard Worker * A little helper that monitors a list of immutable objects and calls a 20*6dbdd20aSAndroid Build Coastguard Worker * callback only when at least one them changes. 21*6dbdd20aSAndroid Build Coastguard Worker */ 22*6dbdd20aSAndroid Build Coastguard Workerexport class Monitor { 23*6dbdd20aSAndroid Build Coastguard Worker private cached: unknown[]; 24*6dbdd20aSAndroid Build Coastguard Worker 25*6dbdd20aSAndroid Build Coastguard Worker constructor(private reducers: Reducer[]) { 26*6dbdd20aSAndroid Build Coastguard Worker this.cached = reducers.map(() => undefined); 27*6dbdd20aSAndroid Build Coastguard Worker } 28*6dbdd20aSAndroid Build Coastguard Worker 29*6dbdd20aSAndroid Build Coastguard Worker /** 30*6dbdd20aSAndroid Build Coastguard Worker * Invokes all reducers and compares values against with the previous values. 31*6dbdd20aSAndroid Build Coastguard Worker * 32*6dbdd20aSAndroid Build Coastguard Worker * If any of the values have changed, |callback| is called (if present) and 33*6dbdd20aSAndroid Build Coastguard Worker * returns true, otherwise no callback is called and returns false. 34*6dbdd20aSAndroid Build Coastguard Worker * 35*6dbdd20aSAndroid Build Coastguard Worker * @param callback Optional callback to call when diffs are detected. 36*6dbdd20aSAndroid Build Coastguard Worker * @returns True if diffs were detected, false otherwise. 37*6dbdd20aSAndroid Build Coastguard Worker */ 38*6dbdd20aSAndroid Build Coastguard Worker ifStateChanged(callback?: Callback): boolean { 39*6dbdd20aSAndroid Build Coastguard Worker const oldState = this.cached; 40*6dbdd20aSAndroid Build Coastguard Worker const newState = this.reducers.map((f) => f()); 41*6dbdd20aSAndroid Build Coastguard Worker this.cached = newState; 42*6dbdd20aSAndroid Build Coastguard Worker if (newState.some((x, i) => x !== oldState[i])) { 43*6dbdd20aSAndroid Build Coastguard Worker callback?.(); 44*6dbdd20aSAndroid Build Coastguard Worker return true; 45*6dbdd20aSAndroid Build Coastguard Worker } 46*6dbdd20aSAndroid Build Coastguard Worker 47*6dbdd20aSAndroid Build Coastguard Worker return false; 48*6dbdd20aSAndroid Build Coastguard Worker } 49*6dbdd20aSAndroid Build Coastguard Worker} 50