Отправить пользовательские заголовки с UIWebView loadRequest

Я хочу, чтобы иметь возможность отправлять дополнительные заголовки с помощью моего метода UIWebView loadRequest.

Я пробовал:

NSMutableURLRequest *req = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:@"http://www.reliply.org/tools/requestheaders.php"]];
[req addValue:@"hello" forHTTPHeaderField:@"aHeader"];

[self.theWebView loadRequest:req];

Я также попытался подклассифицировать UIWebView и перехватить метод - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType.

В этом методе у меня был блок кода, который выглядел так:

NSMutableURLRequest *newRequest = [request mutableCopy];
for(NSString *key in [customHeaders allKeys]) {
    [newRequest setValue:[customHeaders valueForKey:key] forHTTPHeaderField:key];
}
[self loadRequest:newRequest];

Но по какой-то неизвестной причине веб-просмотр не загружал ничего (пустой кадр), и появляется сообщение об ошибке NSURLErrorCancelled (-999) (все известные исправления не исправляют его для меня).

Итак, я не понимаю, что делать. Как я могу отправить настраиваемый заголовок вместе с запросом UIWebView?

Большое спасибо!

Ответы

Ответ 1

Я обнаружил, что это способ добавить заголовки в мой запрос UIWebView - переопределить этот метод делегата:

- (BOOL) webView:(UIWebView*)webView shouldStartLoadWithRequest:(NSURLRequest*)request navigationType:(UIWebViewNavigationType) navigationType

С помощью этого кода:

BOOL headerIsPresent = [[request allHTTPHeaderFields] objectForKey:@"my custom header"]!=nil;

if(headerIsPresent) return YES;

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    dispatch_async(dispatch_get_main_queue(), ^{
        NSURL *url = [request URL];
        NSMutableURLRequest* request = [NSMutableURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:60.0];

        // set the new headers
        for(NSString *key in [self.customHeaders allKeys]){
            [request addValue:[self.customHeaders objectForKey:key] forHTTPHeaderField:key];
        }

        // reload the request
        [self loadRequest:request];
    });
});
return NO;

Ответ 2

Ответ Томаса не будет работать для (большинства) веб-страниц с несколькими iFrames. Решение загрузит запрос iFrame по всему UIWebView. Например. в случае, если он вызывает loadRequest для Google advt. (который находится в некотором небольшом iFrame), advt. загружается по всему UIWebView и ничего больше.

Ответ 3

Я нахожу другой способ, могу использовать NSURLProtocol.

-(BOOL)webView:(IMYVKWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
{
    NSMutableDictionary* mapObject = [NSMutableDictionary dictionary];
    mapObject[@"headers"] = request.allHTTPHeaderFields;
    mapObject[@"navigationType"] = @(navigationType);
    [webViewRequestMap setObject:mapObject forKey:request.URL.absoluteString];
    return YES;
}

webViewRequestMap - статический NSMutableDictionary *

в вашем пользовательском коде NSURLProtocol:

    @interface IMYCustomURLProtocol : NSURLProtocol
@end
@implementation IMYCustomURLProtocol 
+(void)load
{
     [NSURLProtocol registerClass:self];
}
+ (BOOL)canInitWithRequest:(NSURLRequest *)request
{
    NSString* urlString = request.URL.absoluteString;
    NSDictionary* dictionary = webViewReuqestMap[urlString];
    if (dictionary)
    {
        [webViewRequestMap removeObjectForKey:urlString];
        if ([request isKindOfClass:[NSMutableURLRequest class]]) {
           [(id)request setValue:@"HAHA" forHTTPHeaderField:@"MeiYou Co.,Ltd"];
        }
    }
    return NO;
}
@end