/*
 * Copyright (C) 2022 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.
 */

#include "VirtualProgram.h"

#include <broadcastradio-utils-aidl/Utils.h>
#include "resources.h"

#include <android-base/logging.h>

namespace aidl::android::hardware::broadcastradio {

using ::std::vector;

VirtualProgram::operator ProgramInfo() const {
    ProgramInfo info = {};

    info.selector = selector;

    IdentifierType programType = selector.primaryId.type;
    bool isDigital = (programType != IdentifierType::AMFM_FREQUENCY_KHZ &&
                      programType != IdentifierType::RDS_PI);

    auto selectId = [&info](const IdentifierType& type) {
        return utils::makeIdentifier(type, utils::getId(info.selector, type));
    };

    switch (programType) {
        case IdentifierType::AMFM_FREQUENCY_KHZ:
            info.logicallyTunedTo = info.physicallyTunedTo =
                    selectId(IdentifierType::AMFM_FREQUENCY_KHZ);
            break;
        case IdentifierType::RDS_PI:
            info.logicallyTunedTo = selectId(IdentifierType::RDS_PI);
            info.physicallyTunedTo = selectId(IdentifierType::AMFM_FREQUENCY_KHZ);
            break;
        case IdentifierType::HD_STATION_ID_EXT:
            info.logicallyTunedTo = selectId(IdentifierType::HD_STATION_ID_EXT);
            info.physicallyTunedTo = selectId(IdentifierType::AMFM_FREQUENCY_KHZ);
            break;
        case IdentifierType::DAB_SID_EXT:
            info.logicallyTunedTo = selectId(IdentifierType::DAB_SID_EXT);
            info.physicallyTunedTo = selectId(IdentifierType::DAB_ENSEMBLE);
            break;
        case IdentifierType::DRMO_SERVICE_ID:
            info.logicallyTunedTo = selectId(IdentifierType::DRMO_SERVICE_ID);
            info.physicallyTunedTo = selectId(IdentifierType::DRMO_FREQUENCY_KHZ);
            break;
        case IdentifierType::SXM_SERVICE_ID:
            info.logicallyTunedTo = selectId(IdentifierType::SXM_SERVICE_ID);
            info.physicallyTunedTo = selectId(IdentifierType::SXM_CHANNEL);
            break;
        default:
            LOG(FATAL) << "unsupported program type: " << toString(programType);
            return {};
    }

    info.infoFlags |= (ProgramInfo::FLAG_TUNABLE | ProgramInfo::FLAG_STEREO);
    info.signalQuality = isDigital ? kSignalQualityDigital : kSignalQualityNonDigital;

    info.metadata = vector<Metadata>({
            Metadata::make<Metadata::rdsPs>(programName),
            Metadata::make<Metadata::songTitle>(songTitle),
            Metadata::make<Metadata::songArtist>(songArtist),
            Metadata::make<Metadata::stationIcon>(resources::kDemoPngId),
            Metadata::make<Metadata::albumArt>(resources::kDemoPngId),
    });

    info.vendorInfo = vector<VendorKeyValue>({
            {"com.android.sample", "sample"},
            {"com.android.sample.VirtualProgram", "VirtualProgram"},
    });

    return info;
}

bool operator<(const VirtualProgram& lhs, const VirtualProgram& rhs) {
    auto& l = lhs.selector;
    auto& r = rhs.selector;

    // Two programs with the same primaryId are considered the same.
    if (l.primaryId.type != r.primaryId.type) return l.primaryId.type < r.primaryId.type;

    return l.primaryId.value < r.primaryId.value;
}

}  // namespace aidl::android::hardware::broadcastradio
