import 'dart:collection';
import 'dart:convert';
import 'package:aitrainer_app/model/customer.dart';
import 'package:aitrainer_app/model/customer_property.dart';
import 'package:aitrainer_app/model/property.dart';
import 'package:aitrainer_app/model/user.dart';
import 'package:aitrainer_app/service/api.dart';
import 'package:aitrainer_app/model/cache.dart';

class CustomerApi {
  final APIClient _client = new APIClient();

  Future<List<Customer>> getRealCustomers(String param) async {
    final body = await _client.get("customers/", param);
    final Iterable json = jsonDecode(body);
    final List<Customer> customers = json.map((customer) => Customer.fromJson(customer)).toList();

    return customers;
  }

  Future<void> saveCustomer(Customer customer) async {
    String body = JsonEncoder().convert(customer.toJson());
    print(" ===== saving customer id: " + customer.customerId.toString() + ":" + body);
    await _client.post("customers/" + customer.customerId.toString(), body);
  }

  Future<void> updateFirebaseUid(int customerId, String uid) async {
    print(" ===== update Firebase uid : " + customerId.toString() + ": " + uid);
    await _client.post("customers/update_firebase_uid/" + customerId.toString(), uid);
  }

  Future<void> addCustomer(Customer customer) async {
    String body = JsonEncoder().convert(customer.toJson());
    print(" ===== add new customer: " + body);
    await _client.post("customers", body);
  }

  Future<void> addUser(User user) async {
    String body = JsonEncoder().convert(user.toJson());
    print(" ===== add new user: " + body);
    final String responseBody = await _client.post("registration", body);
    Customer customer;
    try {
      int status = jsonDecode(responseBody)['status'];
      if (status != null) {
        throw new Exception(jsonDecode(responseBody)['error']);
      } else {
        customer = Customer.fromJson(jsonDecode(responseBody));
        Cache().afterRegistration(customer);
      }
    } on FormatException {
      throw new Exception(responseBody);
    }
  }

  Future<void> getUser(User user) async {
    String body = JsonEncoder().convert(user.toJson());
    print(" ===== login the user: " + body);
    final String responseBody = await _client.post("login", body);
    Customer customer;
    try {
      customer = Customer.fromJson(jsonDecode(responseBody));
      await Cache().afterLogin(customer);
    } on FormatException {
      throw new Exception(responseBody);
    }
  }

  Future<void> getUserByEmail(String email) async {
    print(" ===== User getByEmail : " + email);
    final String responseBody = await _client.get("customers/find_by_email/" + email, "");
    Customer customer;
    try {
      customer = Customer.fromJson(jsonDecode(responseBody));
      if (customer.firebaseUid == null) {
        await this.updateFirebaseUid(customer.customerId, Cache().firebaseUid);
      }
      Cache().userLoggedIn = customer;
    } on FormatException {
      throw new Exception(responseBody);
    }
  }

  Future<void> getCustomer(int customerId) async {
    String body = "";
    print(" ===== get the customer by id: " + customerId.toString());
    try {
      final String responseBody = await _client.get("customers/" + customerId.toString(), body);
      Customer customer = Customer.fromJson(jsonDecode(responseBody));
      print(" --- Customer: " + customer.toJson().toString());
      final List properties = await this.getActualProperties(customerId);
      print(" ---- Props: " + properties.toString());
      Cache().afterRegistration(customer);
      this._initProperties(properties);
    } catch (exception) {
      print("Exception: " + exception.toString());
      print(" === go to registration ");
      Cache().logout();
      Cache().startPage = "registration";
    }
  }

  void _initProperties(List<CustomerProperty> customerProperties) {
    List<Property> properties = Cache().getProperties();
    Customer customer = Cache().userLoggedIn;
    customer.properties = LinkedHashMap<String, CustomerProperty>();

    // reset Properties
    properties.forEach((property) {
      CustomerProperty customerProperty =
          CustomerProperty(propertyId: property.propertyId, customerId: customer.customerId, dateAdd: null, propertyValue: 0);
      customer.properties[property.propertyName] = customerProperty;
    });

    customerProperties.forEach((customerProperty) {
      properties.forEach((property) {
        if (customerProperty.propertyId == property.propertyId) {
          customer.properties[property.propertyName] = customerProperty;
        }
      });
    });
  }

  Future<Customer> getTrainee(int customerId) async {
    String body = "";
    Customer customer;
    print(" ===== get Trainee customer by id: " + customerId.toString());
    try {
      final String responseBody = await _client.get("customers/" + customerId.toString(), body);
      customer = Customer.fromJson(jsonDecode(responseBody));
      print(" --- Trainee: " + customer.toJson().toString());
    } catch (exception) {
      print("Exception: " + exception.toString());
      throw Exception(exception);
    }
    return customer;
  }

  Future<List<Customer>> getTrainees(int trainerId) async {
    List<Customer> trainees = List<Customer>();
    print("Get trainees list");
    try {
      String body = "";
      final String responseBody = await _client.get("customers/trainees/" + trainerId.toString(), body);
      final Iterable json = jsonDecode(responseBody);
      trainees = json.map((customer) => Customer.fromJson(customer)).toList();
    } catch (exception) {
      print("Exception: " + exception.toString());
      throw Exception(exception);
    }
    return trainees;
  }

  Future<List<CustomerProperty>> getAllProperties(int customerId) async {
    final body = await _client.get("customer_property/", customerId.toString());
    final Iterable json = jsonDecode(body);
    final List<CustomerProperty> properties = json.map((property) => CustomerProperty.fromJson(property)).toList();

    return properties;
  }

  Future<List<CustomerProperty>> getActualProperties(int customerId) async {
    final body = await _client.get("customer_property/last/", customerId.toString());
    final Iterable json = jsonDecode(body);
    final List<CustomerProperty> properties = json.map((property) => CustomerProperty.fromJson(property)).toList();

    return properties;
  }

  Future<void> addProperty(CustomerProperty property) async {
    String body = JsonEncoder().convert(property.toJson());
    print(" ===== add new customer property: " + body);
    final String responseBody = await _client.post("customer_property", body);
    try {
      int status = jsonDecode(responseBody)['status'];
      if (status != null) {
        throw new Exception(jsonDecode(responseBody)['error']);
      } else {
        CustomerProperty customerProperty = CustomerProperty.fromJson(jsonDecode(responseBody));
        if (customerProperty == null) {
          throw new Exception("Property Insert was not successful");
        }
      }
    } on FormatException {
      throw new Exception(responseBody);
    }
  }
}