An overview of the
Previously, UIWebView was mainly used for page loading, but UIWebView had many problems and was officially abandoned by Apple in 2020. So this article will focus on WKWebView. WKWebView has been supported since iOS8. Most apps don’t support iOS7 anymore.
UIWebView has two problems, one is high memory consumption, the other is poor performance. Compared with UIWebView, WKWebView performs much better than UIWebView, with a refresh rate of 60FPS. It also has a smaller memory footprint than UIWebView.
WKWebView is a multi-process component, Network, UI Render are completed in an independent process.
Because WKWebView and App are not in the same process, if the WKWebView process crashes, the application will not crash, only the page blank screen and other exceptions. Page loading, rendering and other operations that consume memory and performance are processed in the WKWebView process, and then the results are given to the App process for display, so the performance consumption of the App process will be much smaller.
Page loading process
- Request the server by domain name, the browser will make a request before
DNS
Parse, and willIP
The address is returned to the browser. - Browser usage
IP
The address requests the server and begins the handshake process.TCP
Three handshakes if usedhttps
It still needs to be doneTLS
After the handshake, select whether to keep the connection according to the protocol field. - After the handshake is complete, the browser sends a request to the server to obtain
html
File. - The server parses the request, and the
CDN
The server returns the corresponding resource file. - The browser received the return from the server
html
Document tohtml
The parser parses. - parsing
html
Parse from top to bottomxml
Tag, if encountered in the processcss
Or resource files, are asynchronously loaded, encounteredjs
Will suspend the currenthtml
Parse the task, requestjs
And return to continue parsing. becausejs
The file might be rightDOM
The tree is modified. - Parsing the
html
And executejs
Code, form the finalDOM
The tree. throughDOM
Cooperate withcss
The file finds the final presentation style of each node and sends it to the browser for rendering - End the link.
Proxy method
The proxy methods for WKWebView and UIWebView have changed a bit, and the WKWebView process has been more detailed. For example, when the UI finishes the request, it immediately renders to the webView. WKWebView, on the other hand, calls back a proxy method before rendering to the screen, which decides whether to render to the screen. This allows you to validate the requested data to prevent it from being changed or to verify that the view is allowed to be displayed on the screen.
In addition, WKWebView has more customization operations than UIWebView.
- The callback to the redirect can be retrieved when the redirect is requested.
- when
WKWebView
If the process exits unexpectedly, you can obtain the value by calling back. - Custom processing certificate.
- The deeper
UI
Custom operations willalert
Etc.UI
The operation is handled by the native plane, whileUI
planUIAlertView
Is a directwebView
Display.
WKUIDelegate
WKWebView leaves much of the UI display to the native layer, such as the display of pop-ups or input fields. In this way, if the project has a unified popover, you can call the custom popover directly, instead of just showing the system popover.
In WKWebView, the display of the pop-up window is controlled by the client. The client can use the following callback method to get the pop-up display information, and the client can call UIAlertController to display. Parameter contains a callback block for the completionHandler that the client must call or crash if it doesn’t.
- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void(^) (void))completionHandler;
Copy the code
Sometimes H5 asks the user for some input, such as username and password. The client can retrieve the input field event in the following way and display the input field by the client. When the user finishes typing, the result is called back to the completionHandler.
- (void)webView:(WKWebView *)webView runJavaScriptTextInputPanelWithPrompt:(NSString *)prompt defaultText:(nullable NSString *)defaultText initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void(^) (NSString * _Nullable result))completionHandler;
Copy the code
WKNavigationDelegate
The load process-related methods are abstracted into the WKNavigationDelegate. Here are some of the more common methods.
The following method, via the decisionHandler callback, returns an enumerated parameter indicating whether the page is allowed to load. You can determine the domain name. If it is an off-site domain name, you can prompt the user whether to jump to another domain name. If it is to jump to the URL of another App or store, it can jump through openURL and intercept the request. Cookie processing is also done in this method, and more on cookie processing later.
In addition, a lot of the logic processing before the page is displayed is also done in this method. However, do not do too much time-consuming processing in the method, which will affect the page load speed.
- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void(^) (WKNavigationActionPolicy))decisionHandler;
Copy the code
Start loading the page and request the server.
- (void)webView:(WKWebView *)webView didStartProvisionalNavigation:(null_unspecified WKNavigation *)navigation;
Copy the code
This method is called back when the page fails to load, including errors such as timeout. This page displays error pages, clears progress bars, resets network indicators, and more. Note that this method is also executed when goBack is called, and can be filtered out by determining whether NSURLErrorCancelled is cancelled based on the status of error.
- (void)webView:(WKWebView *)webView didFailProvisionalNavigation:(WKNavigation *)navigation withError:(NSError *)error;
Copy the code
When the page is loaded and rendered, this method will be called. When this method is called, the DOM of H5 has been parsed and rendered and displayed on the screen. So in this method, you can do some loading operations, such as removing the progress bar, resetting the network indicator, and so on.
- (void)webView:(WKWebView *)webView didFinishNavigation:(null_unspecified WKNavigation *)navigation;
Copy the code
WKUserContentController
The callback
WKWebView will interact with JS by the WKUserContentController class, which is collectively called userContent.
If you need to receive and process the js calls, by calling the addScriptMessageHandler: name: method, and the incoming object, an implementation of a WKScriptMessageHandler agreement can receive js callback, because userContent will strong references to objects, So you should create a new object, not self. When an object is registered, the following name is the name of the function called by js.
WKUserContentController *userContent = [[WKUserContentController alloc] init];
[userContent addScriptMessageHandler:[[WKWeakScriptMessageDelegate alloc] initWithDelegate:self] name:@"clientCallback"];
Copy the code
Dealloc should remove processing for the specified name by using the following method.
[userContent removeScriptMessageHandlerForName:@"clientCallback"];
Copy the code
H5 makes a call to the client using the postMessage function to pass a JSON string with the transfer character. After receiving the call, the client gets the body dictionary according to the WKScriptMessage object passed in by the callback method and parses the passed parameters.
window.webkit.messageHandlers.clientCallback.postMessage("{\"funName\":\"getMobileCode\",\"value\":\"srggshqisslfkj\"}");
Copy the code
call
The same is true for the native H5 call, creating a WKUserScript object and passing in the JS code as an argument. In addition to calling JS code, it is possible to change the page DOM by injecting code in this method, but this is not recommended because of the large amount of code.
WKUserScript *wkcookieScript = [[WKUserScript alloc] initWithSource:self.javaScriptString
injectionTime:WKUserScriptInjectionTimeAtDocumentStart
forMainFrameOnly:NO];
[webView.configuration.userContentController addUserScript:wkcookieScript];
Copy the code
WKUserScript vs evaluateJavaScript
WKWebView provides two ways for the js code, through the way of add a WKUserScript userContent object, and through the webView evaluateJavaScript: completionHandler: way, into the js code.
NSString *removeChildNode = @""
"var header = document.getElementsByTagName:('header')[0];"
"header.parentNote.removeChild(header);"
[self.webView evaluateJavaScript:removeChildNode completionHandler:nil];
Copy the code
First of all, it should be noted that both methods can inject JS code, but I did not go into the internal implementation of the WebKit kernel is open source, interested students can have a look. However, there are some functional differences between the two methods. You can choose the CORRESPONDING API according to the specific business scenario.
Say first evaluateJavaScript: completionHandler: way, this way is usually show in the page after the completion of the execution of operations, used to invoke the js function and obtain the return value is very convenient. Of course, it can also be used to inject a piece of JS code, but you need to control the injection timing.
WKUserScript can control the injection timing, you can choose to inject JS according to whether the document is finished loading. And whether the injected JS is valid on the current page or including its children. In contrast to the evaluateJavaScript: method, this method does not get the return value after js execution, so there is a functional difference between the two methods.
Container design
Design ideas
WKWebView is not used directly in a project, but is wrapped as a WKWebViewController to be used by the business layer. WebViewVC design should follow the simple and flexible idea to design, itself only provide display function, does not involve any business logic. The display navigation bar, title setting, progress bar and other functions can be assigned through WKWebViewConfiguration and passed in when the WKWebViewController is instantiated.
Callbacks for JS interaction, webView life cycle, load errors, etc., are provided to callers and handled externally by corresponding callbacks. These callbacks are optional and have no effect on webView loading if not implemented. The following is example code, which can also be split into different types of callback to define different proxies.
@protocol WKWebViewControllerDelegate <NSObject>
@optional
- (void)webViewDidStartLoad:(WKWebViewController *)webViewVC;
- (void)webViewDidFinishLoad:(WKWebViewController *)webViewVC;
- (void)webView:(WKWebViewController *)webViewVC didFailLoadWithError:(NSError *)error;
- (void)webview:(WKWebViewController *)webViewVC closeWeb:(NSString *)info;
- (void)webview:(WKWebViewController *)webViewVC login:(NSDictionary *)info;
- (void)webview:(WKWebViewController *)webViewVC jsCallbackParams:(NSDictionary *)params;
@end
Copy the code
In addition, WKWebViewController should be responsible for handling public parameters and can be extended based on public parameters. Here we define a method that can specify the location of the underlying argument, which is added through URL concatenation, header, JS injection, etc. The enumeration is multi-selective, meaning that it can be injected at multiple locations. In addition to basic parameters, additional custom parameters can be added, also to the specified location.
- (void)injectionParamsType:(SVParamsType)type additionalParams:(NSDictionary *)additionalParams;
Copy the code
Reuse pool
When WKWebView is initialized for the first time, it starts the webKit kernel and does some initialization, which is very performance intensive. Therefore, the first step in the reuse pool design is to initialize a global WKWebView at App startup.
Also, create two pools, create a visiblePool for the in-use pool, and create a reusablePool for the idle pool. Also, when the page exits, the page should be reclaimed to clean up the data on the page at the same time it is put into the reusablePool from the visiblePool.
When you need to initialize a webView container, take a container from the reusablePool and drop it into the visiblePool. The reuse pool implementation reduces the time from initializing a webView container to displaying the page.
WKProcessPool
The processPool property is defined in WKWebView to specify the corresponding processPool object. Each webView has its own content process, which defaults to a new content process if not specified. The content process contains local cookies, resources, and the like, which cannot be shared if it is not in a content process.
You can create a common WKProcessPool, which is a singleton object. All webViews are created using the same content process to achieve resource sharing.
UserAgent
User-agent is a request header field in THE HTTP protocol, used to inform the server of some information. User-agent contains many fields, such as system version, browser kernel version, and network environment. This field can be directly provided by the system, or other fields can be added based on the original User-Agent.
For example, the following user-agent is obtained from the webView of the system.
Mozilla/5.0 (iPhone; CPU iPhone OS 10_3_2 like Mac OS X) AppleWebKit/603.24. (KHTML, like Gecko) Mobile/14F89
Copy the code
After iOS9, customUserAgent property is provided, which directly sets user-Agent for WKWebView. Before iOS9, user-Agent needs to be injected into H5 by js writing.
Communication protocol
A well-designed WebView container should communicate well with each other and be flexible and extensible. The communication between H5 and the client is mainly described in the following scenarios.
js
Call the client, as welljs
Get the client after invoking the clientcallback
Callbacks and parameters.- Client call
js
, and the calljs
After thecallback
Callbacks and parameters. - The client actively notifies
H5
, some lifecycle changes for the client. For example, enter the lock screen and enter the foreground.
Take the JS call client as an example, there are two latitude calls. A module can be called directly through the URLRouter, which can be invoked following the URL definition of the client and supports parameter passing. You can also use the userContentController to make page-level calls, such as closing the webView and setting up the login function. In other words, you can use JS to call a certain function of the client, which requires the client to provide the corresponding processing code.
The two call each other, try to avoid high-frequency call, and generally there is no need for high-frequency call. However, if the same function is called frequently, you need to set an actionID to distinguish between different calls to ensure that the callback can be distinguished normally.
The callback method can also be passed as an argument, which is flexible and version-restricted if the write is fixed and older clients may not support the callback.
To deal with the callback
WebView callback in addition to the basic call, such as refresh refresh the current page, close close the current page, directly by the corresponding function class to deal with the call, the other time should be handed over to the outside world.
The design here is not that one event corresponds to one callback method, and then the outside world follows the proxy and implements multiple proxy methods. Instead, encapsulate each callback event into an object that can be directly called back to the outside world for more flexibility and more information. For the definition of the event model, see the following.
@interface WKWebViewCallbackModel : NSObject
@property(nonatomic.strong) WKWebViewController *webViewVC;
@property(nonatomic.strong) WKCallType *type;
@property(nonatomic.copy) NSDictionary *parameters;
@property(nonatomic.copy) NSString *callbackID;
@property(nonatomic.copy) NSString *callbackFunction;
@end
Copy the code
persistence
At present, the persistence scheme of H5 page is mainly WebKit’s localStorage and Cookie, but Cookie is not used for persistence operation, so it should not be used for persistence of H5. If you want more stable persistence, you can consider providing a CRUD interface of THE JS Bridge, so that H5 can be used to store and query data.
The persistence scheme adopts the same scheme as the client, and creates a separate data table for H5.
Caching mechanisms
Caching rules
Front-end browsers, including WKWebView, cache resources to ensure fast page opening and reduce user traffic consumption. This caching rule can also be specified in WKWebView, or we can choose not to use caching if we want to keep the resource files up to date every time, but we generally don’t do this.
NSURLRequestUseProtocolCachePolicy = 0
, default cache policy, andSafari
The kernel cache behaves the same.NSURLRequestReloadIgnoringLocalCacheData = 1,
Ignore the local cache and get the data directly from the server.NSURLRequestReturnCacheDataElseLoad = 2
If there is a local cache, use the cache. Otherwise, load server data. This strategy does not verify that the cache is expired.NSURLRequestReturnCacheDataDontLoad = 3
, only from the local, and does not determine the validity and change, local does not request server data, the request will fail.NSURLRequestReloadIgnoringLocalAndRemoteCacheData = 4
To get the latest data from the server, ignoring local and routing caches.NSURLRequestReloadRevalidatingCacheData = 5
, verifies whether the cache is available from the server, and requests server data if it is not available locally.NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData
.
According to Apple’s default cache policy, there are three checks.
- Whether the cache exists.
- Verify that the cache is expired.
- Whether the cache has changed.
Cache file
IOS9 apple provides the cache management class WKWebsiteDataStore, through which you can query and delete cache files of a specified type on disks. This API is highly recommended for managing local caches, as well as cookies, since many apps are supported from iOS9 onwards. The local file cache types are cookie, diskCache, and memoryCache.
WKWebsiteDataTypeFetchCache
, disk cache, according to the source can be seen, type isDOMCache
WKWebsiteDataTypeDiskCache
, local disk caching, andfetchCache
The implementation is different for all cached dataWKWebsiteDataTypeMemoryCache
, local memory cacheWKWebsiteDataTypeOfflineWebApplicationCache
, offlineweb
Application cacheWKWebsiteDataTypeCookies
.cookie
The cacheWKWebsiteDataTypeSessionStorage
.html
The session storageWKWebsiteDataTypeLocalStorage
.html
Local data cacheWKWebsiteDataTypeWebSQLDatabases
.WebSQL
Database dataWKWebsiteDataTypeIndexedDBDatabases
, database indexWKWebsiteDataTypeServiceWorkerRegistrations
, server registration data
The following method is used to retrieve all of the local cache file types and return the collection string, which is the type defined above.
+ (NSSet<NSString *> *)allWebsiteDataTypes;
Copy the code
You can delete a specified type of data in a specified period of time. After deletion, the block is called back.
- (void)removeDataOfTypes:(NSSet<NSString *> *)dataTypes modifiedSince:(NSDate *)date completionHandler:(void(^) (void))completionHandler;
Copy the code
The system also provides a more customized method to get all WKWebsiteDataRecord objects of a specified type using the fetchDataRecordsOfTypes: method. This object contains the domain name and type parameters. You can determine by domain name and type, and then call removeDataOfTypes: to pass in the object to be deleted to delete the data under the specified domain name.
/ / to get
- (void)fetchDataRecordsOfTypes:(NSSet<NSString *> *)dataTypes completionHandler:(void(^) (NSArray<WKWebsiteDataRecord *> *))completionHandler;
/ / delete
- (void)removeDataOfTypes:(NSSet<NSString *> *)dataTypes forDataRecords:(NSArray<WKWebsiteDataRecord *> *)dataRecords completionHandler:(void(^) (void))completionHandler;
Copy the code
HTTP Cache Policy
When the client and H5 are dealing with, there will often be a page cache problem, H5 developers often say “you try to clear the cache”, in fact, the cause of this problem is H5 cache management strategy has a problem. Here is the CACHE management strategy of H5.
H5 Cache management is actually using HTTP protocol field management, the more commonly used is cache-Control and last-Modified with the way.
Cache-Control
: Indicates the validity period of the file cache, such as the return of the server response header after a file is requestedCache-Control:max-age=600
Is the file validity period600
Seconds. Therefore, network requests will not be sent during the validity period of this file until it expires.Last-Modified
: Returned in the server response header after a file is requested, indicating the last update time of the file. ifCache-Control
After expiration, the server is requested and this time is placed in the request headerIf-Modified-Since
Field, when the server receives the request, it compares the time and returns if the time has not changed304
, otherwise returns the new file and response header fields, and returns200
.
Cache-control is an http1.1 extension that specifies the relative lifetime of the file. Previously, there is an Expires field that specifies the absolute lifetime of the file, such as Expires: Thu, 10 Nov 2015 08:45:11 GMT.
Last-modified has a similar field, Etag, except that last-Modified compares time and Etag compares file hashes. When the file validity period expires, the request server adds the Etag value to the if-none-match field in the request header and sends it to the server for comparison.
Cookie handling
As we all know, HTTP protocol supports cookie setting. The server can Set cookies to the browser through set-cookie: field, and can also specify expiration time, domain name, etc. These are suitable for Chrome, but if displayed in the client, the client needs to pass some parameters, so that H5 can obtain the login status.
Although Apple provides some Cookie management apis, there are still a lot of holes in the use of WKWebView. Finally, I will give a more general solution.
WKWebView cookies design
When WE used UIWebView, it read the same area as the traditional cookie management class NSHTTPCookieStorage, or the cookies of UIWebView were managed in this way. However, the cookie design of WKWebView is different, and the cookie of App is not stored in the same memory area, so they need to be processed separately.
There is also synchronization between WKWebView’s cookie and NSHTTPCookieStorage, but this synchronization has a significant delay and the rules are not easy to understand. So for the sake of code stability, it’s better to handle cookies yourself.
WK and APP are two processes, and there are two cookies, but the cookie of WK is in the sandbox of app. There is a timed synchronization, but there is no specific rule, so it is best not to rely on synchronization. There are only two times when WK cookie changes, one is when JS executes the code setCookie, the other is when Response returns cookie.
WKWebsiteDataStore
Cookie management has always been a drawback of WKWebView, which is very inconvenient to handle cookies. In iOS9, cookies can be managed by WKWebsiteDataStore, but it is not intuitive to use. DataType is needed to filter and delete cookies. In addition, WKWebsiteDataStore itself does not have the function of adding cookies, so it can only delete cookies instead of adding cookies.
if (@available(iOS 9.0, *)) {
NSSet *cookieTypeSet = [NSSet setWithObject:WKWebsiteDataTypeCookies];
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:cookieTypeSet modifiedSince:[NSDate dateWithTimeIntervalSince1970:0] completionHandler:^{
}];
}
Copy the code
WKHTTPCookieStore
In iOS11 apple on the basis of WKWebsiteDataStore, for it added WKHTTPCookieStore class dedicated to cookie processing, and support to add, delete, query three operations, can also register an observer to monitor cookie changes, The listener is notified by a callback when the cookie changes.
WKWebsiteDataStore can retrieve cookies written by H5 pages in document.cookie mode, and cookies written by the server in set-cookie mode, so it is recommended to use this class to manage cookies, but it only supports iOS11.
Here is the code to add cookies to WKWebView.
NSMutableDictionary *params = [NSMutableDictionary dictionary];
[params setObject:@"password" forKey:NSHTTPCookieName];
[params setObject:@"e10adc3949ba5" forKey:NSHTTPCookieValue];
[params setObject:@"www.google.com" forKey:NSHTTPCookieDomain];
[params setObject:@ "/" forKey:NSHTTPCookiePath];
[params setValue:[NSDate dateWithTimeIntervalSinceNow:60*60*72] forKey:NSHTTPCookieExpires];
NSHTTPCookie *cookie = [NSHTTPCookie cookieWithProperties:params];
[self.cookieWebview.configuration.websiteDataStore.httpCookieStore setCookie:cookie completionHandler:nil];
Copy the code
Our Company plan
The best way to handle cookies is through WKHTTPCookieStore, but it only supports iOS11 and above, so this is not an option for now. The second is WKWebsiteDataStore, but it can only be used as a delete cookie, not to manage cookies.
Our company’s scheme is to manage the cookies of webView by WKUserContentController launched by iOS8, and manage the cookies of network requests by NSHTTPCookieStorage, such as the requests issued by H5. By default, the requests sent by NSURLSession and NSURLConnection will carry cookies in NSHTTPCookieStorage. The internal requests of H5 will also be handed to NSURLSession by the system.
At the code implementation level, you listen for the didFinishLaunching notification, asking for user information from the server (or from the local) when your program starts. The data is delivered in the form of key and value, spliced in accordance with the form of key=value, and assembled into JS code for setting cookies through document.cookie. All codes are spliced into a string separated by semicolons, which will be executed through this string when the webView is later given cookies.
For cookies requested by the network, cookies directly planted under the root domain name through NSHTTPCookieStorage can take effect for all sub-domain names under the root domain name. The processing here is relatively simple.
SVREQUEST.type(SVRequestTypePost).parameters(params).success(^(NSDictionary *cookieDict) {
self.cookieData = [cookieDict as:[NSDictionary class]];
[self addCookieWithDict:cookieDict forHost:@".google.com"];
[self addCookieWithDict:cookieDict forHost:@".google.cn"];
[self addCookieWithDict:cookieDict forHost:@".google.jp"];
NSMutableString *scriptString = [NSMutableString string];
for (NSString *key in self.cookieData.allKeys) {
NSString *cookieString = [NSString stringWithFormat:@ "% @ = % @", key, cookieDict[key]];
[scriptString appendString:[NSString stringWithFormat:@"document.cookie = '%@; expires=Fri, 31 Dec 9999 23:59:59 GMT; ';", cookieString]];
}
self.webviewCookie = scriptString;
}).startRequest();
- (void)addCookieWithDict:(NSDictionary *)dict forHost:(NSString *)host {
[dict enumerateKeysAndObjectsUsingBlock:^(NSString * _Nonnull key, NSString * _Nonnull value, BOOL * _Nonnull stop) {
NSMutableDictionary *properties = [NSMutableDictionary dictionary];
[properties setObject:key forKey:NSHTTPCookieName];
[properties setObject:value forKey:NSHTTPCookieValue];
[properties setObject:host forKey:NSHTTPCookieDomain];
[properties setObject:@ "/" forKey:NSHTTPCookiePath];
[properties setValue:[NSDate dateWithTimeIntervalSinceNow:60*60*72] forKey:NSHTTPCookieExpires];
NSHTTPCookie *cookie = [NSHTTPCookie cookieWithProperties:properties];
[[NSHTTPCookieStorage sharedHTTPCookieStorage] setCookie:cookie];
}];
}
Copy the code
The webView cookie is written to js by WKUserContentController, which is the js string concatenated above. But one problem with this class is that it can’t persist cookies, that is, cookies follow the declaration cycle of the userContentController, and if you exit the App, the cookie disappears, and you have to plant it again the next time you enter the App, and that’s a big problem.
So our approach is in decidePolicyForNavigationAction: callback method to add the following code, the code will determine whether this domain in the kind of cookies, if you are not the kind of cookies. For cookie handling, I created a new cookieWebview specifically to deal with cookies. After executing addUserScript, loadHTMLString:baseURL: loads an empty local HTML and sets the domain name to the domain name of the current page to be displayed. This makes the cookie valid for all webViews in the current processPool.
In this scheme, all cookies are executed synchronously and have little impact on webView. After my test, it only takes 28ms to add a cookie on average. It is imperceptive from the user’s point of view, with no page stalling or refreshing.
- (void)setCookieWithUrl:(NSURL *)url {
NSString *host = [url host];
if ([self.cookieURLs containsObject:host]) {
return;
}
[self.cookieURLs addObject:host];
WKUserScript *wkcookieScript = [[WKUserScript alloc] initWithSource:self.webviewCookie
injectionTime:WKUserScriptInjectionTimeAtDocumentStart
forMainFrameOnly:NO];
[self.cookieWebview.configuration.userContentController addUserScript:wkcookieScript];
NSString *baseWebUrl = [NSString stringWithFormat:@ % @ : / / "% @", url.scheme, url.host];
[self.cookieWebview loadHTMLString:@ "" baseURL:[NSURL URLWithString:baseWebUrl]];
}
Copy the code
The processing of cookie deletion is relatively simple. NSHTTPCookieStorage traverses the NSHTTPCookie it needs to delete through the cookies attribute, and then calls the method to delete it. WebView delete method is simple and crude, directly call removeAllUserScripts delete all WKUserScript.
- (void)removeWKWebviewCookie {
self.webviewCookie = nil;
[self.cookieWebview.configuration.userContentController removeAllUserScripts];
NSMutableArray<NSHTTPCookie *> *cookies = [NSMutableArray array];
[[NSHTTPCookieStorage sharedHTTPCookieStorage].cookies enumerateObjectsUsingBlock:^(NSHTTPCookie * _Nonnull cookie, NSUInteger idx, BOOL * _Nonnull stop) {
if ([self.cookieData.allKeys containsObject:cookie.name]) { [cookies addObjectOrNil:cookie]; }}]; [cookies enumerateObjectsUsingBlock:^(NSHTTPCookie * _Nonnull cookie, NSUInteger idx, BOOL * _Nonnull stop) {
[[NSHTTPCookieStorage sharedHTTPCookieStorage] deleteCookie:cookie];
}];
}
Copy the code
Bad problems
If WKWebView loads a page that occupies too much memory, the WebContent Process crashes and the page displays a blank screen, or the blank screen may be caused by excessive memory usage by other system processes. The following two solutions can be used to solve the blank screen caused by low memory.
In iOS9 apple introduced the following API to call back when the WebContent process exits unexpectedly. The corresponding processing can be done in this API, such as displaying an exception page.
- (void)webViewWebContentProcessDidTerminate:(WKWebView *)webView;
Copy the code
If returning from another App causes a blank screen, you can check whether webView.title is empty when the view is about to be displayed. If it is empty, the exception page is displayed.