我希望能够使用我的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];
但由于某些未知原因导致Web视图无法加载任何内容(空白帧)并出现错误消息NSURLErrorCancelled (-999)
(所有已知的修复程序都没有为我修复)。
所以我不知道该怎么做。如何发送自定义标头以及UIWebView
请求?
非常感谢!
答案 0 :(得分:20)
我发现这是向我的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;
答案 1 :(得分:7)
托马斯的回答不适用于具有多个iFrame的(大多数)网页。该解决方案将在完整的UIWebView上加载iFrame请求。例如。如果它为Google推荐调用loadRequest。 (这是在一些小的iFrame中)的建议。整个UIWebView&没别的。
答案 2 :(得分:2)
我找到另一种方法,可以使用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 *
:
@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
答案 3 :(得分:0)
这是使用NSURLProticol的完整实现。您可以将此代码放在您喜欢的任何位置(例如,它自己的代码或添加到现有的源文件中),它应该可以工作。自定义的两个关键方法是canInitWithRequest:
和canonicalRequestForRequest:
。
static NSString * const NSURLProtocolHandledKey = @"NSURLProtocolHandledKey";
@interface WTCURLProtocol : NSURLProtocol<NSURLSessionDelegate>
@property (atomic,strong,readwrite) NSURLSessionDataTask *task;
@property (nonatomic,strong) NSURLSession *session;
@end
@implementation WTCURLProtocol
+(void)load
{
[NSURLProtocol registerClass:self];
}
+ (BOOL)canInitWithRequest:(NSURLRequest *)request
{
// customize here by returning true for URLs that you want to handle
return [request.URL.absoluteString hasPrefix:WEB_BASE_URL];
}
+ (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest *)request
{
NSMutableURLRequest *newRequest = request.mutableCopy;
[NSURLProtocol setProperty:@YES forKey:NSURLProtocolHandledKey inRequest:newRequest];
// customize here by setting your custom headers
[newRequest setValue:@"ABCDEFGHIJKLMNOPQRSTUVWXYZ" forHTTPHeaderField:@"API-TOKEN"];
return newRequest;
}
- (void)startLoading
{
NSURLSessionConfiguration *configure = [NSURLSessionConfiguration defaultSessionConfiguration];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
self.session = [NSURLSession sessionWithConfiguration:configure delegate:self delegateQueue:queue];
self.task = [self.session dataTaskWithRequest:self.request];
[self.task resume];
}
- (void)stopLoading
{
[self.session invalidateAndCancel];
self.session = nil;
}
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
if (error != nil) {
[self.client URLProtocol:self didFailWithError:error];
}else
{
[self.client URLProtocolDidFinishLoading:self];
}
}
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
didReceiveResponse:(NSURLResponse *)response
completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
{
[self.client URLProtocol:self didReceiveResponse:response cacheStoragePolicy:NSURLCacheStorageNotAllowed];
completionHandler(NSURLSessionResponseAllow);
}
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
[self.client URLProtocol:self didLoadData:data];
}
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask willCacheResponse:(NSCachedURLResponse *)proposedResponse completionHandler:(void (^)(NSCachedURLResponse * _Nullable))completionHandler
{
completionHandler(proposedResponse);
}
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task willPerformHTTPRedirection:(NSHTTPURLResponse *)response newRequest:(NSURLRequest *)newRequest completionHandler:(void (^)(NSURLRequest *))completionHandler
{
NSMutableURLRequest *redirectRequest = [newRequest mutableCopy];
[[self class] removePropertyForKey:NSURLProtocolHandledKey inRequest:redirectRequest];
[[self client] URLProtocol:self wasRedirectedToRequest:redirectRequest redirectResponse:response];
[self.task cancel];
[[self client] URLProtocol:self didFailWithError:[NSError errorWithDomain:NSCocoaErrorDomain code:NSUserCancelledError userInfo:nil]];
}
- (void)URLSession:(NSURLSession *)session didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition, NSURLCredential * _Nullable))completionHandler
{
if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) {
NSURLCredential *card = [[NSURLCredential alloc] initWithTrust:challenge.protectionSpace.serverTrust];
completionHandler(NSURLSessionAuthChallengeUseCredential,card);
}
}
@end