// Uncomment to force TweetDeck to load a predefined version of the vendor/bundle scripts
// #define FREEZE_TWEETDECK_SCRIPTS

using System.Collections.Specialized;
using CefSharp;
using TweetDuck.Core.Handling.Filters;
using TweetDuck.Core.Utils;

#if FREEZE_TWEETDECK_SCRIPTS
using System.Collections.Generic;
using System.Text.RegularExpressions;
#endif

namespace TweetDuck.Core.Handling{
    sealed class RequestHandlerBrowser : RequestHandlerBase{
        public string BlockNextUserNavUrl { get; set; }

        public RequestHandlerBrowser() : base(true){}

        public override CefReturnValue OnBeforeResourceLoad(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IRequestCallback callback){
            if (request.ResourceType == ResourceType.Script && request.Url.Contains("analytics.")){
                callback.Dispose();
                return CefReturnValue.Cancel;
            }

            return base.OnBeforeResourceLoad(browserControl, browser, frame, request, callback);
        }

        public override bool OnBeforeBrowse(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, bool userGesture, bool isRedirect){
            if (userGesture && request.TransitionType == TransitionType.LinkClicked){
                bool block = request.Url == BlockNextUserNavUrl;
                BlockNextUserNavUrl = string.Empty;
                return block;
            }

            return base.OnBeforeBrowse(browserControl, browser, frame, request, userGesture, isRedirect);
        }

        #if FREEZE_TWEETDECK_SCRIPTS
        private static readonly Regex TweetDeckScriptUrl = new Regex(@"/dist/(.*?)\.(.*?)\.js$", RegexOptions.Compiled);
        
        private static readonly SortedList<string, string> TweetDeckHashes = new SortedList<string, string>(2){
            { "vendor", "942c0a20e8" },
            { "bundle", "1bd75b5854" }
        };
        #endif

        public override bool OnResourceResponse(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IResponse response){
            if (request.ResourceType == ResourceType.Image && request.Url.Contains("/backgrounds/spinner_blue")){
                request.Url = TwitterUtils.LoadingSpinner.Url;
                return true;
            }
            #if FREEZE_TWEETDECK_SCRIPTS
            else if (request.ResourceType == ResourceType.Script){
                Match match = TweetDeckScriptUrl.Match(request.Url);

                if (match.Success && TweetDeckHashes.TryGetValue(match.Groups[1].Value, out string hash)){
                    if (match.Groups[2].Value == hash){
                        System.Diagnostics.Debug.WriteLine($"accepting {request.Url}");
                    }
                    else{
                        System.Diagnostics.Debug.WriteLine($"rewriting {request.Url} to {hash}");
                        request.Url = TweetDeckScriptUrl.Replace(request.Url, "/dist/$1."+hash+".js");
                        return true;
                    }
                }
            }
            #endif

            return base.OnResourceResponse(browserControl, browser, frame, request, response);
        }

        public override IResponseFilter GetResourceResponseFilter(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IResponse response){
            if (request.ResourceType == ResourceType.Script && request.Url.Contains("/dist/vendor")){
                NameValueCollection headers = response.ResponseHeaders;

                if (int.TryParse(headers["x-ton-expected-size"], out int totalBytes)){
                    return new ResponseFilterVendor(totalBytes);
                }
                #if DEBUG
                else{
                    System.Diagnostics.Debug.WriteLine($"Missing uncompressed size header in {request.Url}");

                    foreach(string key in headers){
                        System.Diagnostics.Debug.WriteLine($" {key}: {headers[key]}");
                    }

                    System.Diagnostics.Debugger.Break();
                }
                #endif
            }

            return base.GetResourceResponseFilter(browserControl, browser, frame, request, response);
        }
    }
}