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
リクエストと一緒にカスタムヘッダーを送信するにはどうすればよいですか?
どうもありがとう!
これが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;
Thomasによる回答は、複数のiFrameを含む(ほとんどの)Webページでは機能しません。このソリューションは、完全なUIWebViewを介してiFrameリクエストを読み込みます。例えば。 GoogleadvtのloadRequestを呼び出す場合。 (これはいくつかの小さなiFrameにあります)advt。 UIWebView全体にロードされ、他には何もロードされません。
別の方法を見つけました。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 *です
カスタムNSURLプロトコルコード内:
@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
これは、NSURLProticolを使用した完全な実装です。このコードは好きな場所に置くことができ(つまり、独自のコードにすることも、既存のソースファイルに追加することもできます)、機能するはずです。カスタマイズする2つの主要な方法は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