add service worker and push notifications
This commit is contained in:
		
							parent
							
								
									3fa9b39150
								
							
						
					
					
						commit
						09147cacea
					
				
					 4 changed files with 135 additions and 2 deletions
				
			
		|  | @ -17,16 +17,17 @@ import FollowRequests from '../components/follow_requests/follow_requests.vue' | ||||||
| import OAuthCallback from '../components/oauth_callback/oauth_callback.vue' | import OAuthCallback from '../components/oauth_callback/oauth_callback.vue' | ||||||
| import UserSearch from '../components/user_search/user_search.vue' | import UserSearch from '../components/user_search/user_search.vue' | ||||||
| 
 | 
 | ||||||
| const afterStoreSetup = ({store, i18n}) => { | const afterStoreSetup = ({ store, i18n }) => { | ||||||
|   window.fetch('/api/statusnet/config.json') |   window.fetch('/api/statusnet/config.json') | ||||||
|     .then((res) => res.json()) |     .then((res) => res.json()) | ||||||
|     .then((data) => { |     .then((data) => { | ||||||
|       const {name, closed: registrationClosed, textlimit, server} = data.site |       const { name, closed: registrationClosed, textlimit, server, vapidPublicKey } = data.site | ||||||
| 
 | 
 | ||||||
|       store.dispatch('setInstanceOption', { name: 'name', value: name }) |       store.dispatch('setInstanceOption', { name: 'name', value: name }) | ||||||
|       store.dispatch('setInstanceOption', { name: 'registrationOpen', value: (registrationClosed === '0') }) |       store.dispatch('setInstanceOption', { name: 'registrationOpen', value: (registrationClosed === '0') }) | ||||||
|       store.dispatch('setInstanceOption', { name: 'textlimit', value: parseInt(textlimit) }) |       store.dispatch('setInstanceOption', { name: 'textlimit', value: parseInt(textlimit) }) | ||||||
|       store.dispatch('setInstanceOption', { name: 'server', value: server }) |       store.dispatch('setInstanceOption', { name: 'server', value: server }) | ||||||
|  |       store.dispatch('setInstanceOption', { name: 'vapidPublicKey', value: vapidPublicKey }) | ||||||
| 
 | 
 | ||||||
|       var apiConfig = data.site.pleromafe |       var apiConfig = data.site.pleromafe | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,6 +2,8 @@ import backendInteractorService from '../services/backend_interactor_service/bac | ||||||
| import { compact, map, each, merge } from 'lodash' | import { compact, map, each, merge } from 'lodash' | ||||||
| import { set } from 'vue' | import { set } from 'vue' | ||||||
| 
 | 
 | ||||||
|  | import registerPushNotifications from '../services/push/push.js' | ||||||
|  | 
 | ||||||
| // TODO: Unify with mergeOrAdd in statuses.js
 | // TODO: Unify with mergeOrAdd in statuses.js
 | ||||||
| export const mergeOrAdd = (arr, obj, item) => { | export const mergeOrAdd = (arr, obj, item) => { | ||||||
|   if (!item) { return false } |   if (!item) { return false } | ||||||
|  | @ -125,6 +127,8 @@ const users = { | ||||||
|                   // Fetch our friends
 |                   // Fetch our friends
 | ||||||
|                   store.rootState.api.backendInteractor.fetchFriends({id: user.id}) |                   store.rootState.api.backendInteractor.fetchFriends({id: user.id}) | ||||||
|                     .then((friends) => commit('addNewUsers', friends)) |                     .then((friends) => commit('addNewUsers', friends)) | ||||||
|  | 
 | ||||||
|  |                   registerPushNotifications(store) | ||||||
|                 }) |                 }) | ||||||
|             } else { |             } else { | ||||||
|               // Authentication failed
 |               // Authentication failed
 | ||||||
|  |  | ||||||
							
								
								
									
										96
									
								
								src/services/push/push.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								src/services/push/push.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,96 @@ | ||||||
|  | 
 | ||||||
|  | function urlBase64ToUint8Array (base64String) { | ||||||
|  |   const padding = '='.repeat((4 - base64String.length % 4) % 4) | ||||||
|  |   const base64 = (base64String + padding) | ||||||
|  |     .replace(/-/g, '+') | ||||||
|  |     .replace(/_/g, '/') | ||||||
|  | 
 | ||||||
|  |   const rawData = window.atob(base64) | ||||||
|  |   return Uint8Array.from([...rawData].map((char) => char.charCodeAt(0))) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function isPushSupported () { | ||||||
|  |   return 'serviceWorker' in navigator && 'PushManager' in window | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function registerServiceWorker () { | ||||||
|  |   return navigator.serviceWorker.register('/static/sw.js') | ||||||
|  |     .then(function (registration) { | ||||||
|  |       console.log('Service worker successfully registered.') | ||||||
|  |       return registration | ||||||
|  |     }) | ||||||
|  |     .catch(function (err) { | ||||||
|  |       console.error('Unable to register service worker.', err) | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function askPermission () { | ||||||
|  |   return new Promise(function (resolve, reject) { | ||||||
|  |     if (!window.Notification) return resolve('Notifications disabled') | ||||||
|  | 
 | ||||||
|  |     const permissionResult = window.Notification.requestPermission(function (result) { | ||||||
|  |       resolve(result) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     if (permissionResult) permissionResult.then(resolve, reject) | ||||||
|  |   }).then(function (permissionResult) { | ||||||
|  |     if (permissionResult !== 'granted') { | ||||||
|  |       throw new Error('We weren\'t granted permission.') | ||||||
|  |     } | ||||||
|  |     return permissionResult | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function subscribe (registration, store) { | ||||||
|  |   const subscribeOptions = { | ||||||
|  |     userVisibleOnly: true, | ||||||
|  |     applicationServerKey: urlBase64ToUint8Array(store.rootState.instance.vapidPublicKey) | ||||||
|  |   } | ||||||
|  |   return registration.pushManager.subscribe(subscribeOptions) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function sendSubscriptionToBackEnd (subscription, store) { | ||||||
|  |   return window.fetch('/api/v1/push/subscription/', { | ||||||
|  |     method: 'POST', | ||||||
|  |     headers: { | ||||||
|  |       'Content-Type': 'application/json', | ||||||
|  |       'Authorization': `Bearer ${store.rootState.oauth.token}` | ||||||
|  |     }, | ||||||
|  |     body: JSON.stringify({ | ||||||
|  |       subscription, | ||||||
|  |       data: { | ||||||
|  |         alerts: { | ||||||
|  |           follow: true, | ||||||
|  |           favourite: true, | ||||||
|  |           mention: true, | ||||||
|  |           reblog: true | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |     .then(function (response) { | ||||||
|  |       if (!response.ok) { | ||||||
|  |         throw new Error('Bad status code from server.') | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       return response.json() | ||||||
|  |     }) | ||||||
|  |     .then(function (responseData) { | ||||||
|  |       if (!responseData.id) { | ||||||
|  |         throw new Error('Bad response from server.') | ||||||
|  |       } | ||||||
|  |       return responseData | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default function registerPushNotifications (store) { | ||||||
|  |   if (isPushSupported()) { | ||||||
|  |     registerServiceWorker() | ||||||
|  |       .then(function (registration) { | ||||||
|  |         return askPermission() | ||||||
|  |           .then(() => subscribe(registration, store)) | ||||||
|  |           .then((subscription) => sendSubscriptionToBackEnd(subscription, store)) | ||||||
|  |           .catch((e) => console.warn(`Failed to setup Web Push Notifications: ${e.message}`)) | ||||||
|  |       }) | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										32
									
								
								static/sw.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								static/sw.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,32 @@ | ||||||
|  | /* eslint-env serviceworker */ | ||||||
|  | self.addEventListener('push', function (event) { | ||||||
|  |   if (event.data) { | ||||||
|  |     const data = event.data.json() | ||||||
|  | 
 | ||||||
|  |     const promiseChain = clients.matchAll({ | ||||||
|  |       includeUncontrolled: true | ||||||
|  |     }).then(function (clientList) { | ||||||
|  |       const list = clientList.filter((item) => item.type === 'window') | ||||||
|  |       if (list.length) return | ||||||
|  |       return self.registration.showNotification(data.title, data) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     event.waitUntil(promiseChain) | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | self.addEventListener('notificationclick', function (event) { | ||||||
|  |   event.notification.close() | ||||||
|  | 
 | ||||||
|  |   event.waitUntil(clients.matchAll({ | ||||||
|  |     includeUncontrolled: true | ||||||
|  |   }).then(function (clientList) { | ||||||
|  |     const list = clientList.filter((item) => item.type === 'window') | ||||||
|  | 
 | ||||||
|  |     for (var i = 0; i < list.length; i++) { | ||||||
|  |       var client = list[i] | ||||||
|  |       if (client.url === '/' && 'focus' in client) { return client.focus() } | ||||||
|  |     } | ||||||
|  |     if (clients.openWindow) { return clients.openWindow('/') } | ||||||
|  |   })) | ||||||
|  | }) | ||||||
		Loading…
	
		Reference in a new issue
	
	 Egor Kislitsyn
						Egor Kislitsyn