1.PWA简介:

Progressive Web App 是由谷歌提出推广的,在移动端利用提供的标准化框架,在网页应用中实现和原生应用相近的用户体验。

2.PWA到底有哪些过人之处(特性)?

1、Installability(可安装性):添加到主屏
2、App Shell:第一次渲染渲个壳、等异步数据来了再填充
3、Offline Web App (离线能力)
4、Re-engageable:推送通知的能力。推送通知依赖Service Worker与HTTP Push,不过默认的支持是GCM(Google提供的云服务)

将Web和App的技术相结合:

  •  不需要安装
  •  快速加载
  •  推送消息
  •  桌面图标
  •  全屏体验

3.PWA核心技术:

Web App Manifest
App shell
Service worker
push Notifion

4.manifest.json 配置文件 添加主屏配置

<link rel="manifest" href="/manifest.json">

5.Service worker 用来实现页面的缓存和离线, 后台通知等功能

Service Worker 是什么? service worker 是独立于当前页面的一段运行在浏览器后台进程里的脚本。

要求:需要HTTPS本地调试localhost 浏览器支持(目前andorid的chrome部分支持)

sw生命周期:注册完成 安装-》激活-》接管页面 image

sw里面的一些简单变量说明

  • self: 表示 Service Worker 作用域, 也是全局变量
  • self.skipWaiting: 表示强制当前处在 waiting 状态的脚本进入 activate 状态
  • self.clients.claim():获取到客户端页面的控制权

1.sw 注册:检查 Service Worker API 是否可用,如果可用,则在页面加载后注册位于 /sw.js 的服务工作线程

if ('serviceWorker' in navigator) {
  window.addEventListener('load', function() {
    navigator.serviceWorker.register('/sw.js').then(function(registration) {
      // Registration was successful
      console.log('ServiceWorker registration successful with scope: ', registration.scope);
    }).catch(function(err) {
      // registration failed 
      console.log('ServiceWorker registration failed: ', err);
    });
  });
}

2.sw安装:执行以下步骤

a.打开缓存
b.缓存文件
c.确认文件是否缓存

 

//SW安装
self.addEventListener('install', e=> {
  console.log('[sw]ServiceWorker Install');
  e.waitUntil(
    caches.open(staticCache).then(function(cache) {
      return cache.addAll(filesToCache)
      .then(function () {
        console.log('[sw]All files are cached');
        return self.skipWaiting();//跳过等待直接activate
      })
      .catch(function (error) {
        console.log('[sw]Failed to cache', error);
      })
    })
  );
});

3.sw激活: 如果你的 service worker 已经被安装,但是刷新页面时有一个新版本的可用,新版的 service worker 会在后台安装,但是还没激活。当不再有任何已加载的页面在使用旧版的 service worker 的时候,新版本才会激活。一旦再也没有更多的这样已加载的页面,新的 service worker 就会被激活。

//SW激活
self.addEventListener('activate', function(e) {
  console.log('[sw]ServiceWorker Activate');
  var arr = [htmlCache,staticCache];
  e.waitUntil(
    caches.keys().then(function(keyList) {
      return Promise.all(keyList.map(function(key) {
        if (arr.indexOf(key) === -1) {
          console.log('[sw]ServiceWorker Removing old cache', key);
          return caches.delete(key);
        }
      }));
    })
  );
  return self.clients.claim();  //更新客户端取得页面控制权
});

  

4.sw fetch缓存和返回请求:

self.addEventListener('fetch', e=> {
  // console.info('Event: Fetch');
  var request = e.request;
  let url = e.request.url
  e.respondWith(
    //If request is already in cache, return it
    caches.match(request).then(function(response) {
      if (response) {
        console.log('[sw]Found response in cache:', url);
        return response;
      }
      //if request is not cached, add it to cache
      console.log('[sw]No response found in cache. About to fetch from network...');
      return fetch(request).then(function(response) {
        var responseToCache = response.clone();
        //请求和响应流只能被读取一次。为了给浏览器返回响应以及把它缓存起来,我们不得不克隆一份。所以原始的会返回给浏览器,
        //克隆的会发送到缓存中。它们都是读取了一次
        caches.open(htmlCache).then(
          function(cache) {
            cache.put(request, responseToCache).catch(function(err) {
              console.warn(request.url + ': ' + err.message);
            });
          });
        console.log('[sw]Response from network is:', url);
        return response;
      });
    })
  );
});

  

caches.match(event.request) 允许我们对网络请求的资源和 cache 里可获取的资源进行匹配,查看是否缓存中有相应的资源

如果没有在缓存中找到匹配的资源,你可以告诉浏览器对着资源直接去fetch 默认的网络请求: fetch(event.request)

如果没有在缓存中找到匹配的资源,同时网络也不可用,你可以用 match() 把一些回退的页面作为响应来匹配这些资源,比如: caches.match('/offline.html');

6.app shell

示例链接:速卖通

7.消息推送

示例链接:https://progressive-web-application.herokuapp.com/

8.sw-toolbox

示例链接:线上环境 m.tomtop.com
api:https://googlechrome.github.io/sw-toolbox/

posted on 2018-01-03 10:05  夏沙1234  阅读(439)  评论(1编辑  收藏  举报