iOS-IOS 與 JS 的相互調用

IOS開發中Objective-C與JS互相調用實例教程
本文我們來講講iOS系統上通用的本地Objective-C代碼與Javascript互操作的基本方法,本文還涉及到html5的應用,這里把實現方法簡單介紹。
項目中要用到html5來實現,涉及到Objective-C調用JS,以及JS調用Objective-C的方法,這里把遇到的問題以及實現方法介紹一下。

Objective-C

    //  
    //  ViewController.h  
    //  OC_And_JS  
    //  
    //  Created by 張杰 on 15/7/9.  
    //  Copyright ? 2015年 張杰. All rights reserved.  
    //  
       
    #import <UIKit/UIKit.h>  
       
    @interface ViewController : UIViewController <UIWebViewDelegate>  
       
    @property (weak, nonatomic) IBOutlet UIButton *oc_call_js_no_params;  
    @property (weak, nonatomic) IBOutlet UIButton *oc_call_js_has_params;  
    @property (weak, nonatomic) IBOutlet UIWebView *mWebView;  
    @property (weak, nonatomic) IBOutlet UILabel *js_call_oc_show;  
       
    - (IBAction)ocCallJsNoParams:(id)sender;  
    - (IBAction)ocCallJsHasParams:(id)sender;  
       
       
    @end

Objective-C

    //  
    //  ViewController.m  
    //  OC_And_JS  
    //  
    //  Created by 張杰 on 15/7/9.  
    //  Copyright ? 2015年 張杰. All rights reserved.  
    //  
       
    #import "ViewController.h"  
       
    @interface ViewController ()  
       
    @end  
       
    @implementation ViewController  
       
    - (void)viewDidLoad {  
        [super viewDidLoad];  
        _mWebView.delegate = self;  
           
        //打開URL  
        NSString *path = [[NSBundle mainBundle] pathForResource:@"index" ofType:@"html"];  
        [self.mWebView loadRequest:[NSURLRequest <a href="/tags.php/request/" target="_blank">request</a>WithURL:[NSURL fileURLWithPath: path]]];  
    }  
       
    - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{  
        NSString *urlstr = request.URL.absoluteString;  
        NSRange range = [urlstr rangeOfString:@"ios://jwzhangjie"];  
        if (range.length!=0) {  
            _js_call_oc_show.text = [NSString stringWithFormat:@"請訪問地址:%@", urlstr];  
        }  
        return YES;  
    }  
       
    -(void)webView:(nonnull UIWebView *)webView didFailLoadWithError:(nullable NSError *)error{  
        NSLog(@"加載失敗");  
    }  
       
    -(void)webViewDidStartLoad:(nonnull UIWebView *)webView{  
        NSLog(@"開始加載");  
    }  
       
       
    -(void)webViewDidFinishLoad:(nonnull UIWebView *)webView{  
        NSLog(@"開始結束");  
    //    對于調用js的時候最好這個方法里面或者之后  
    }  
       
       
    - (void)didReceiveMemoryWarning {  
        [super didReceiveMemoryWarning];  
        // Dispose of any resources that can be recreated.  
    }  
       
       
       
    - (IBAction)ocCallJsNoParams:(id)sender {  
        NSString *js = [NSString stringWithFormat:@"ocCallJsNoParamsFunction();"];  
        [self.mWebView stringByEvaluatingJavaScriptFromString:js];  
    }  
       
    - (IBAction)ocCallJsHasParams:(id)sender {  
        NSString *js = [NSString stringWithFormat:@"ocCallJsHasParamsFunction('%@','%@');",@"jwzhangjie",@"http://jwzhangjie.cn"];  
        [self.mWebView stringByEvaluatingJavaScriptFromString:js];  
    }  
    @end

JavasSctipt

    function ocCallJsNoParamsFunction()  
    {  
        alert("OC調用JS中的無參方法");  
        var e = document.getElementById("js_shouw_text");  
        e.options.add(new Option("OC調用JS中的無參方法", 2));  
    }  
       
    function ocCallJsHasParamsFunction(name, url)  
    {  
        alert(name+"的博客地址為:"+url);  
        var e = document.getElementById("js_shouw_text");  
        e.options.add(new Option("OC調用JS中的有參方法", 2));  
    }

html

    <!DOCTYPE html>  
    <html>  
    <head>  
        <meta charset="utf-8">  
        <title>OC與JS互相調用</title>  
    </head>  
    <body>  
        <div >  
            <<a href="/tags.php/select/" target="_blank">select</a> id="js_shouw_text">  
                <option>  
                    展示OC調用JS無參數  
                </option>  
            </select>  
        </div>  
        <div>                                                                     
            <BR/>  
            <input type="button" value="JS調用OC方法" onclick="js_call_oc()"/>  
        </div>  
        <!--  這里要清楚,雖然test.js跟index.html不同及目錄,實際安裝到程序里面后,是在同級目錄的,所以這里src不能加目錄,同樣css也是一樣的  -->  
        <script type="text/<a href="/js_a/js.html" target="_blank">javascript</a>" src="test.js" charset="UTF-8"></script>  
        <script type="text/javascript">  
            function js_call_oc()  
            {  
                var iFrame;  
                iFrame = document.createElement("<a href="/tags.php/iframe/" target="_blank">iframe</a>");  
                iFrame.setAttribute("src", "ios://jwzhangjie");  
                iFrame.setAttribute("style", "display:none;");  
                iFrame.setAttribute("height", "0px");  
                iFrame.setAttribute("width", "0px");  
                iFrame.setAttribute("frameborder", "0");  
                document.body.appendChild(iFrame);  
                // 發起請求后這個iFrame就沒用了,所以把它從dom上移除掉  
                iFrame.parentNode.removeChild(iFrame);  
                iFrame = null;  
            }  
               
        </script>  
    </body>  
       
    </html>

規避1:對于OC去調用JS內容最好在webViewDidFinishLoad方法里或者之后

規避2:在html里面引用js或者css的時候src不要帶有路徑,因為安裝后文件都在同級目錄下面

規避3:OC調用JS的規范

NSString *js = [NSString stringWithFormat:@"ocCallJsHasParamsFunction('%@','%@');",@"jwzhangjie",@"http://jwzhangjie.cn"];  
   [self.mWebView stringByEvaluatingJavaScriptFromString:js];  

規避4:JS調用OC,這里通過html里面發送一個請求,然后在ios中使用shouldStartLoadWithRequest攔截請求,根據請求url的不同進行處理。

javascript

    function js_call_oc()  
           {  
               var iFrame;  
               iFrame = document.createElement("iframe");  
               iFrame.setAttribute("src", "ios://jwzhangjie");  
               iFrame.setAttribute("style", "display:none;");  
               iFrame.setAttribute("height", "0px");  
               iFrame.setAttribute("width", "0px");  
               iFrame.setAttribute("frameborder", "0");  
               document.body.appendChild(iFrame);  
               // 發起請求后這個iFrame就沒用了,所以把它從dom上移除掉  
               iFrame.parentNode.removeChild(iFrame);  
               iFrame = null;  
           }
 
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{  
        NSString *urlstr = request.URL.absoluteString;  
        NSRange range = [urlstr rangeOfString:@"ios://jwzhangjie"];  
        if (range.length!=0) {  
            _js_call_oc_show.text = [NSString stringWithFormat:@"請訪問地址:%@", urlstr];  
        }  
        return YES;  
    }

javascript與Objective-C的相互調用

  1.   Objective-C調用Javascript方法:
    

前提:

UIWebView已經完全加載完成包含需要調用的頁面(注:在UIWebViewDelegate托管中通過監視- (void)webViewDidFinishLoad:(UIWebView *)webView調用來確定頁面是否加載完成)。

方法:

假設某視圖對象的子視圖屬性self.webview加載的頁面包含如下Javascript函數:

function getString(){ return “Hello javascript!”;}

并且該函數在該頁面上可被正常調用,則可以通過形如下面的Objectvie-C方法調用此函數:

NSString *str = [self.webview stringByEvaluatingJavaScriptFromString:@"getString();"];

該函數調用實際模擬了頁面上的一次Javascript函數調用,因此在該函數內部任何有效的Javascript代碼都可被執行!該Objective-C代碼返回值為被調用的Javascript代碼的返回值,在上例中,str的值將被賦為@”Hello javascript!”。

被調用的函數可以帶有字符串或數值型的參數。若調用的Javascript函數名稱帶有參數,需保證傳進去的參數的格式正確性,特別是字符串參數的引號很容易被忽略。

  1.   系統響應頁面Javascript代碼:
    

前提:

   需要實現UIWebViewDelegate的
  • (BOOL)webView:(UIWebView *)webView

shouldStartLoadWithRequest:(NSURLRequest *)request

navigationType:(UIWebViewNavigationType)navigationType;

托管函數(以下簡稱跳轉監視函數)并將實現了該托管函數的對象設置為UIWebView的delegate。

方法:

   UIWebView的當前頁面將要被Javascript代碼通過以下方式轉向的時候,會調用上述跳轉監視函數:

   window.location.href=”http://www.strongsoft.net”;

此時實現了該托管協議的對象的跳轉監視函數會被調用,通過如下代碼監視瀏覽器跳轉的地址:

   NSString *url = [[request URL] absoluteString];

若該托管函數返回值為NO,則頁面UIWebView的頁面跳轉將被否決。利用這一思路,監視頁面上的特定格式的跳轉地址加以攔截,并執行相應的本地代碼,即可實現Javascript與Objective-C代碼的交互。

   例如,規定攔截URL地址前綴為”objc:”的所有地址,并且用if…else…語句判定本地需要執行的代碼,則實現形如以下格式的Objective-C托管函數:
- (BOOL)webView:(UIWebView *)webView

shouldStartLoadWithRequest:(NSURLRequest *)request

navigationType:(UIWebViewNavigationType)navigationType

{

NSString *urlString = [[request URL] absoluteString];

       NSArray *urlComps = [urlString componentsSeparatedByString:@":"];

       if([urlComps count] && [[urlComps objectAtIndex:o] isEqualToString:@"objc"])

       {

              NSString *funcStr = [urlComps objectAtIndex:1];

              if([funcStr isEqualToString:@"doFunc1"])

              {

                     

              }

              else if([funcStr isEqualToString:@"doFunc2"])

              {

                     

              }

              return NO;

       }

       return YES;

}

當需要調用本地函數1時,可以通過如下Javascript函數進行:

   window.location.href=”objc:doFunc1”;

根據上述思路,可以完善通過Javascript調用本地帶參數的Objective-C代碼。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,237評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,957評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,248評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,356評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,081評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,485評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,534評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,720評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,263評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,025評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,204評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,787評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,461評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,874評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,105評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,945評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,205評論 2 375

推薦閱讀更多精彩內容