using BankContracts.BindingModels;
using BankContracts.SearchModels;
using BankContracts.StoragesContracts;
using BankContracts.ViewModels;
using BankDataBaseImplement.Models;
using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace BankDataBaseImplement.Implemets
{
    public class DinnerStorage : IDinnerStorage
    {
        public DinnerViewModel? Delete(DinnerBindingModel model)
        {
            using var context = new BankDataBase();

            var element = context.Dinners.FirstOrDefault(rec => rec.Id == model.Id);

            if (element != null)
            {
                context.Dinners.Remove(element);
                context.SaveChanges();

                return element.GetViewModel;
            }

            return null;
        }

        public DinnerViewModel? GetElement(DinnerSearchModel model)
        {
            if (string.IsNullOrEmpty(model.DinnerName) && !model.Id.HasValue)
            {
                return null;
            }

            using var context = new BankDataBase();

            return context.Dinners
                .Include(x => x.RoomDinners)
                .ThenInclude(x => x.Room)
                .Include(x => x.ConferenceBookingDinners)
                .ThenInclude(x => x.ConferenceBooking)
                .FirstOrDefault(x => (!string.IsNullOrEmpty(model.DinnerName) && x.DinnerName == model.DinnerName) || (model.Id.HasValue && x.Id == model.Id))?
                .GetViewModel;
        }

        public List<DinnerViewModel> GetFilteredList(DinnerSearchModel model)
        {
            if (string.IsNullOrEmpty(model.DinnerName))
            {
                return new();
            }

            using var context = new BankDataBase();

            return context.Dinners
                .Include(x => x.RoomDinners)
                .ThenInclude(x => x.Room)
                .Include(x => x.ConferenceBookingDinners)
                .ThenInclude(x => x.ConferenceBooking)
                .Where(x => x.DinnerName.Contains(model.DinnerName))
                .ToList()
                .Select(x => x.GetViewModel)
                .ToList();
        }

        public List<DinnerViewModel> GetFullList()
        {
            using var context = new BankDataBase();

            return context.Dinners
                .Include(x => x.RoomDinners)
                .ThenInclude(x => x.Room)
                .Include(x => x.ConferenceBookingDinners)
                .ThenInclude(x => x.ConferenceBooking)
                .ToList()
                .Select(x => x.GetViewModel)
                .ToList();
        }

        public DinnerViewModel? Insert(DinnerBindingModel model)
        {
            using var context = new BankDataBase();

            var newDinner = Dinner.Create(model);

            if (newDinner == null)
            {
                return null;
            }

            context.Dinners.Add(newDinner);
            context.SaveChanges();

            return newDinner.GetViewModel;
        }

        public DinnerViewModel? Update(DinnerBindingModel model)
        {
            using var context = new BankDataBase();

            var dinner = context.Dinners.FirstOrDefault(x => x.Id == model.Id);

            if (dinner == null)
            {
                return null;
            }

            dinner.Update(model);
            context.SaveChanges();

            return dinner.GetViewModel;
        }
    }
}