1 /*
2  * Copyright (C) 2019 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.settings.dashboard;
18 
19 import android.util.Log;
20 
21 import androidx.annotation.NonNull;
22 
23 import com.android.settings.core.BasePreferenceController;
24 import com.android.settingslib.utils.ThreadUtils;
25 
26 import java.util.HashSet;
27 import java.util.List;
28 import java.util.Set;
29 import java.util.concurrent.CountDownLatch;
30 import java.util.concurrent.TimeUnit;
31 
32 /**
33  * Control ui blocker data and check whether it is finished
34  *
35  * @see BasePreferenceController.UiBlocker
36  * @see BasePreferenceController.UiBlockListener
37  */
38 public class UiBlockerController {
39     private static final String TAG = "UiBlockerController";
40     private static final int TIMEOUT_MILLIS = 300;
41 
42     private CountDownLatch mCountDownLatch;
43     private boolean mBlockerFinished;
44     private Set<String> mKeys;
45     private long mTimeoutMillis;
46 
UiBlockerController(@onNull List<String> keys)47     public UiBlockerController(@NonNull List<String> keys) {
48         this(keys, TIMEOUT_MILLIS);
49     }
50 
UiBlockerController(@onNull List<String> keys, long timeout)51     public UiBlockerController(@NonNull List<String> keys, long timeout) {
52         mCountDownLatch = new CountDownLatch(keys.size());
53         mBlockerFinished = keys.isEmpty();
54         mKeys = new HashSet<>(keys);
55         mTimeoutMillis = timeout;
56     }
57 
58     /**
59      * Start background thread, it will invoke {@code finishRunnable} if any condition is met
60      *
61      * 1. Waiting time exceeds {@link #mTimeoutMillis}
62      * 2. All background work that associated with {@link #mCountDownLatch} is finished
63      */
start(Runnable finishRunnable)64     public boolean start(Runnable finishRunnable) {
65         if (mKeys.isEmpty()) {
66             // Don't need to run finishRunnable because it doesn't start
67             return false;
68         }
69         ThreadUtils.postOnBackgroundThread(() -> {
70             try {
71                 mCountDownLatch.await(mTimeoutMillis, TimeUnit.MILLISECONDS);
72             } catch (InterruptedException e) {
73                 Log.w(TAG, "interrupted");
74             }
75             mBlockerFinished = true;
76             ThreadUtils.postOnMainThread(finishRunnable);
77         });
78 
79         return true;
80     }
81 
82     /**
83      * Return {@code true} if all work finished
84      */
isBlockerFinished()85     public boolean isBlockerFinished() {
86         return mBlockerFinished;
87     }
88 
89     /**
90      * Count down latch by {@code key}. It only count down 1 time if same key count down multiple
91      * times.
92      */
countDown(String key)93     public boolean countDown(String key) {
94         if (mKeys.remove(key)) {
95             mCountDownLatch.countDown();
96             return true;
97         }
98 
99         return false;
100     }
101 }
102