takeaway
With the development of Internet technology, Web application has become more and more popular, and the development of technology is more and more rapid, especially the arrival of mobile Internet makes HTML5 technology, Hybrid development, more popular, but Web application failed to get rid of some fundamental problems in the PC era, the required resources rely on network download. The user experience always depends on the browser, which makes the experience of Web applications, especially in mobile phones, feel “irregular” compared with Native applications. The arrival of PWA technology makes the next generation of Web applications finally on the right track!
Based on this, this paper mainly consists of the following parts:
- Explain basic concepts of PWA
- Service Worker principle explanation
- Explain the Web Push protocol
- Transform a SPA project into a PWA
- Manifest.json configuration parsing
- Service Worker resource cache
- Added the save to desktop feature
- Receive message push
- conclusion
What is PWA?
Misc Its Continuous Supply Web App (PWA), a progressive web application, is The next generation Web application model presented by Google at the 2016 Google LEI /O conference and has been developing rapidly in the following days. A PWA application is first and foremost a Web page, and a Web application can be written using Web technology. Then the App Manifest and Service Worker are used to realize the installation and offline functions of PWA.
The characteristics of the PWA
- Progressive: It is suitable for all users of any browser, because it is developed with progressive enhancement as its core purpose.
- Adaptive: Suitable for any model: desktop, mobile, tablet, or any future device.
- Connection independence: Ability to work offline or in low-quality network conditions with the help of service worker threads.
- Offline push: With push notification, our App can improve user experience like Native App.
- Keep up to date: Keep up to date with the service worker update process.
- Security: Provided over HTTPS to prevent snooping and ensure that content is not tampered with.
For mobile terminal, we use simple sentence to profile a PWA application is that we develop the H5 page increase can be added to the functionality of the screen, click on the home screen icon can be start animation and hide the address bar to offline caching function, even if the user mobile phone no network, can still use some offline capabilities. These features and functions are exactly what we are currently optimizing for the mobile Web. With these features, web applications will gradually become closer to native apps and truly realize second to open optimization.
What is a Service Worker
Service Worker is a feature based on HTML5 API and the most important feature in PWA technology stack. It adds persistent offline cache and network proxy capabilities on the basis of Web Worker. The ability to manipulate the browser Cache with JavaScript combined with the Cache API makes Service workers and PWA inseparable.
Service Worker Overview:
- A separate thread of execution, a separate scope, a separate execution environment, its own separate context.
- Once installed, it exists forever unless it is manually unregistered. It runs in the background even when Chrome is closed. This feature can be used to implement offline message push function.
- To ensure security, it must work in the HTTPS environment. Of course, when debugging locally, using localhost is not restricted by HTTPS.
- Provides an interface for intercepting browser requests and can control all page requests in the open scope. Note that once the request is taken over by the Service Worker, it means that any request is under your control. You must make sure that the fault tolerance mechanism is in place to ensure the normal operation of the page.
- Since it is a separate thread, the Service Worker cannot directly manipulate the page DOM. But it can be handled through the event mechanism. For example, using postMessage.
Service Worker lifecycle:
- Register (register) : here generally refers to in the browser to parse the JavaScript has registered the Service logic, when the Worker is called the navigator. ServiceWorker. The register () when dealing with things.
- Installing: This happens after the Service Worker is registered and the installation begins.
- After installation (installed/waiting) : When the Service Worker has completed the installation, the install event is triggered and static resources are cached offline. If the old Service Worker is still running, it will be waiting. If you close the browser or call self.skipWaiting(), The Service Worker in waiting state is forced to enter the activate state.
- Activating: Call self.clients.claim() to force control of an uncontrolled client. For example, if your browser has multiple Windows with the Service Worker open, it will not cut the window. Replacing the old Service Worker script no longer controls these pages and will then be stopped. The Activate event is triggered.
- Activated: This state indicates that the Service Worker is activated successfully. During the Activate callback, the last version of the static resource cache will be cleared or other policies will be used to update the cache. This means that the Service Worker can handle functional fetch, sync, push, and Message events.
- Redundant state: This state indicates the end of a Service Worker’s life cycle.
The whole process can be explained as follows:
Events supported by Service workers:
Service Worker browser compatibility:
As a new technology, Service Worker is bound to have browser compatibility problems. As can be seen from the figure, it has good support for most Android devices, especially Chrome for Android. However, for iOS system, Service workers are not supported before 11.3, which may be one of the reasons why Service workers are not popular. However, the good news is that Apple announced that it would continue to update support for Service workers in the future, so the prospect is still worth looking forward to.
Being pushed
Message push, as the name implies, is the message push of an APP that you receive on your mobile phone. Compared with the mobile end Native application, Web application lacks this commonly used function. By virtue of the Push feature of PWA, the user can receive the message pushed by the website without entering the specific website when opening the browser, for example: With the help of Android Chrome, we can receive our project push without opening any browser or application, just like a real mobile push.
What is Web Push
Web Push is a process specification based on client, server and Push server, which can be divided into three steps:
- The client completes the logic that asks to subscribe to a user.
- The server invokes a Web Push protocol-compliant interface to deliver a push message to a push server (which is determined by the browser, and all the developer can do is control what data is sent).
- The push server pushes the message to the corresponding browser, and the user receives the push.
The following figure shows a user subscription process:
The so-called subscriber subscription means that IF I want to receive push notifications from your website or APP, I need to tell you who I am and I need to send my logo to you, otherwise how will you know to push me.
The following figure shows how the server pushes a subscription request from the user:
- First, in the background of Your project, Your Server stores the identity that users send you when they subscribe.
- When the background needs to Push to you, find the logo, and then contact the Push Service to send the content and logo to the Push Service, and then ask the Push Service to Push the message to the client. (iOS and Android have their own push servers, depending on the operating system).
- There is a convention that the identity of the user should be consistent with the push service, for example using Chrome, which is Google’s Push service (FCM).
Start to transform existing SPA applications
This chapter will be a SPA project based on Vue. Js2.6 version for PWA transformation, the original project development process will not be explained, readers can go to Github to see the source, the final experience address: App.nihaoshijie.com.cn, please use Safari or Android Chrome to open the experience.
Add manifest.json configuration page parameters:
Adding shortcuts to the desktop itself is part of the PWA application, which makes our application look more like a Web App. We create a new manifest.json file in the public folder of the front-end project:
{
"name": "WECIRCLE"."short_name": "WECIRCLE"."icons": [{"src": "./img/icons/android-chrome-192x192.png"."sizes": "192x192"."type": "image/png"
},
{
"src": "./img/icons/android-chrome-512x512.png"."sizes": "512x512"."type": "image/png"}]."start_url": "./index.html"."display": "standalone"."background_color": "# 000000"."theme_color": "# 181818"
}
Copy the code
Among them:
- Name: Specifies the name of the Web App, that is, the name of the icon saved to the desktop.
- Short_name: If the name is too long, the system displays short_name instead of name, which is the short name of the Web App.
- Start_url: Specifies the URL to load when the user opens the Web App. The relative URL is relative to manifest.json. Here we specify index.html as the start page for the Web App.
- display: specifies the display mode to apply. It has four values to choose from:
fullscreen
: Full screen display, will try to fill all display areas.standalone
: Browser-specific UI (navigation bar, toolbar, etc.) will be hidden so it looks more like a Native App.minimal-ui
Similar to the standalone display, the browser-specific UI is minimized to a single button, with slightly different implementations in different browsers.browser
: Generally speaking, it will open in the same style as normal browser use. It should be noted that the standalone effect will appear when browsers on some systems do not support fullscreen, the standalone effect will appear when browsers do not support standalone, and the standalone effect will appear when browsers do not support standalone, and so on. - icons: specifies the application’s desktop icon and launch page image, represented in an array:
sizes
: Icon size. By specifying the size, the system will select the most appropriate icon to display in the corresponding position.src
: File path of the icon. The relative path is relative to the manifest.json file. Absolute paths can also be used for exampleXXX. PNG.type
: Image type of the icon. The browser will choose the ICONS closest to 128dp(px = dp * (dpi / 160)) as the launcher image. - Background_color: Using the same color can achieve a smooth transition from the launch screen to the home page, and can also be used to improve the user experience when page resources are loading. Combining the ICONS property, we can define the launch page effect of the background color + image icon. Similar to the splash Screen effect of Native App.
- Theme_color: Specifies the theme color of the Web App. You can use this property to control the color of the browser UI. Such as status bar, content page status bar, address bar color.
Of course, here we just list the manifest.json properties used in our project. For more parameter configuration, you can refer to MDN. Of course, if you think these configurations are too tedious, you can also use the Web App Manifest Generator to achieve visual configuration.
Setting iOS parameters:
Manifest.json is powerful but also suffers from browser compatibility issues, as shown below:Manifest.json is partially supported in iOS, so we need to configure meta properties in the head to make iOS more complete:
<meta name="apple-mobile-web-app-capable" content="yes">
<meta name="apple-mobile-web-app-title" content="WECIRCLE">
<link rel="apple-touch-icon" sizes="76x76" href="./img/icons/apple-touch-icon-76x76-1.png" />
<link rel="apple-touch-icon" sizes="152x152" href="./img/icons/apple-touch-icon-152x152.png-1" />
<link rel="apple-touch-icon" sizes="180x180" href="./img/icons/apple-touch-icon-180x180.png-1" />
Copy the code
- Apple-touch-icon: Specifies the ICONS of the application, similar to the ICONS configuration of the manifest.json file, which also supports sizes for different scenarios.
- Apple-mobile-web-app-capable: Similar to display in manifest.json, the standalone mode is available by setting it to yes, which is currently supported on iOS.
- Apple-mobile-web-app-title: specifies the name of the application.
- Apple-mobile-web-app-status-bar-style: Specifies an iOS mobile device’s status bar, as Default, Black, or black-always. 3.
Apple-touch-icon, apple-mobile-web-app-status-bar-style are valid, while manifest.json ICONS are not recognized by iOS. Here is a screenshot of safari on iOS saved to the desktop:
In Chrome for Android:
Finally, don’t forget to introduce the manifest.json file in HTML:
<link rel="manifest" href="manifest.json">
Copy the code
Register and use the Service Worker’s caching feature:
1. Register the Service Worker:
Add the following code to index.html under the public folder of the front-end project:
if ('serviceWorker' in navigator) {
window.addEventListener('load'.function () {
navigator.serviceWorker.register('/sw-my.js', {scope: '/'})
.then(function (registration) {
// Registration succeeded
console.log('ServiceWorker registration successful with scope: ', registration.scope)
})
.catch(function (err) {
// registration failed :(
console.log('ServiceWorker registration failed: ', err)
})
})
}
Copy the code
Using serviceWorkerContainer. The register () to register the Service Worker, here should be fault-tolerant, guarantee certain models in does not support the Service Worker will be able to run normally, without error.
Note also that the serviceWorker object will only be in the Navigator if HTTPS is used.
2. Create a front-end project in the public foldersw-my.js
And define the file path to cache:
// Define the file to cache
var cacheFiles = [
'./lib/weui/weui.min.js'.'./lib/slider/slider.js'.'./lib/weui/weui.min.css'
]
// Define the cache key
var cacheName = '20190301'
Copy the code
3. Listen for install events to cache related files:
// Listen for install events and cache files after installation
self.addEventListener('install'.function (e) {
console.log('Service Worker status: install')
// Find the cache corresponding to the key and get the operable cache object
var cacheOpenPromise = caches.open(cacheName).then(function (cache) {
// Add the files that need to be cached
return cache.addAll(cacheFiles)
})
// Pass the Promise object to event
e.waitUntil(cacheOpenPromise)
})
Copy the code
The standard web worker programming method we use in sw-my.js runs in another global context (self), which is different from window, so we use self.addeventListener ().
Cache API is the interface provided by the Service Worker to operate the Cache. These interfaces are implemented based on promises, including Cache and Cache Storage. Cache directly deals with requests. CacheStorage provides a storage mechanism for cached pairs of Request/Response objects. CacheStorage represents a storage instance of a Cache object, and we can directly access the Cache API using the global Caches property.
Cache API description:
Cache.match(request, options) returns a Promise object, and resolve is the first cached request that matches the Cache object. Cache.matchall (request, options) returns a Promise object. The result of resolve is an array of all requests matching the Cache object. Cache.addall (requests) receives an array of urls, retrieves and adds the returned Response object to the given Cache object. Cache.delete(request, options) Searches for Cache entries whose key is request. If found, the Cache entry is deleted and a Promise object that resolves to true is returned. If not found, return a Promise object that resolves to false. Cache.keys(request, options) returns a Promise object, and resolve returns an array of Cache keys.Copy the code
4. Listen for the fetch event to use cached data:
self.addEventListener('fetch'.function (e) {
console.log('Now requesting:' + e.request.url)
e.respondWith(
// Determine whether the current request needs caching
caches.match(e.request).then(function (cache) {
// If there is a cache, use the cache
return cache || fetch(e.request)
}).catch(function (err) {
console.log(err)
// If the cache fails, the request will be reissued
return fetch(e.request)
})
)
})
Copy the code
On step we will related resources to the cache, then use the cache, it also should be fault-tolerant logic, remember once the request is a Service Worker took over, the default browser requests are no longer effective, which means that the request and not hair, wrong or not all controlled by your own code, there must be compatible, When the cache fails or an internal error occurs, fetch is called in time to re-initiate the request. As mentioned above in the life cycle of the Service Worker, the trigger of the FETCH event must depend on the Service Worker to enter the activated state, which leads to step 5.
Listen on the activate event to update the cached data:
Updating the cache is an essential step in using the cache. If the cache cannot be updated, it is meaningless. We added the following code to sw-my.js:
// Listen for the activate event and check whether static resources in the cache are updated by the cache key
self.addEventListener('activate'.function (e) {
console.log('Service Worker status: activate')
var cachePromise = caches.keys().then(function (keys) {
// Iterate over the key used by the current scope
return Promise.all(keys.map(function (key) {
// If the newly obtained key is inconsistent with the previously cached key, the cache of the previous version will be deleted
if(key ! == cacheName) {return caches.delete(key)
}
}))
})
e.waitUntil(cachePromise)
// Ensure that the fetch is triggered at the first load
return self.clients.claim()
})
Copy the code
- Service Worker script updates are triggered every time an installed Service Worker page is opened.
- A Service Worker script update is triggered when the last script update has been written to the Service Worker database for more than 24 hours.
- Service Worker script updates are triggered when the sw-my.js file changes.
The update process is similar to the installation process, except that it does not enter immediately after the update is successfully installedactive
Status, the updated Service Worker will co-exist with the original Service Worker and run itinstall
, once the new Service Worker is installed successfully, it enters the wait state and needs to wait for the old Service Worker to terminate.self.skipWaiting()
You can prevent waiting and activate the new Service Worker as soon as it is successfully installed.self.clients.claim()
Method to control the unmanaged clients, that is, set the Service Worker itself to activate.
Open the Chrome console and click Application to check the Service Worker status:
- Status Indicates the status of the current Service Worker.
- Clients Indicates how many Windows are currently connected to the Service Worker.
It is important to note that if your browser has multiple Windows open, the window must be closed and open for the Service Worker to update successfully without calling self.skipwaiting ().
Use offline-plugin to improve Service Worker:
The Service Worker logic we wrote above has been completed, but there are still some defects. For example, after each build, the MD5 of each file will change, so we need to manually modify the cache file list each time we write:
var cacheFiles = [
'./static/js/vendor.d70d8829.js'
'./static/js/app.d70d8869.js'
]
Copy the code
The offline-plugin in Webpack will help us fix this and automatically generate sw-my.js.
1. Install offline-plugin:
npm install offline-plugin --save
Copy the code
In 2.vue.config.js
In the configuration:
configureWebpack: {
plugins: [
new OfflinePlugin({
// Requires a ServiceWorker event callback to be triggered
ServiceWorker: {
events: true
},
// The update policy selects all updates
updateStrategy: 'all'.// Remove some files that do not need to be cached
excludes: ['/. * * *'.'**/*.map'.'**/*.gz'.'**/*.png'.'**/*.jpg'].// Add an update to index.html
rewrites (asset) {
if (asset.indexOf('index.html') > -1) {
return './index.html'
}
return asset
}
})
]
}
Copy the code
3. Create a front-end project in the SRC directoryregisterServiceWorker.js
Register for Service Worker:
import * as OfflinePluginRuntime from 'offline-plugin/runtime'
OfflinePluginRuntime.install({
onUpdateReady: () = > {
// After the update is complete, call the applyUpdate (skipWaiting () method
OfflinePluginRuntime.applyUpdate()
},
onUpdated: () = > {
// Pop a confirmation box
weui.confirm('New version found. Update? '.() = >{
// Refresh the page
window.location.reload()
}, () = >{}, {title: ' '}); }})Copy the code
When the Service Worker is updated, pop up the window to confirm whether it is updated, as shown below:
Here’s an explanation:
- After offline-plugin is selected, the logics related to Service Worker registration and Service Worker cache can be removed, because offline-plugin can do these things for us.
- Offline-plugin automatically scans the dist directory created by WebPack and configates the cache list for these files, just as in the previous plugin.
excludes
: specifies a list of files that do not need to be cached. For example, we do not want to cache image resources and support regular expressions.updateStrategy
: specifies the cache policy to select full update, or incremental updatechanged
.Event: true,
Specifies the callback to trigger the Service Worker event, thismain.js
If this is set to true, the callback will be triggered.- We are in
main.js
The configuration in is to update the Service Worker immediately when it is updated without letting the Service Worker enter the wait state, which corresponds to the update process of the Service Worker mentioned above.
For more offline-plugin configurations, you can also go to the official website to see the documentation.
After executing the NPM run build command, the corresponding sw.js file will be generated and deployed to replace our hand-written sw-my.js.
In addition, we have the option of installing Chrome on PC to the desktop, which makes our application look more like a desktop application:
Add notification push logic:
The message Push logic is mainly divided into two schemes. One is very simple to use the Push function of the developer tool of Chrome on PC, which can be realized by Application->Service Worker panel ->Push button, as shown below:
This method can only simulate simple Push, and rely on the browser, most of which is used for debugging, while adding Push message for APP needs to be realized in combination with Web Push protocol. Meanwhile, Push message mainly includes front-end logic and back-end logic, in which:
Front-end logic includes:
- User authorization subscription logic
- After receiving a push, use the Notification logic
The back-end logic includes:
- Storage user authorization ID
- Send push requests to push servers based on the identity
1. Front-end subscription logic
To get the user id, you need to use the Service Worker for everything related to Web Push and notification. Add the following code to registerServiceworker.js that you created earlier:
navigator.serviceWorker.ready.then((registration) = > {
// The publicKey corresponds to the publicKey in the background
const publicKey = 'BAWz0cMW0hw4yYH-DwPrwyIVU0ee3f4oMrt6YLGPaDn3k5MNZtqjpYwUkD7nLz3AJwtgo-kZhB_1pbcmzyTVAxA';// The public key of the client defined by web-push, which corresponds to the web-push at the back end
// Get the subscription request (the browser will pop up a confirmation box to see if the user agrees with the notification push)
try {
if (window.PushManager) {
registration.pushManager.getSubscription().then(subscription= > {
// If the user is not subscribed and is a logged in user
if (subscription && window.localStorage.getItem('cuser')) {
const subscription = registration.pushManager.subscribe({
userVisibleOnly: true.// Indicates whether the push needs to be explicitly displayed to the user, i.e., whether there will be a message notification when the push is made. If there is no notification, it indicates a "silent" push. In Chrome, you must set it to true or the browser will report an error on the console
applicationServerKey: urlBase64ToUint8Array(publicKey)// The public key of the client defined by web-push, which corresponds to the web-push at the back end
})
// The user agrees
.then(function(subscription) {
console.log(subscription)
alert(subscription)
if (subscription && subscription.endpoint) {
// Store the database
let resp = service.post('users/addsubscription', {
subscription: JSON.stringify(subscription)
})
}
})
// The user disagrees or the generation fails
.catch(function(err) {
console.log("No it didn't. This happened: ", err)
});
} else {// The user has subscribed
console.log("You have subscribed our notification"); }}); }}catch(e){
console.log(e)
}
});
Copy the code
The logic of the above code is as follows:
- through
registration.pushManager.getSubscription()
First to determine whether the user has subscribed, that is, whether they have obtained the identity, and then getsubscription
That’s the logo we’re looking for. (latersubscription
In lieu of identity). - If the user has not subscribed, pass
registration.pushManager.subscribe()
You can getsubscription
When this method is called, the browser will ask the user to accept the subscription. This will pop a box:
If it is on the mobile terminal, the premise is to use Android Chrome, you will receive this prompt, as shown in the figure:3. When we click Agree, we getsubscription
And then sends the request to the background storage via service. thissubscription
It’s actually an object that looks like this:
{
"endpoint": "https://fcm.googleapis.com/fcm/send/eekuJ6272vI:APA91bEdnUY1cpyTfRFVMUJBx2CNQdA6Qg2FwP0oPibqltHxgZz__2ggmgSpE5bGRoI81cg inuT2clRDuqmmmtiqgYiG_WXQtvw83Mv41bJxJj89y1rglr5mvyiyHpBRml_y07uq1pVIc"."expirationTime": null."keys": {
"p256dh": "BLcOaaco6_dIjfIo3uiR6nDqERiCUwOuVT1mD5W45V99hvuYoqJxJZzKrKLsgE16zI_DA7o5PXXa8HVZvNz8PHg1"."auth": "vRqwuyij2AR9qkzUOwP3Pwx"}}Copy the code
Subscription is unique to each client.
2. Back-end push logic
Back end is based on node.js Express framework developed interface services, including Push logic, other logic is no longer described in the article, you can directly browse the source code, here only explain the Web Push logic. Since the user id sent by the front end is to be stored, a database is required to persist the data; here MongoDB is used to store the data, creating a new table Subscription. Create a new model.js file under the models folder of the back-end project as follows:
var mongoose = require('mongoose');
var Schema = mongoose.Schema;
var SubscriptionSchema = new mongoose.Schema({
subscription: { type: String.required:true },
userid: {type: String.unique: true },// Note that ref foreign keys are not used here
update: { type: Date.default: Date.now },
create: { type: Date.default: Date.now },
},{timestamps: {createdAt: 'create'.updatedAt:'update'}});
module.exports = mongoose.model('Subscription', SubscriptionSchema);
Copy the code
- subscriptionThe: field is a string that we will pass to the front-end object
JSON.stringify()
Once in a while. - userid: This field is used to identify the user
unique:true
The ref foreign key is not used here so that the field can be null, so that some messages may be pushed to users who have not logged in before.
Add an Express-based routing method to the back-end project:
/* * Add subscription information */
router.post('/addsubscription'.async (req, res, next)=> {
var userid = req.user ? req.user._id : ' ';
try {
var result = await Subscription.create({
subscription: req.body.subscription,
userid:userid
})
res.json({
code:0.data:result
})
}catch(e){
// console.log(e)
res.json({
code:0.data: e.errmsg.indexOf('dup key')?'has scription' : e.errmsg // Indicates that the user has subscribed}}}));Copy the code
The above code stores a Subscription using the Subscription. Create (). When the userID is stored, an error will be thrown indicating that the user has subscribed.
Create a push.js tool method under the utils folder of the backend project to implement the push logic. First install web-push, which is a Node.js web-based push package, and of course Java or Php based on it:
npm install web-push --save
Copy the code
Add code to push.js, first generate vapidKeys, which is the key we used in the front end, as follows:
var vapidKeys = webpush.generateVAPIDKeys();
Copy the code
Just generate it once, set it, and use it for the rest of your life, like this:
var vapidKeys = { publicKey:
'BAWz0cMW0hw4yYH-DwPrwyIVU0ee3f4oMrt6YLGPaDn3k5MNZ1tqjpYwUkD7nLz3AJwtgo-kZhB_1pbcmzyTVAxA'.privateKey: 'BJ_V2wtPYaVCl7Ef2GAkVxXB2ft9cTgw-b5lM2ggc8lo' };
webpush.setVapidDetails(
'mailto:[email protected]'.// If you don't need email notification, you can fill in this blank
vapidKeys.publicKey,
vapidKeys.privateKey
);
Copy the code
After completing these Settings, you can communicate with the Push Service to implement the Push. The code is as follows:
module.exports = async function(userid,data){
// Set the proxy
var option = {
proxy: 'http://113.10.152.92:3128' //http://www.freeproxylists.net/zh/hk.html
}
// Find subscription from the database
var obj = await Subscription.findOne({
userid: userid
}).exec();
console.log('Check if there is pushable Subscription')
console.log(obj)
if (obj && obj.subscription) {
console.log('Find Subscription can be pushed')
// Call sendNotification of Webpush to initiate push notifications
webpush.sendNotification(JSON.parse(obj.subscription), JSON.stringify(data),option).catch(function(err) {
console.error(err) }); }}Copy the code
To explain, since the push server we use is based on Google’s FCM, this service is not available in China (or sometimes available and sometimes not), so we need to set up an agent. Of course, there are many free foreign agents online. You can search on www.freeproxylists.net/zh/hk.html, if you want a stable one can buy a VPN service.
3. A notification will pop up after receiving the push
How do you notify the Notification after you receive a Push? That’s where the API for Notification comes in.
Next, register the push event in the Service Worker to receive the push request, and create a new sw-push.js in the public directory of the front-end project:
// Add a service worker listener for push
self.addEventListener('push'.function (e) {
var data = e.data
if (e.data) {
data = data.json()
e.waitUntil(
self.registration.showNotification(data.title, {
body: data.body || ' '.icon: data.img || "https://app.nihaoshijie.com.cn/img/icons/apple-touch-icon-180x180-1-touming.png".actions: [{
action: 'go-in'.title: 'Enter program'}]})); ; }else {
console.log('Push doesn't have any data')}})Copy the code
When the browser to receive push notifications, will enter this event, we through self. Registration. ShowNotification ()
- Title: The title of the message, which is a required value.
- Body: Indicates the entity of the message.
- Icon: The image of the configuration message that appears in the message.
- Actions: Configures the action item of the message before combining
notificationclick
Events enable click-to-click interaction of messages.
A notification box will pop up if you click OK in the previous notification permission:
On the phone it looks like this:
To receive notifications on the mobile or PC, the front-end must meet the following conditions:
- Chrome on PC should be able to jump the wall, which means it can use Google services.
- Chrome on mobile phones should have Chrome service (GMS) built in. According to the author’s experiments in China, Huawei, Vivo and Xiaomi series basically do not have Chrome service built in, while Nexus series phones can be used normally. Being able to connect to Google Play means yes.
Why all these conditions, you ask? The reason is that Web-push is a push implementation based on Google’s FCM(Cloud messaging mechanism), which is included in GMS. You can’t even get a message when Google bans Huawei phones from using Google services abroad.
Add click events to Notification for complete message flow and add the following code to sw-push.js:
self.addEventListener('notificationclick'.function (e) {
var action = e.action;
e.waitUntil(
// Get all clients
self.clients.matchAll().then(function (clientList) {
if (clientList.length > 0) {
return clientList[0].focus();
}
if (action === 'go-in') {
return self.clients.openWindow('https://app.nihaoshijie.com.cn/index.html#/mypage'); }})); e.notification.close(); });Copy the code
In the self. Registration. ShowNotification (), an action, we pass here corresponds to a message pop-up, there are options to choose: the PC:
On the mobile end:
According to the offline APP we talked about in the last section, the conditions for receiving message notification are not limited to the fact that you have to open the APP. After verification, you can also receive push message and call even if the APP is closedself.clients.openWindow()
You can call up the APP. You can see the process below:
Finally, our sw-push.js needs to be combined in offline-plugin. Finally, there is only one sw.js for the Service Worker. Modify the code in vue.config.js as follows:
. ServiceWorker: {events: true, // Push event logic in another file entry: './public/sw-push.js'},...Copy the code
conclusion
This article focuses on PWA and how to turn a SPA project into a PWA application. Related knowledge:
- The concept of PWA application and characteristics of PWA application.
- Service Worker compatibility and basic concepts such as life cycles and events.
- The concepts and basic flow of Web Push.
- Manifest.json file for the various configuration items.
- intercept
fetch
Event, caching front-end static resource files. - With offline-plugin, the project was transformed into a PWA application.
- Use Web Push in Node.js and Push it to the front end.
- API for Web Notification and associated configuration to prompt messages.
This article transformed SPA project complete source code address: Github, more about PWA technology can pay attention to “HTML5 mobile Web + vue.js application development actual combat”, the new book has just been published thank you for your support!
Code words are not easy. If this article helped you, give it a thumbs up!