本文介绍: 本文字数:13900字预计阅读时间:35分钟iOS APP添加桌面快捷方式背景新接到一个需求需要APP内的某些功能能够把入口添加到桌面点击桌面到入口可以直接跳转APP对应界面(类似于下面这张示例图),于是就做了一番调研其实很多APP目前都已经实现了类似的功能比如支付宝、云闪付等等,其中的每一个独立功能可以单独添加到桌面,所以网上很多实现方法,笔者做的是整理和试错。实现首先,添…

e4fd443920b497dbc86f470ea19353ab.jpeg

 0acffd46c00676fa04b547e2b401fc0d.gif 

本文字数13900

预计阅读时间35 分钟

iOS APP添加桌面快捷方式

背景

新接到一个需求需要APP内的某些功能,能够把入口添加到桌面点击桌面到入口可以直接跳转APP对应界面(类似于下面这张示例图),于是就做了一番调研

其实很多APP目前都已经实现了类似的功能,比如支付宝、云闪付等等,其中的每一个独立功能都可以单独添加到桌面,所以网上很多实现方法,笔者做的是整理和试错。

f80622eb74b5759d9da883b7be752c4e.png

实现

首先,添加到桌面功能的操作流程是:

客户端打开APP -> 进入对应到APP功能模块 -> 点击添加快捷方式到桌面按钮 -> 跳转浏览器,并加载引导页面点击分享选择添加到主屏幕 -> 从主屏幕点击刚刚添加到快捷功能,跳转到APP的对应界面

ae44f31ca554c03fe36be1634dd5b638.png

iOS开发内部功能生成桌面快捷方式.png

根据笔者了解到的信息,目前实现这种功能,大致可以分为两种实现方式

准备

一步 客户端:iOS 打开已有Xcode项目选中Target,添加URL Scheme这个URL Scheme是自己定义的,在这个地方定义xxx之后,可以通过浏览器输入xxx://来唤起APP,比如笔者定义一个mkaddtohomescreen然后浏览器输入mkaddtohomescreen://,就会弹出是否打开对应APP的提示

a539e10256ad8a1084239f697bdd4d49.png

59640204157d4eaf2bbe69f8dc0e3652.png


定义好了Scheme之后,可以考虑Scheme添加参数问题通过scheme后添加参数,在Appdelegate
applicaiton:open:options:
方法拦截到,根据对应参数跳转不同界面

比如Scheme为mkaddtohomescreen://page/view1,在applicaiton:open:options:中,url.absouluteString = mkaddtohomescreen://page/view1,url.host = pageurl.path = /view1,所以可以根据path不同跳转不同界面

func application(_ app: UIApplicationopen url: URL, options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool {
        if let navController = window?.rootViewController as? UINavigationController,
            let topController = navController.topViewController {
            // egmkaddtohomescreen://page/view1
            // url.host = page
            // url.path = /view1
            if url.absoluteString.hasPrefix("mkaddtohomescreen://") {
                // 说明是APP的URL Scheme,处理
                
                let targetVC = targetViewController(from: url.path)
                if targetVC != nil {
                    // 判断当前显示界面是否是要跳转的界面
                    if topController.classForCoder == targetVC?.classForCoder {
                        return true
                    }
                    
                    navController.pushViewController(targetVC!, animatedtrue)
                }
                else {
                    return true
                }
            }
        }
        return true
    }

    // 根据URL path返回要跳转的界面
    func targetViewController(fropath: String) -> UIViewController? {
        var targetVC: UIViewController?
        switch path {
        // 根据URL的path跳转不同路径
        case "/view1":
            targetVC = Method1ViewController()
            break
        case "/view2":
            targetVC = Method2ViewController()
            break
        case "/view3":
            targetVC = Method3ViewController()
            break
        default:
            targetVC = nil
            break
        }
        return targetVC
    }

第二步 H5参考47.给App某个功能添加快捷方式中的shortcuts.html,其中共有三部分,大致为:

代码如下

<!DOCTYPE html>
<html lang="zh-CN">
    <head>
        <meta charset="UTF-8">
        <meta http-equiv="Content-Typecontent="text/htmlcharset=UTF-8">
        <meta name="viewportcontent="width=device-widthinitial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
        <meta name="apple-mobile-web-app-capablecontent="yes">
        <meta name="apple-mobile-web-app-status-bar-stylecontent="#ffffff">
        <meta name="apple-mobile-web-app-title" content="(title)">
            
        <link rel="apple-touch-icon-precomposedhref="data:image/jpeg;base64,(feature_icon)"/>
        <title>(title)</title>
        
    </head>
    <script>document.documentElement.style.fontSize = 100 * document.documentElement.clientWidth / 375 + "px"</script>
    <style>
       /* css代码链接 https://github.com/mokong/WKAddToHomeScreen */
    </style>
    <body>
        <a id="redirecthref="(urlToRedirect.absoluteString)"></a>
        <div id="container">
            <div class="main">
                <div class="subject">添加快捷功能到桌面</div>
            </div>
            <div class="guide">
                <div class="content">
                    <p class="tips">
                    点击下方工具栏上的<img class="iconsrc="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/XEbFrgamEdvSxVFOBeuZ.png">
                    </p>
                    <p class="tips">
                        并选择<img class="iconsrc="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/IkKEhyTLQpYtqXMZBYtQ.png">“<strong>添加到主屏幕</strong>”
                    </p>
                    <img class="toolbarsrc="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/oFNuXVhPJYvBDJPXJTmt.jpg">
                    <img class="arrowsrc="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/FlBEnTRnlhMyLyVhlfZT.png">
                </div>
            </div>
        </div>
    </body>
</html>

<scriptype="text/javascript">
    
    if (window.navigator.standalone) {
        
        var element = document.getElementById('container');
        element.style.display = "none";
        
        vaelement = document.getElementById('redirect');
        vaevent = document.createEvent('MouseEvents');
        event.initEvent('click', truetruedocument.defaultView, 1, 0, 0, 0, 0, falsefalsefalsefalse, 0, null);
        document.body.style.backgroundColor = '#FFFFFF';
        setTimeout(function() { element.dispatchEvent(event); }, 25);
        
    } else {
        
        vaelement = document.getElementById('container');
        element.style.display = "inline";
    }

</script>

其中有关于涉及到桌面快捷方式图标和标题设置解释参考苹果官方的Configuring Web Applications,如下

<!-- Specifying a Webpage Icon for Web Clip -->
<link rel="apple-touch-iconhref="touch-icon-iphone.png">

<!-- apple-touch-icon-precomposed与apple-touch-icon的区别为前者是原图,后者是会被苹果处理图片,这两个使用二选一即可
<link rel="apple-touch-icon-precomposed" href="xxx.png"> -->

<!-- Specifying a Launch Screen Image -->
<link rel="apple-touch-startup-image" href="/launch.png">

<!-- Adding a Launch Icon Title -->
<meta name="apple-mobile-web-app-title" content="AppTitle">

<!-- Hiding Safari User Interface Components -->
<meta name="apple-mobile-web-app-capable" content="yes">

<!-- Changinthe Status Bar Appearance> -->
<meta name="apple-mobile-web-app-status-bar-style" content="black">

<!-- Linking to Other Native Apps -->
<a href="tel:1-408-555-5555">Call me</a>

方法一

H5提供网页,比如上面示例代码然后把其中header部分内容设置固定

获取图片DataURI格式数据Swift代码如下

// 获取图片DataURI格式数据
  if let iconData = UIImage(named: "homeScreen")?.jpegData(compressionQuality: 0.5) {
      let iconDataURI = iconData.base64EncodedString()
  }

具体步骤如下

  1. 配置好客户端的URLScheme

  2. H5提供编写好的网页,如果没有H5,可使用上面的shortcuts.html内容,把其中的待替换字段(title)、(feature_icon)、以及(urlToRedirect.absoluteString)设置自己APP的,其中的apple-touchicon-precomposed需要放图标经过DataURI后的String

  3. 需要服务端提供URL,返回这个网页,然后客户端打开这个URL。如果服务端也没有。。。那就跟我一样,使用模拟接口返回,打开mocky,(可能需要注册),Response Content Type 设置text/html,HTTP Response Body 中放入下面的网页内容,然后点击底部的GENERATE MY HTTP RESPONSE,就会生成一个URL

    b52da4259fd00c0f41b35a77572889eb.png

  4. 点击加快捷方式的地方,直接openURL即可

    func addMethod1(_ sender: Any) {
            // 方法一,不需要本地放H5数据,只需要打开指定URL即可
            // 可使用mocky来提供模拟接口
            let urlStr = "https://run.mocky.io/v3/98baaf4a-edec-4956-8506-7bbfca349d07"
            
            UIApplication.shared.open(URL(string: urlStr)!, options: [:], completionHandler: nil)
        }

方法二

参考给App的某个功能添加快捷方式,文章中使用的是客户端自建服务器返回DataURI数据的方法,具体操作如下

  1. 配置好客户端的URL Schemes

  2. 客户端使用Pod添加Swifter,用于自建服务器

  3. H5提供编写好的网页,使用上面的shortcuts.html内容,其中的待替换字段不要动

  4. 点击加快捷方式时,客户端读取html内容替换里面指定字段,转为DataURI,启动本地服务器,并返回DataURI数据

    func addMethod2(_ sender: Any) {
            // 定义好的URL Scheme
            let schemeStr = "mkaddtohomescreen://page/view2"
            // 要替换的桌面快捷方式图标
            let shortcutImageData = UIImage(named: "homescreen")?.jpegData(compressionQuality: 0.5)
            // 要替换的桌面快捷方式标题
            let shortcutTitle = "添加到主屏幕2"
    
            guard  let schemeURL = URL(string: schemeStr),
                   let shortcutImageStr = shortcutImageData?.base64EncodedString() else {
                return
            }
    
            // 替换H5中的内容
            let htmlStr = htmlFor(title: shortcutTitle, urlToRedirect: schemeURL.absoluteString, icon: shortcutImageStr)
    
            guard let base64 = htmlStr.data(using: .utf8)?.base64EncodedString() else {
                return
            }
    
            // 启动本地服务器端口号是9081
            if let shortcutUrl = URL(string: "http://localhost:9081/s") {
                // 转为dataURI格式
                let server = HttpServer()
                server["/s"] = { request in
                    return .movedPermanently("data:text/html;base64,(base64)")
                }
                tryserver.start(9081)
                UIApplication.shared.open(shortcutUrl, options: [:], completionHandler: nil)
            }
        }
        
        func htmlFor(title: String, urlToRedirect: String, icon: String) -> String {
            let shortcutsPath = Bundle.main.path(forResource: "content2", ofType: "html")
            
            vashortcutsContent = try! String(contentsOfFile: shortcutsPath!) as String
            shortcutsContent = shortcutsContent.replacingOccurrences(of: "\(title)", with: title)
            shortcutsContent = shortcutsContent.replacingOccurrences(of: "\(urlToRedirect.absoluteString)", with: urlToRedirect)
            shortcutsContent = shortcutsContent.replacingOccurrences(of: "\(feature_icon)", with: icon)
    
            print(shortcutsContent)
            return shortcutsContent
        }

还没完

代码放到了Github大家下载运行后会发现还有一个问题解决,即:当使用方法一,添加快捷标签到桌面后,第一次点击桌面的快捷标签打开了APP;再次点击桌面的快捷标签,显示白屏原因第一次打开快捷标签没有关闭,重新打开时没有触发加载,所以也就没有跳转APP。而使用了DataURI加载的方法二,则没有这个问题,每次点击图标均可以直接跳转。

但是对比支付宝的添加到桌面发现支付宝的也是采用的方法一,第一次从桌面添加的快捷打开时自动跳转到支付宝,第二次点击桌面到快捷图标时,发现也是停留在一个页面,但是支付宝在这个页面上放了东西,可以称之为中间页。如下

77f40fd70beb6c61699429f3c7205c72.png

要怎么实现中间页那种效果呢,目前笔者方法一的实现,点击时依赖的是服务端返回的H5网页内容,里面的Script会根据进入方式的不同,直接自跳转打开APP的URL Scheme;所以想要添加中间页,嗯,想法是:嵌套一层。即:

  • 服务端返回的H5网页内容,里面的Script不直接跳转打开APP的URL Scheme,而是跳转中间页的链接

  • 中间页的页面,同样的逻辑,再次跳转打开APP的URLScheme;同时中间页的页面添加按钮,点击也是跳转APP的URLScheme。这样,第一次点击时,是桌面-中间页-APP的URL Scheme;第二次点击时,则是直接显示中间页,然后手动点击中间页上的立即进入,再次打开APP。

下面来尝试一下:首先编辑中间页面的H5,大致内容如下,仅供参考就是把之前H5页面的body部分简单修改一下,添加一button事件是点击打开Scheme,同时自动跳转Scheme的逻辑也还存在

<!DOCTYPE html>
<htmlang="zh-CN">
    <head>
        <meta charset="UTF-8">
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
        <meta name="apple-mobile-web-app-capable" content="yes">
        <meta name="apple-mobile-web-app-status-bar-style" content="#ffffff">
        <meta name="apple-mobile-web-app-title" content="方法三中间页内容">
        <title>"方法三中间页标题"</title>
        
    </head>
    <script>document.documentElement.style.fontSize = 100 * document.documentElement.clientWidth / 375 + "px"</script>
    <style>
          /* css代码链接 https://github.com/mokong/WKAddToHomeScreen */
    </style>
    <body>
        <a id="redirect" href="mkaddtohomescreen://page/view3"></a>
        <div id="B_container" class="backguidestyle="displayblock; width='100%'; background-color=#cyan">
          <div class="tips">你即将进入</div>
          <img id="B_icon" class="icon" src=""></img>
          <div id="B_appname" class="appname">MKAddToHomeScreen</div>
          <button class="enter" onclick="jumpSchema()" style="background-color: #red; widht=100%; height=64px">立即进入</button>
        </div>
    </body>
</html>

<script type="text/javascript">
    
    function jumpSchema() {
        vaelement = document.getElementById('redirect');
        var event = document.createEvent('MouseEvents');
        event.initEvent('click', truetrue, document.defaultView, 1, 0, 0, 0, 0, false, false, false, false, 0, null);
        document.body.style.backgroundColor = '#FFFFFF';
        setTimeout(function() { element.dispatchEvent(event); }, 25);
    }

    
    if (window.navigator.standalone) {
        vaelement = document.getElementById('redirect');
        var event = document.createEvent('MouseEvents');
        event.initEvent('click', true, truedocument.defaultView, 1, 0, 0, 0, 0, false, false, false, false, 0, null);
        document.body.style.backgroundColor = '#FFFFFF';
        setTimeout(function() { element.dispatchEvent(event); }, 25);
        
    } else {
        
        var element = document.getElementById('container');
        element.style.display = "inline";
    }

</script>

把中间页的网页放到mocky中按照同样的方式(Response Content Type 设置为text/html,HTTP Response Body 中放入网页内容),生成一个URL,然后把这个URL放到之前网页要自跳转的href中,然后再把之前网页再用mocky生成一个链接,在APP中使用openURL的方式打开最后生成的这个链接,运行,调试

发现结果期望的,即第一次打开直接跳转,第二次打开显示中间页上面有点击跳转按钮;但是中间页的样式看起来确跟支付宝的不一样,这样生成的中间页因为经过了一次跳转,所以顶部底部都显示了Safari二级页面的样式,嗯哼,这个不是笔者所希望的效果,而且体验支付宝的效果之后,发现支付宝的中间页是没有二级页面的那种头部底部的,所以,那是怎么实现的呢?

05258b287482e4a10a6882cc942251a5.jpeg

如果不想要中间页显示为二级页面的形式,就不能采用上面那种经过一次跳转方法。只能采用一页面的方法,在一个H5页面上想办法。所以现在想要的是,在同一个页面上,从APP跳转的时候显示“引导添加到桌面”的样式,从桌面打开时显示“中间页”的样式。

按照这个逻辑来,用两个div,包括两段样式,根据进入方式的不同,设置两个div的显示隐藏是不是就可以了呢?说做就做,把上面第二个html中的内容和样式放到第一个html中,代码如下middle_container是中间页的div,jump_container是引导页div,然后根据window.navigator.standalone判断显示哪一个div,middle_container中按钮点击是跳转打开APP,同时再把第一个html的跳转由跳转中间页改为打开APP

Ps:

检测Web应用程序当前是否运行在全屏状态,只要检测window.navigator.standalone是否为true就可以了,如果这个属性为true表示Web应用程序当前运行在全屏状态,否则运行在非全屏状态。可用于检测到Web应用程序运行在非全屏状态时提示用户把Web应用程序的图标添加到主屏幕。

<!DOCTYPE html>
<html lang="zh-CN">
    <head>
        <meta charset="UTF-8">
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
        <meta name="apple-mobile-web-app-capable" content="yes">
        <meta name="apple-mobile-web-app-status-bar-style" content="#ffffff">
        <meta name="apple-mobile-web-app-title" content="方法三标题">
        
        <link rel="apple-touch-icon-precomposed" href="data:image/jpeg;base64,imageData"/>
        <title>方法三网页标题</title>
        
    </head>
    <script>document.documentElement.style.fontSize = 100 * document.documentElement.clientWidth / 375 + "px"</script>
    <style>
       /* css代码见链接 https://github.com/mokong/WKAddToHomeScreen */
    </style>
    <body>
        <a id="redirect" href="mkaddtohomescreen://page/view3"></a>
        <div id="middle_container" class="backguide">
          <div class="middle_tips">你即将进入</div>
          <img class="middle_icon" src=""></img>
          <div class="middle_appname">MKAddToHomeScreen</div>
          <button class="middle_enter" onclick="jumpSchema()" style="background-color: #red; widht=100%; height=64px">立即进入</button>
        </div>
        <div id="jump_container">
            <div class="main">
                <div class="subject">添加快捷功能到桌面</div>
            </div>
            <div class="guide">
                <div class="content">
                    <p class="tips">
                    点击下方工具栏上的<img class="icon" src="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/XEbFrgamEdvSxVFOBeuZ.png">
                    </p>
                    <p class="tips">
                        并选择<img class="icon" src="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/IkKEhyTLQpYtqXMZBYtQ.png">“<strong>添加到主屏幕</strong>”
                    </p>
                    <img class="toolbar" src="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/oFNuXVhPJYvBDJPXJTmt.jpg">
                    <img class="arrow" src="https://dariel-1256714552.cos.ap-shanghai.myqcloud.com/FlBEnTRnlhMyLyVhlfZT.png">
                </div>
            </div>
        </div>
    </body>
</html>

<scriptype="text/javascript">
    
    function jumpSchema() {
        var element = document.getElementById('redirect');
        var event = document.createEvent('MouseEvents');
        event.initEvent('click', true, true, document.defaultView, 1, 0, 0, 0, 0, false, false, false, false, 0, null);
        document.body.style.backgroundColor = '#FFFFFF';
        setTimeout(function() { element.dispatchEvent(event); }, 25);
    }
    
    if (window.navigator.standalone) {
        
        var middle_element = document.getElementById('middle_container');
        var jump_element = document.getElementById('jump_container');
        
        middle_element.style.display = "inline";
        jump_element.style.display = "none"
        
        var element = document.getElementById('redirect');
        var event = document.createEvent('MouseEvents');
        event.initEvent('click', true, true, document.defaultView, 1, 0, 0, 0, 0, false, false, false, false, 0, null);
        document.body.style.backgroundColor = '#FFFFFF';
        setTimeout(function() { element.dispatchEvent(event); }, 25);
        
    } else {
        
        var middle_element = document.getElementById('middle_container');
        var jump_element = document.getElementById('jump_container');
        
        middle_element.style.display = "none";
        jump_element.style.display = "inline";
    }

</script>

然后用当前内容放到mocky生成一个链接,在程序中打开这个链接,运行调试,发现结果和预期一致,没有了二级界面的样式,而且再次打开,页面显示也不是空白,而是如下样式:

a93395ab6f2c36092dc4c48ac7a59ecf.png

总结

完整代码在:Github

笔者感觉两种方式各有优缺点:方法一依赖于网络,因为需要服务端返回的网页内容,加载完成后才能进行下一步跳转。而方法二采用DataURI方式的,把数据已经转为string放在了本地,点击时直接加载,故而不依赖网络。但方法一实现简单,客户端、H5、和服务端配合虽然有些冗余,但工作量小,很容易实现。方法二的加载采用DataURI,查看调试数据不方便。根据笔者的观察,支付宝其实采用的是方法二,没网络时候也可以加载打开主APP,且在方法二的基础上还加上了中间页。

附图:

2ea1e80ee087c03ef73e96eb964f86fc.png

参考

  • iOS开发 将App内部功能块生成桌面快捷方式

  • 给App的某个功能添加桌面快捷方式

  • ios app内页面添加到桌面

  • ios 关于支付宝添加桌面快捷方式的探究

  • 模拟接口返回

原文地址:https://blog.csdn.net/SOHU_TECH/article/details/128337388

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_44458.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注