How to set custom HTTP headers to requests made by a WKWebView

11,807

Solution 1

I've got it working in a way, but only get requests will have the custom header. As jbelkins answered in the linked so from Gabriel Cartiers comment to your question, you will have to manipulate the request and load it anew.

I've got it working for GET-Requests like this:

(it's in xamarin 0> c#, but i think you will get the idea)

i've created a private field

private bool _headerIsSet

which i check every time a request is made in the deligate method:

[Foundation.Export("webView:decidePolicyForNavigationAction:decisionHandler:")]
    public void DecidePolicy(WKWebView webView, WKNavigationAction navigationAction, Action<WKNavigationActionPolicy> decisionHandler)
    {
        var request = navigationAction.Request;
        // check if the header is set and if not, create a muteable copy of the original request
        if (!_headerIsSet && request is NSMuteableUrlRequest muteableRequest);              
        {
            // define your custom header name and value
            var keys = new object[] {headerKeyString};
            var values = new object[] {headerValueString};
            var headerDict = NSDictionary.FromObjectsAndKeys(values, keys);
            // set the headers of the new request to the created dict
            muteableRequest.Headers = headerDict;
            _headerIsSet = true;
            // attempt to load the newly created request
            webView.LoadRequest(muteableRequest);
            // abort the old one
            decisionHandler(WKNavigationActionPolicy.Cancel);
            // exit this whole method
            return;
        }
        else
        {
            _headerIsSet = false;                
            decisionHandler(WKNavigationActionPolicy.Allow);
        }
    }

As i said, this only works for GET-Requests. Somehow, POST-Requests don't contain the body data of the original request (request.Body and request.BodyStream are null), so the muteableRequest (which is a muteable copy of the original request) won't contain the body data of the original request.

I hope this will help you or others that approach the same problem.

Edit: For your needs, set "Accept-Language" as the key

Solution 2

Simply can set needed language ISO 639-1 code in URL request like below, so that we can get user preferred or locale language response from server side.

Swift 4 & above

var request = URLRequest(url: URL(string: "YourUrlStr"))
request.setValue("en", forHTTPHeaderField: "Accept-Language")
wkWebView.load(request)

Objective-C

NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:YourUrlStr]];
[request setValue:@"en" forHTTPHeaderField:@"Accept-Language"];
[wkWebView loadRequest:urlRequest];
Share:
11,807
Admin
Author by

Admin

Updated on June 13, 2022

Comments

  • Admin
    Admin almost 2 years

    I have built an app that includes a WKWebView, and the website that the web view loads supports multiple languages. How can I change the Accept-Language header in a WKWebView, or other HTTP headers for that matter?

  • Alex Cohn
    Alex Cohn over 2 years
    This workaround has another problem except being limited to GET. It also assumes that the request it will intercept, replaces the whole content of the WebView. This isn't always true: if such secondary request is intended to change some elements of the page, e.g. an image displayed there, the result will be far from what was intended.
  • DerDingens
    DerDingens over 2 years
    @AlexCohn I'm not sure about that, did you encounter such a behavior? DecidePolicy should only be called if a navigation action is happening, e.g. the page changes
  • Alex Cohn
    Alex Cohn over 2 years
    This is exactly the problem: ajax requests, image requests, etc. are not handled by this approach.
  • DerDingens
    DerDingens over 2 years
    Oh okay, misunderstood your point there, true that won't be handled this way
  • Alex Cohn
    Alex Cohn over 2 years
    This can be achieved by intercepting the HTTP requests, but this may be like using a cannon to kill a fly.