using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
using FarmmapsApi;
using FarmmapsApi.Models;
using FarmmapsApi.Services;
using FarmmapsPoten.Models;
using Microsoft.Extensions.Logging;
using Newtonsoft.Json;
using static FarmmapsApiSamples.Constants;


namespace FarmmapsVRApoten
{
    public class PotenApplication : IApplication {
        private const string DownloadFolder = "Downloads";

        private readonly ILogger<PotenApplication> _logger;
        private readonly FarmmapsApiService _farmmapsApiService;
        private readonly PotenService _potenService;
        private readonly GeneralService _generalService;


        public PotenApplication(ILogger<PotenApplication> logger, FarmmapsApiService farmmapsApiService,
            GeneralService generalService, PotenService potenService) {
            _logger = logger;
            _farmmapsApiService = farmmapsApiService;
            _generalService = generalService;
            _potenService = potenService;
        }

        public async Task RunAsync() {
            // read field data from separate json file
            var VRAPotenInputJson = File.ReadAllText("PotenInput.json");
            List<PotenInput> potenInputs = JsonConvert.DeserializeObject<List<PotenInput>>(VRAPotenInputJson);


            if (!Directory.Exists(DownloadFolder))
                Directory.CreateDirectory(DownloadFolder);

            // !! this call is needed the first time an api is called with a fresh clientid and secret !!
            await _farmmapsApiService.GetCurrentUserCodeAsync();
            var roots = await _farmmapsApiService.GetCurrentUserRootsAsync();

            foreach (var input in potenInputs) {
                try {
                    await Process(roots, input);
                } catch (Exception ex) {
                    _logger.LogError(ex.Message);
                }
            }
        }

        private async Task Process(List<UserRoot> roots, PotenInput input) {
            var meanDensity = input.MeanDensity;
            var variation = input.Variation;
            var fieldName = input.FieldName;
            bool useShadow = input.UseShadow;
            bool countPerArea = input.CountPerArea;
            var inputLayerName = input.InputLayerName;

            var myDrive = roots.SingleOrDefault(r => r.Name == "USER_FILES");
            if (myDrive == null) {
                _logger.LogError("Could not find a needed root item");
                return;
            }

            var uploadedRoot = roots.SingleOrDefault(r => r.Name == "USER_IN");
            if (uploadedRoot == null) {
                _logger.LogError("Could not find a needed root item");
                return;
            }

            _logger.LogInformation("Creating cropfield");

            var cropfieldItem = await _generalService.CreateCropfieldItemAsync(myDrive.Code,
                    $"VRA Poten cropfield {input.OutputFileName}", input.PlantingYear,
                input.GeometryJson.ToString(Formatting.None));

            _logger.LogInformation($"CropfielditemCode: {cropfieldItem.Code}");


            //Downloading shadowMap for own interpretation
            if (useShadow) {
                _logger.LogInformation("Calculate shadow map for field");
                var shadowItem = await _generalService.RunShadowTask(cropfieldItem);
                if (shadowItem == null) {
                    _logger.LogError("Something went wrong while obtaining the shadow map");
                    return;
                }

                _logger.LogInformation("Downloading shadow map");
                await _farmmapsApiService.DownloadItemAsync(shadowItem.Code,
                    Path.Combine(DownloadFolder, $"{input.OutputFileName}.shadow.zip"));
            }

            _logger.LogInformation("Looking for local data to use");
            var localDataAvailable = input.File;
            var geotiffItem = (Item)null;


            if (String.IsNullOrEmpty(localDataAvailable)) {
                _logger.LogInformation("Could not find item for uploaded data, using BOFEK");

                //Retreiving BOFEK
                _logger.LogInformation("Get BOFEK for field");
                var bofekItem = await _generalService.RunBofekTask(cropfieldItem);
                if (bofekItem == null) {
                    _logger.LogError("Something went wrong while obtaining the BOFEK data");
                    return;
                }

                _logger.LogInformation("Downloading Bofek map");
                await _farmmapsApiService.DownloadItemAsync(bofekItem.Code,
                    Path.Combine(DownloadFolder, $"{input.OutputFileName}.BOFEK.zip"));
            } else if (input.File.Contains(".tif") || input.File.Contains(".geotiff")) {
                _logger.LogInformation("input = tiff data");
                var dataPath = Path.Combine("Data", input.File);
                geotiffItem = await _generalService.UploadDataAsync(uploadedRoot, GEOTIFF_PROCESSED_ITEMTYPE, dataPath,
                        Path.GetFileNameWithoutExtension(input.File));

                if (geotiffItem == null) {
                    _logger.LogError("Could not find item for uploaded data");
                    return;
                }
            } else {
                var isGeoJson = input.File.Contains("json");
                var dataPath = Path.Combine("Data", input.File);
                var shapeItem = isGeoJson
                    ? await _generalService.UploadDataAsync(uploadedRoot, SHAPE_PROCESSED_ITEMTYPE, dataPath,
                        Path.GetFileNameWithoutExtension(input.File))
                    : await _generalService.UploadZipWithShapeAsync(uploadedRoot, dataPath,
                        Path.GetFileNameWithoutExtension(input.File), input.GeometryJson.ToString(Formatting.None));

                if (shapeItem == null) {
                    _logger.LogError("Something went wrong while searching for the shape file");
                    return;
                }

                // transform shape to geotiff as VRA poten only supports tiff input item
                _logger.LogInformation($"Converting shape to geotiff");

                geotiffItem = await _generalService.ShapeToGeotiff(shapeItem);
                if (geotiffItem == null) {
                    _logger.LogError("Something went wrong with shape to geotiff transformation");
                    return;
                }
                
                _logger.LogInformation($"ShapeToGeotiff_GeotiffItemcode: {geotiffItem.Code}");

                _logger.LogInformation("Downloading geotiff file");
                await _farmmapsApiService.DownloadItemAsync(geotiffItem.Code,
                    Path.Combine(DownloadFolder, $"VRApoten_inputGeotiff_{input.OutputFileName}.zip"));
            }



            // create appliance map
            _logger.LogInformation("Calculating application map");

            // INPUT IS NEEDED as GEOTIFF
            var applianceMapItem =
                await _potenService.CalculateApplicationMapAsync(cropfieldItem, geotiffItem, meanDensity, variation, countPerArea, useShadow, inputLayerName);

            if (applianceMapItem == null) {
                return;
            }

            _logger.LogInformation("Downloading application map");
            await _farmmapsApiService.DownloadItemAsync(applianceMapItem.Code,
                Path.Combine(DownloadFolder, $"VRApoten_appliancemap_{input.OutputFileName}.zip"));

            string finalOutput = Path.Combine(DownloadFolder, $"VRApoten_appliancemap_{input.OutputFileName}.zip");
            _logger.LogInformation(File.Exists(finalOutput)
                ? "Download application map completed."
                : "Something went wrong while downloading.");
   
            

            if(input.GenerateTaskmap) { 
            //GEOTIFF TO Taskmap
            _logger.LogInformation($"Converting geotiff to taskmap");

                var taskmap = (Item)null;
            if (input.OutputType == "isoxml") 
                {
                    
                    if (input.DdiCode == null)
                    {
                        _logger.LogInformation("DDi not given. Using expected identifiers");
                        if (countPerArea == true) {input.DdiCode = input.DdiCode = "0011";}
                        else { input.DdiCode = "0016"; };
                    }
                    taskmap = await _generalService.CreateTaskmap(cropfieldItem: cropfieldItem, tiffItem: applianceMapItem, outputType: input.OutputType, cellWidth: input.CellWidth,
                        cellHeight: input.CellHeight, startPoint: input.StartPoint.ToString(Formatting.None), ddiCode: input.DdiCode, centered: input.Centered, 
                        endPoint: input.EndPoint.ToString(Formatting.None), angle: input.Angle, precision: input.Precision, 
                        cropTypeName: null, costumerName: null, ProductGroupName: null, productName : null, resolution: "3", unitScale: null, maximumClasses: input.MaximumClasses);

                } else
                {
                    taskmap = await _generalService.CreateTaskmap(cropfieldItem: cropfieldItem, tiffItem: applianceMapItem, outputType: input.OutputType, cellWidth: input.CellWidth,
                        cellHeight: input.CellHeight, startPoint: input.StartPoint.ToString(Formatting.None), centered: input.Centered,
                        endPoint: input.EndPoint.ToString(Formatting.None), angle: input.Angle, precision: input.Precision, maximumClasses: input.MaximumClasses);
                }



            if (taskmap == null)
            {
                _logger.LogError("Something went wrong with geotiff to taskmap transformation");
                return;
            }

            _logger.LogInformation("Downloading taskmap");
            await _farmmapsApiService.DownloadItemAsync(taskmap.Code,
                Path.Combine(DownloadFolder, $"VRApoten_taskmap_{input.OutputFileName}_isoxml.zip"));



            }
        }
    }
}