WebView和JS交互

前言

在开发IOS应用的过程中,难免的会遇到和WebView打交道的场景,通常为了实现产品经理的功能需求还要去和WebView里面的JS进行交互。作为一个刚IOS开发的新人来说,第一次肯定会遇到各种问题和各种坑。在这里我把我的问题和解决方法罗列出来,希望对其他的大胸弟们有所帮助。

概述

IOS提供给我们的WebView控件类型总共有3中,UIWebViewWkWebViewSFSafariView。这三种类型的控件有这不同的使用限制和要求,在实际项目中,我们需要根据项目的实际要求选择其中的一个就可以了。接下来,我会介绍一下这三种类型的区别。

UIWebView

UIWebView作为IOS应用提供给开发者最早的一个控件,最初还是一个很不错的控件,但是基于今天的现状而言,却显得有些鸡肋了。加载速度慢,内存开销大对于产品和开发者来说都是很大的问题,特别是在第一次加载网页时,经常会出现加载10秒以上的情况。

UIWebView加载过程

UIWebView加载网页主要会用到一下四个方法

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

当WebView收到一个打开链接请求时,会触发此方法,向我们询问是否要加载这个链接请求,返回YES就表示允许加载这个链接请求,否则不加载链接请求。

- (void) webViewDidStartLoad:(UIWebView *)webView

当上一步允许加载链接请求后,WebView会触发此方法,告知我们要开始加载了,这时我们可以加一个loading的效果提示用户。

- (void) webViewDidFinishLoad:(UIWebView *)webView

当整个网页加载完成之后会触发此方法,如果上一步加上loading效果的话,在这个就要去掉了

- (void) webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error

当加载过程中出错,比如无网络、404什么的,会执行这个方法。这里我们要把loading去掉,如果为了友好的话,可以展示一个加载失败的界面。

UIWebView和JS交互

UIWebView和JS交互主要依托于一个JSExport的协议,具体来说就是在加载网页的过程中,我们去拿到网页运行JS的环境,然后JS执行某些方法时我们就可以捕获到,然后执行自己的逻辑。

注意:和JS的交互过程中需要分两种情况,一种是捕获JS中无参数或一种参数的方法,另一种是捕获JS中大于等于2个参数的方法。因为OC中的方法名是由传统意义上的方法名+外部参数名构成的,所以当我们捕获第二种情况的JS方法时需要注意和OC方法名的对应关系。下面的示例中会有介绍。

OK,接下来给大家放出一段示例代码,示例中会讲解到UIWebView和加载一个网页的流程,以及网页的JS如何和原生的代码交互。

ViewController.h

//
//  ViewController.h
//  webviewDemo
//
//  Created by 孙天文 on 16/11/15.
//  Copyright © 2016年 孙天文. All rights reserved.
//

#import <UIKit/UIKit.h>
#import <JavaScriptCore/JavaScriptCore.h>

//定义一个和JS交互的协议,用来处理JS中的方法,这个协议中列出了4个方法,分别是无参数、一个参数、两个参数以及三个参数的方法
@protocol demoJsResponseProtocol <JSExport>

- (void) sayHello;

- (void) setTitle:(NSString *)title;

//下面是大于等于两个参数的情况,要注意这个格式和JS中的格式对比
- (void) setTitle:(NSString *)title Left:(NSString *)leftTitle;

- (void) setTitle:(NSString *)title Left:(NSString *)leftTitle Right:(NSString *)rightTitle;

@end

@interface ViewController : UIViewController<demoJsResponseProtocol>

@end

ViewController.m

//
//  ViewController.m
//  webviewDemo
//
//  Created by 孙天文 on 16/11/15.
//  Copyright © 2016年 孙天文. All rights reserved.
//

#import "ViewController.h"

@interface ViewController ()<UIWebViewDelegate,JSExport>
    
@property (strong,nonatomic) UIWebView *webView;
 
 //JS运行环境   
@property (strong,nonatomic) JSContext *jsContext;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.webView = [[UIWebView alloc] initWithFrame:CGRectMake(0, 64, [UIScreen mainScreen].bounds.size.width, [UIScreen mainScreen].bounds.size.height)];
    self.webView.delegate = self;
    
    //加载本地html文件
    NSString *path = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"html"];
    NSString *htmlString = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];
    NSString *basePath = [[NSBundle mainBundle] bundlePath];
    NSURL *baseURL = [NSURL fileURLWithPath:basePath];
    
    [self.webView loadHTMLString:htmlString baseURL:baseURL];
    
    [self.view addSubview:self.webView];
}
    
//网页加载前调用的方法
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{
    
    return YES;
}
    
//开始加载
- (void) webViewDidStartLoad:(UIWebView *)webView{
    NSLog(@"start laod");
}
    
//加载完成
- (void) webViewDidFinishLoad:(UIWebView *)webView{
    self.jsContext = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    self.jsContext[@"demoapi"] = self;
    NSLog(@"laod finish");
}

//加载出错
- (void) webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error{
    NSLog(@"load error %@",error);
}
    
//执行JS方法
- (void) doJavaScriptFunction:(NSString *)jsfunction{
    [self.jsContext evaluateScript:jsfunction];
    
}

- (void) sayHello{
    [self showAlertDesc:@"hello world"];
}
    
- (void) setTitle:(NSString *)title{
    [self showAlertDesc:@"setTitle"];
}
    
//下面是大于等于两个参数的情况,要注意这个格式和JS中的格式对比    
- (void) setTitle:(NSString *)title Left:(NSString *)leftTitle{
    [self showAlertDesc:@"setTitleLeft"];
}
    
- (void) setTitle:(NSString *)title Left:(NSString *)leftTitle Right:(NSString *)rightTitle{
    [self showAlertDesc:@"setTitleLeftRight"];
}
    
- (void) showAlertDesc:(NSString *)desc{
    UIAlertController *alertController  = [UIAlertController alertControllerWithTitle:@"" message:desc preferredStyle:UIAlertControllerStyleAlert];
    
    UIAlertAction *okAction = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:nil];
    [alertController addAction:okAction];
    [self presentViewController:alertController animated:YES completion:nil];
}
    
- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

test.html

<!DOCTYPE html>
<html>
    <head>
        <style>
            .button{
                background-color: #d9edf7!important;
                display:block;
                margin-top:40px;
                height:30px;
            }
        </style>
    </head>
    <body>
        <a href = "#" class = "button" onclick = "sayHello()">Say Hello</a>
        <a href = "#" class = "button" onclick = "setTitle()">Set Title</a>
        <a href = "#" class = "button" onclick = "setTitle2()">Set Title Left</a>
        <a href = "#" class = "button" onclick = "setTitle3()">Set Title Left Right</a>
    </body>
    
    <script type="text/javascript">
        function sayHello(){
            demoapi.sayHello();
        }
    
        function setTitle(){
            demoapi.setTitle("Title1");
        }
    
        //下面是大于等于两个参数的情况,要注意这个格式和JS中的格式对比
        function setTitle2(){
            demoapi.setTitleLeft("Title2","LeftTitle2");
        }
    
        function setTitle3(){
            demoapi.setTitleLeftRight("Title3","LeftTitle3","RightTitle3");
        }
    </script>
</html>

WKWebView

WKWebView作为苹果官方推荐的Web控件,在UIWebView的基础上进行重构,加载速度和内存开销上都有很大的提升,当我们需要集成该控件时,需要去注意改控件提供和方法和UIWebView的不同。需要注意一点,这个控件要求系统版本最低是IOS8,如果你的项目需要覆盖IOS8一下的用户的话,还是乖乖的去用UIWebView吧。

详细情况后续补充

SFSafariView

SFSafariView给人直观的印象是把Safari内嵌到了APP当中,目前笔者还没有接触过,这里就不在叙述更多的信息了,如果后面结果的话,会再写一篇博客进行详细说明。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,716评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,558评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,431评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,127评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,511评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,692评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,915评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,664评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,412评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,616评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,105评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,424评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,098评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,096评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,869评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,748评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,641评论 2 271

推荐阅读更多精彩内容

  • Webview和JS交互方式 前言 现在很多App里都内置了Web网页(Hybrid App),比如说很多电商平台...
    badcyc阅读 263评论 1 2
  • 一、WebView WebView就是一个内嵌浏览器控件,在iOS中主要有两种WebView:UIWebView和...
    iOS祎阅读 1,039评论 0 2
  • 前言 关于UIWebView的介绍,相信看过上文的小伙伴们,已经大概清楚了吧,如果有问题,欢迎提问。 本文是本系列...
    CoderLF阅读 8,856评论 2 12
  • 有时我们有这样的需求:需要从app的网页中点击调用软件内部的页面,又或者是软件内部调用网页的js方法。 js代码调...
    鸡蛋掉了阅读 564评论 2 8
  • 移动应用中许多复杂的且经常改动的页面会使用H5进行代替native,这里就会使用到js和webview的交互 iO...
    黄成阅读 11,026评论 0 30