/*
 * Copyright (C) 2021 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.android.car.settings.enterprise;

import android.annotation.Nullable;
import android.car.drivingstate.CarUxRestrictions;
import android.content.Context;

import androidx.preference.Preference;

import com.android.car.settings.common.FragmentController;
import com.android.car.settings.enterprise.CallbackTranslator.Callback;
import com.android.car.settingslib.applications.ApplicationFeatureProvider;

/**
 * Base class for async controllers.
 */
abstract class BaseAsyncPreferenceController<P extends Preference, R,
        T extends CallbackTranslator<R>> extends BaseEnterprisePrivacyPreferenceController<P> {

    private R mResult;

    @Nullable
    private T mCallbackTranslator;

    @Nullable
    private Integer mAvailabilityStatus;

    protected BaseAsyncPreferenceController(Context context,
            String preferenceKey, FragmentController fragmentController,
            CarUxRestrictions uxRestrictions,
            @Nullable ApplicationFeatureProvider applicationFeatureProvider) {
        super(context, preferenceKey, fragmentController, uxRestrictions,
                /* enterprisePrivacyFeatureProvider= */ null, applicationFeatureProvider);
    }

    /**
     * Calls the method that will lazy-load the counter.
     */
    protected abstract void lazyLoad(T callbackHolder);

    /**
     * Creates a translator between this class Callback and {@code SettingsLib}.
     * @param callback
     * @return
     */
    protected abstract T newCallbackTranslator(Callback<R> callback);

    /**
     * Checks if the result of the callback should make the controller available.
     */
    protected abstract boolean isAvailable(R result);

    /**
     * Gets the result returned by the callback.
     */
    protected R getResult() {
        return mResult;
    }

    @Override
    protected final int getDefaultAvailabilityStatus() {
        if (mAvailabilityStatus != null) {
            // Already calculated
            mLogger.d("getAvailabilityStatus(): returning cached result " + mAvailabilityStatus);
            return mAvailabilityStatus;
        }

        if (mCallbackTranslator != null) {
            mLogger.d("getAvailabilityStatus(): already waiting for callback...");
        } else {
            mLogger.d("getAvailabilityStatus(): lazy-loading number of apps");
            mCallbackTranslator = newCallbackTranslator((result) -> onLazyLoaded(result));
            lazyLoad(mCallbackTranslator);
        }

        // Calculating the number of apps can takes a bit of time, so we always return
        // CONDITIONALLY_UNAVAILABLE, so the actual visibility will be set when the result is called
        // back (on onLazyLoaded()).
        return CONDITIONALLY_UNAVAILABLE;
    }

    private void onLazyLoaded(R result) {
        mResult = result;
        boolean available = isAvailable(result);
        mAvailabilityStatus = available
                ? AVAILABLE
                : DISABLED_FOR_PROFILE;
        mLogger.d("onLazyLoaded(): result=" + result + ", available=" + available);

        if (available)  {
            refreshUi();
        }
    }
}