اگر از اتاقی از توسعهدهندگان بپرسید که چه ویژگیهای دستگاه تلفن همراه در وب وجود ندارد، اعلانهای فشاری همیشه در لیست بالا هستند.
اعلانهای فشاری به کاربران شما این امکان را میدهد که بهروزرسانیهای بهموقع سایتهایی را که دوست دارند انتخاب کنند و به شما این امکان را میدهند که دوباره آنها را با محتوای سفارشیسازیشده و جذاب جذب کنید.
از نسخه 42 کروم، Push API و Notification API برای توسعه دهندگان در دسترس است.
Push API در کروم به چند تکه فناوری مختلف متکی است، از جمله مانیفستهای برنامههای وب و کارکنان خدمات . در این پست ما به هر یک از این فناوریها نگاه میکنیم، اما تنها حداقل آن برای راهاندازی و اجرای پیامهای فشاری. برای درک بهتر برخی از ویژگیهای دیگر مانیفستها و قابلیتهای آفلاین کارکنان خدمات، لطفاً پیوندهای بالا را بررسی کنید.
همچنین به مواردی که در نسخههای بعدی کروم به API اضافه میشود نگاه میکنیم و در نهایت یک پرسش متداول خواهیم داشت.
پیاده سازی Push Messaging برای کروم
این بخش هر مرحله ای را که باید برای پشتیبانی از پیام رسانی فشاری در برنامه وب خود تکمیل کنید، توضیح می دهد.
ثبت نام کارگر خدماتی
وابستگی داشتن یک سرویس دهنده برای پیاده سازی پیام های فشار برای وب وجود دارد. دلیل این امر این است که وقتی یک پیام فشار دریافت میشود، مرورگر میتواند یک سرویسکار را راهاندازی کند که در پسزمینه بدون باز شدن صفحه اجرا میشود و یک رویداد را ارسال میکند تا بتوانید تصمیم بگیرید که چگونه آن پیام فشار را مدیریت کنید.
در زیر نمونه ای از نحوه ثبت یک سرویس دهنده در برنامه وب خود آورده شده است. هنگامی که ثبت نام با موفقیت به پایان رسید، ابتدا () را فراخوانی می کنیم که به زودی به آن خواهیم پرداخت.
var isPushEnabled = false;
…
window.addEventListener('load', function() {
var pushButton = document.querySelector('.js-push-button');
pushButton.addEventListener('click', function() {
if (isPushEnabled) {
unsubscribe();
} else {
subscribe();
}
});
// Check that service workers are supported, if so, progressively
// enhance and add push messaging support, otherwise continue without it.
if ('serviceWorker' in navigator) {
navigator.serviceWorker.register('/service-worker.js')
.then(initialiseState);
} else {
console.warn('Service workers aren\'t supported in this browser.');
}
});
کنترل کننده کلیک دکمه، کاربر را برای ارسال پیام ها مشترک یا لغو اشتراک می کند. isPushEnabled یک متغیر جهانی است که به سادگی ردیابی می کند که آیا پیام های فشاری در حال حاضر مشترک هستند یا خیر. اینها در سراسر قطعه کد ارجاع خواهند شد.
سپس قبل از ثبت فایل service-worker.js
که منطق مدیریت یک پیام فشار را دارد، بررسی میکنیم که سرویسکاران پشتیبانی میشوند. در اینجا ما به سادگی به مرورگر می گوییم که این فایل جاوا اسکریپت سرویس دهنده سایت ما است.
حالت اولیه را تنظیم کنید
پس از ثبت نام سرویسکار، باید وضعیت رابط کاربری خود را تنظیم کنیم.
کاربران انتظار دارند که یک رابط کاربری ساده برای فعال یا غیرفعال کردن پیامهای فشاری برای سایت شما، و انتظار دارند که هر تغییری که رخ میدهد بهروز باشد. به عبارت دیگر، اگر آنها پیامهای فشاری را برای سایت شما فعال میکنند، آن را ترک میکنند و یک هفته بعد برمیگردند، رابط کاربری شما باید مشخص کند که پیامهای فشار قبلاً فعال شدهاند.
شما می توانید برخی از دستورالعمل های UX را در این سند بیابید، در این مقاله ما بر جنبه های فنی تمرکز خواهیم کرد.
در این مرحله ممکن است فکر کنید فقط دو حالت وجود دارد، فعال یا غیرفعال. با این حال، وضعیتهای دیگری پیرامون اعلانها وجود دارد که باید آنها را در نظر بگیرید.
تعدادی API وجود دارد که باید قبل از فعال کردن دکمه خود بررسی کنیم، و اگر همه چیز پشتیبانی میشود، میتوانیم رابط کاربری خود را فعال کرده و وضعیت اولیه را طوری تنظیم کنیم که نشان دهد پیامهای فشاری مشترک است یا خیر.
از آنجایی که اکثر این بررسی ها منجر به غیرفعال شدن رابط کاربری ما می شود، باید حالت اولیه را روی غیرفعال تنظیم کنید. همچنین اگر مشکلی در جاوا اسکریپت صفحه شما وجود داشته باشد، به عنوان مثال فایل JS قابل دانلود نیست یا کاربر جاوا اسکریپت را غیرفعال کرده است، از هرگونه سردرگمی جلوگیری می کند.
<button class="js-push-button" disabled>
Enable Push Messages
</button>
با این حالت اولیه، میتوانیم بررسیهای ذکر شده در بالا را در متد initialiseState()
انجام دهیم، یعنی پس از ثبت نام سرویسکار ما.
// Once the service worker is registered set the initial state
function initialiseState() {
// Are Notifications supported in the service worker?
if (!('showNotification' in ServiceWorkerRegistration.prototype)) {
console.warn('Notifications aren\'t supported.');
return;
}
// Check the current Notification permission.
// If its denied, it's a permanent block until the
// user changes the permission
if (Notification.permission === 'denied') {
console.warn('The user has blocked notifications.');
return;
}
// Check if push messaging is supported
if (!('PushManager' in window)) {
console.warn('Push messaging isn\'t supported.');
return;
}
// We need the service worker registration to check for a subscription
navigator.serviceWorker.ready.then(function(serviceWorkerRegistration) {
// Do we already have a push message subscription?
serviceWorkerRegistration.pushManager.getSubscription()
.then(function(subscription) {
// Enable any UI which subscribes / unsubscribes from
// push messages.
var pushButton = document.querySelector('.js-push-button');
pushButton.disabled = false;
if (!subscription) {
// We aren't subscribed to push, so set UI
// to allow the user to enable push
return;
}
// Keep your server in sync with the latest subscriptionId
sendSubscriptionToServer(subscription);
// Set your UI to show they have subscribed for
// push messages
pushButton.textContent = 'Disable Push Messages';
isPushEnabled = true;
})
.catch(function(err) {
console.warn('Error during getSubscription()', err);
});
});
}
مروری کوتاه بر این مراحل:
- بررسی میکنیم که
showNotification
در نمونه اولیه ServiceWorkerRegistration موجود است. بدون آن، ما نمیتوانیم در هنگام دریافت پیام فشار، اعلانی از کارمند خدمات خود نشان دهیم. - ما بررسی میکنیم که «مجوز
Notification.permission
چیست تا مطمئن شویم که"denied"
است. مجوز رد شده به این معنی است که تا زمانی که کاربر به صورت دستی مجوز را در مرورگر تغییر ندهد، نمیتوانید اعلانها را نشان دهید. - برای بررسی اینکه آیا پیامهای فشاری پشتیبانی میشوند، بررسی میکنیم که
PushManager
در شی پنجره موجود است. - در نهایت، ما از
pushManager.getSubscription()
استفاده کردیم تا بررسی کنیم که آیا قبلا اشتراکی داریم یا خیر. اگر این کار را انجام دهیم، جزئیات اشتراک را به سرور خود ارسال می کنیم تا اطمینان حاصل کنیم که اطلاعات درستی داریم و رابط کاربری خود را طوری تنظیم می کنیم که نشان دهد پیام های فشار قبلاً فعال شده است یا خیر. بعداً در این مقاله به جزئیات موجود در شیء اشتراک خواهیم پرداخت.
ما منتظر می مانیم تا navigator.serviceWorker.ready
برای بررسی اشتراک و فعال کردن دکمه فشاری حل شود زیرا فقط پس از فعال شدن سرویس دهنده است که می توانید در پیام های فشاری مشترک شوید.
مرحله بعدی رسیدگی به زمانی است که کاربر میخواهد پیامهای فشاری را فعال کند، اما قبل از انجام این کار، باید یک پروژه کنسول برنامهنویس Google راهاندازی کنیم و برخی پارامترها را به مانیفست خود اضافه کنیم تا از Firebase Cloud Messaging (FCM) استفاده کنیم. به عنوان Google Cloud Messaging (GCM).
یک پروژه در کنسول برنامه نویس Firebase بسازید
Chrome از FCM برای مدیریت ارسال و تحویل پیامهای فشار استفاده میکند. با این حال، برای استفاده از FCM API، باید یک پروژه در کنسول توسعه دهنده Firebase راه اندازی کنید.
مراحل زیر مختص کروم، اپرا برای اندروید و مرورگر سامسونگ است که از FCM استفاده می کنند. در ادامه مقاله در مورد نحوه عملکرد این کار در سایر مرورگرها بحث خواهیم کرد.
یک پروژه توسعه دهنده Firebase جدید ایجاد کنید
برای شروع باید یک پروژه جدید در https://console.firebase.google.com/ با کلیک بر روی «ایجاد پروژه جدید» ایجاد کنید.
یک نام پروژه اضافه کنید، پروژه را ایجاد کنید و به داشبورد پروژه هدایت خواهید شد:
از این داشبورد، روی چرخ دنده کنار نام پروژه خود در گوشه بالا سمت چپ کلیک کنید و روی «تنظیمات پروژه» کلیک کنید.
در صفحه تنظیمات، روی تب 'Cloud Messaging' کلیک کنید.
این صفحه حاوی کلید API برای پیامرسانی فشاری است که بعداً از آن استفاده خواهیم کرد و شناسه فرستنده که باید در مانیفست برنامه وب در بخش بعدی قرار دهیم.
یک مانیفست برنامه وب اضافه کنید
برای push، باید یک فایل مانیفست با فیلد gcm_sender_id اضافه کنیم تا اشتراک push موفق شود. این پارامتر فقط توسط Chrome، Opera for Android و Samsung Browser مورد نیاز است تا بتوانند از FCM / GCM استفاده کنند.
gcm_sender_id توسط این مرورگرها هنگامی که یک دستگاه کاربر را با FCM مشترک می کند استفاده می شود. این بدان معنی است که FCM می تواند دستگاه کاربر را شناسایی کند و مطمئن شود که شناسه فرستنده شما با کلید API مربوطه مطابقت دارد و کاربر به سرور شما اجازه داده است که برای آنها پیام های فشار ارسال کند.
در زیر یک فایل مانیفست فوق العاده ساده وجود دارد:
{
"name": "Push Demo",
"short_name": "Push Demo",
"icons": [{
"src": "images/icon-192x192.png",
"sizes": "192x192",
"type": "image/png"
}],
"start_url": "/index.html?homescreen=1",
"display": "standalone",
"gcm_sender_id": "<Your Sender ID Here>"
}
شما باید مقدار gcm_sender_id را روی شناسه فرستنده از پروژه Firebase خود تنظیم کنید.
هنگامی که فایل مانیفست خود را در پروژه خود ذخیره کردید (manifest.json نام خوبی است)، آن را از HTML خود با تگ زیر در سر صفحه خود ارجاع دهید.
<link rel="manifest" href="/manifest.json">
اگر مانیفست وب را با این پارامترها اضافه نکنید، هنگامی که میخواهید کاربر را برای ارسال پیامها مشترک کنید، با خطای "Registration failed - no sender id provided"
یا "Registration failed - permission denied"
استثنا خواهید داشت.
در Push Messaging مشترک شوید
اکنون که یک مانیفست تنظیم کرده اید، می توانید به جاوا اسکریپت سایت خود بازگردید.
برای اشتراک، باید متد subscribe() را در شی PushManager فراخوانی کنید، که از طریق ServiceWorkerRegistration به آن دسترسی دارید.
با این کار از کاربر میخواهد که به منبع شما اجازه ارسال اعلانهای فشاری بدهد. بدون این مجوز، نمی توانید با موفقیت مشترک شوید.
اگر وعده بازگشتی توسط متد subscribe () حل شود، یک شی PushSubscription به شما داده می شود که حاوی یک نقطه پایانی است.
نقطه پایانی باید برای هر کاربر در سرور شما ذخیره شود، زیرا به آنها برای ارسال پیام های فشار در تاریخ بعدی نیاز خواهید داشت.
کد زیر کاربر را برای ارسال پیام های فشاری مشترک می کند:
function subscribe() {
// Disable the button so it can't be changed while
// we process the permission request
var pushButton = document.querySelector('.js-push-button');
pushButton.disabled = true;
navigator.serviceWorker.ready.then(function(serviceWorkerRegistration) {
serviceWorkerRegistration.pushManager.subscribe()
.then(function(subscription) {
// The subscription was successful
isPushEnabled = true;
pushButton.textContent = 'Disable Push Messages';
pushButton.disabled = false;
// TODO: Send the subscription.endpoint to your server
// and save it to send a push message at a later date
return sendSubscriptionToServer(subscription);
})
.catch(function(e) {
if (Notification.permission === 'denied') {
// The user denied the notification permission which
// means we failed to subscribe and the user will need
// to manually change the notification permission to
// subscribe to push messages
console.warn('Permission for Notifications was denied');
pushButton.disabled = true;
} else {
// A problem occurred with the subscription; common reasons
// include network errors, and lacking gcm_sender_id and/or
// gcm_user_visible_only in the manifest.
console.error('Unable to subscribe to push.', e);
pushButton.disabled = false;
pushButton.textContent = 'Enable Push Messages';
}
});
});
}
در این مرحله برنامه وب شما برای دریافت پیام فشار آماده است، اگرچه تا زمانی که شنونده رویداد push را به فایل سرویس کار خود اضافه نکنیم، هیچ اتفاقی نخواهد افتاد.
شنونده رویداد Push Worker Service
هنگامی که یک پیام فشار دریافت می شود (ما در مورد نحوه ارسال پیام فشار در بخش بعدی صحبت خواهیم کرد)، یک رویداد فشار در سرویس دهنده شما ارسال می شود، در این مرحله باید یک اعلان نمایش دهید.
self.addEventListener('push', function(event) {
console.log('Received a push message', event);
var title = 'Yay a message.';
var body = 'We have received a push message.';
var icon = '/images/icon-192x192.png';
var tag = 'simple-push-demo-notification-tag';
event.waitUntil(
self.registration.showNotification(title, {
body: body,
icon: icon,
tag: tag
})
);
});
این کد یک شنونده رویداد فشار را ثبت می کند و یک اعلان را با عنوان از پیش تعریف شده، متن متن، نماد و یک تگ اعلان نمایش می دهد. یکی از نکات ظریفی که باید با این مثال برجسته شود، متد event.waitUntil()
است. این روش یک وعده را می گیرد و طول عمر یک کنترل کننده رویداد را افزایش می دهد (یا می توان آن را به عنوان زنده نگه داشتن کارگر خدماتی در نظر گرفت)، تا زمانی که وعده تسویه شود. در این مورد، وعده به event.waitUntil
ارسال می شود Promise برگشتی از showNotification()
.
تگ اعلان به عنوان یک شناسه برای اعلان های منحصر به فرد عمل می کند. اگر دو پیغام فشار را به یک نقطه پایانی با یک تاخیر کوتاه بین آنها ارسال کنیم و اعلانها را با همان برچسب نمایش دهیم، مرورگر اولین اعلان را نمایش میدهد و با دریافت پیام فشار، اعلان دوم را جایگزین آن میکند.
اگر می خواهید چندین اعلان را به طور همزمان نشان دهید، از یک برچسب متفاوت یا بدون برچسب استفاده کنید. در ادامه این پست به نمونه کاملتری از نمایش اعلان نگاه خواهیم کرد. در حال حاضر، اجازه دهید همه چیز را ساده نگه داریم و ببینیم آیا ارسال یک پیام فشار این اعلان را نشان می دهد یا خیر.
ارسال یک پیام فشاری
ما در پیامهای فشار مشترک شدهایم و کارمند خدمات ما آماده است تا یک اعلان را نشان دهد، بنابراین زمان ارسال پیام فشار از طریق FCM فرا رسیده است.
این فقط برای مرورگرهایی که از FCM استفاده می کنند قابل اجرا است.
هنگامی که متغیر PushSubscription.endpoint
را به سرور خود ارسال می کنید، نقطه پایانی FCM ویژه است. این یک پارامتر در انتهای URL دارد که یک registration_id
است.
یک مثال پایانی این خواهد بود:
https://fcm.googleapis.com/fcm/send/APA91bHPffi8zclbIBDcToXN_LEpT6iA87pgR-J-MuuVVycM0SmptG-rXdCPKTM5pvKiHk2Ts-ukL1KV8exGOnurOAKdbvH9jcvg8h2gSi-zZJyToiiydjAJW6Fa9mE3_7vsNIgzF28KGspVmLUpMgYLBd1rxaVh-L4NDzD7HyTkhFOfwWiyVdKh__rEt15W9n2o6cZ8nxrP
آدرس FCM این است:
https://fcm.googleapis.com/fcm/send
registration_id
خواهد بود:
APA91bHPffi8zclbIBDcToXN_LEpT6iA87pgR-J-MuuVVycM0SmptG-rXdCPKTM5pvKiHk2Ts-ukL1KV8exGOnurOAKdbvH9jcvg8h2gSi-zZJyToiiydjAJW6Fa9mE3_7vsNIgzF28KGspVmLUpMgYLBd1rxaVh-L4NDzD7HyTkhFOfwWiyVdKh__rEt15W9n2o6cZ8nxrP
این مخصوص مرورگرهایی است که از FCM استفاده می کنند. در یک مرورگر معمولی شما به سادگی یک نقطه پایانی دریافت می کنید و آن نقطه پایانی را به روشی استاندارد فراخوانی می کنید و بدون توجه به URL کار می کند.
این به این معنی است که در سرور خود باید بررسی کنید که آیا نقطه پایانی مربوط به FCM است یا خیر، ثبت_id را استخراج کنید. برای انجام این کار در پایتون می توانید کارهایی مانند زیر انجام دهید:
if endpoint.startswith('https://fcm.googleapis.com/fcm/send'):
endpointParts = endpoint.split('/')
registrationId = endpointParts[len(endpointParts) - 1]
endpoint = 'https://fcm.googleapis.com/fcm/send'
پس از دریافت شناسه ثبت نام، می توانید با API FCM تماس بگیرید. می توانید اسناد مرجع را در FCM API در اینجا بیابید.
نکات کلیدی که هنگام تماس با FCM باید به خاطر بسپارید عبارتند از:
- یک عنوان مجوز با مقدار
key=<YOUR_API_KEY>
باید هنگام تماس با API تنظیم شود، جایی که<YOUR_API_KEY>
کلید API پروژه Firebase است.- کلید API توسط FCM برای یافتن شناسه فرستنده مناسب استفاده می شود، اطمینان حاصل شود که کاربر مجوز پروژه شما را داده است و در نهایت اطمینان حاصل می کند که آدرس IP سرور برای آن پروژه در لیست مجاز قرار دارد.
- یک هدر
Content-Type
مناسب برایapplication/json
یاapplication/x-www-form-urlencoded;charset=UTF-8
بسته به اینکه داده ها را به صورت JSON یا فرم ارسال کنید. - آرایهای از
registration_ids
- اینها شناسههای ثبت نام هستند که از نقاط پایانی کاربران خود استخراج میکنید.
لطفاً اسناد مربوط به نحوه ارسال پیام های فشاری از سرور خود را بررسی کنید ، اما برای بررسی سریع کارمند خدمات خود می توانید از cURL برای ارسال پیام فشار به مرورگر خود استفاده کنید.
<YOUR_API_KEY>
و <YOUR_REGISTRATION_ID>
در این دستور cURL با دستور خودتان و آن را از یک ترمینال اجرا کنید.
باید یک اعلان با شکوه ببینید:
curl --header "Authorization: key=<YOUR_API_KEY>" --header
"Content-Type: application/json" https://fcm.googleapis.com/fcm/send -d
"{\"registration_ids\":[\"<YOUR_REGISTRATION_ID>\"]}"
هنگام توسعه منطق باطن خود، به یاد داشته باشید که هدر و فرمت Authorization بدنه POST مختص نقطه پایانی FCM است، بنابراین تشخیص دهید که نقطه پایانی برای FCM است و به صورت مشروط هدر را اضافه کنید و بدنه POST را قالب بندی کنید. برای سایر مرورگرها (و امیدواریم Chrome در آینده) باید پروتکل Web Push را پیاده سازی کنید.
نقطه ضعف اجرای فعلی Push API در کروم این است که نمی توانید هیچ داده ای را با یک پیام فشار ارسال کنید. نه هیچی دلیل این امر این است که در اجرای آینده، دادههای payload قبل از ارسال به یک نقطه پایانی پیامرسان فشاری باید روی سرور شما رمزگذاری شوند. به این ترتیب نقطه پایانی، هر ارائه دهنده فشاری که باشد، نمی تواند محتوای پیام فشار را به راحتی مشاهده کند. این همچنین در برابر آسیبپذیریهای دیگر مانند اعتبار سنجی ضعیف گواهیهای HTTPS و حملات انسان در وسط بین سرور شما و ارائهدهنده فشار محافظت میکند. با این حال، این رمزگذاری هنوز پشتیبانی نمیشود، بنابراین در این مدت باید برای دریافت اطلاعات مورد نیاز برای تکمیل یک اعلان، واکشی انجام دهید.
یک مثال کامل تر Push Event
اعلانی که تاکنون دیدهایم بسیار ابتدایی است و تا آنجا که نمونهها پیش میروند، در پوشش یک مورد استفاده در دنیای واقعی بسیار ضعیف است.
در واقع، اکثر مردم می خواهند قبل از نمایش اعلان، اطلاعاتی را از سرور خود دریافت کنند. این ممکن است دادهای برای پر کردن عنوان و پیام اعلان با چیزی خاص باشد، یا یک قدم فراتر رفته و برخی از صفحات یا دادهها را در حافظه پنهان نگه میدارد تا زمانی که کاربر روی اعلان کلیک میکند، همه چیز بلافاصله پس از باز شدن مرورگر در دسترس باشد - حتی اگر شبکه در آن زمان در دسترس نیست
در کد زیر مقداری داده از یک API واکشی می کنیم، پاسخ را به یک شی تبدیل می کنیم و از آن برای پر کردن اعلان خود استفاده می کنیم.
self.addEventListener('push', function(event) {
// Since there is no payload data with the first version
// of push messages, we'll grab some data from
// an API and use it to populate a notification
event.waitUntil(
fetch(SOME_API_ENDPOINT).then(function(response) {
if (response.status !== 200) {
// Either show a message to the user explaining the error
// or enter a generic message and handle the
// onnotificationclick event to direct the user to a web page
console.log('Looks like there was a problem. Status Code: ' + response.status);
throw new Error();
}
// Examine the text in the response
return response.json().then(function(data) {
if (data.error || !data.notification) {
console.error('The API returned an error.', data.error);
throw new Error();
}
var title = data.notification.title;
var message = data.notification.message;
var icon = data.notification.icon;
var notificationTag = data.notification.tag;
return self.registration.showNotification(title, {
body: message,
icon: icon,
tag: notificationTag
});
});
}).catch(function(err) {
console.error('Unable to retrieve data', err);
var title = 'An error occurred';
var message = 'We were unable to get the information for this push message';
var icon = URL_TO_DEFAULT_ICON;
var notificationTag = 'notification-error';
return self.registration.showNotification(title, {
body: message,
icon: icon,
tag: notificationTag
});
})
);
});
ارزش دارد، یک بار دیگر تاکید کنیم که event.waitUntil()
یک قول میگیرد که منجر به قولی میشود که توسط showNotification()
بازگردانده میشود، به این معنی که شنونده رویداد ما تا زمانی که فراخوانی fetch()
ناهمزمان کامل نشود و اعلان انجام شود خارج نمیشود. نشان داده شده است.
متوجه خواهید شد که حتی زمانی که خطایی وجود دارد، اعلان نشان می دهیم. این به این دلیل است که اگر این کار را نکنیم، Chrome اعلان عمومی خود را نشان خواهد داد.
باز کردن URL زمانی که کاربر روی یک اعلان کلیک می کند
هنگامی که کاربر روی یک اعلان کلیک می کند، یک رویداد notificationclick
در سرویس کار شما ارسال می شود. در کنترلر خود، میتوانید اقدامات مناسب را انجام دهید، مانند فوکوس کردن یک برگه یا باز کردن یک پنجره با یک URL خاص:
self.addEventListener('notificationclick', function(event) {
console.log('On notification click: ', event.notification.tag);
// Android doesn't close the notification when you click on it
// See: http://crbug.com/463146
event.notification.close();
// This looks to see if the current is already open and
// focuses if it is
event.waitUntil(
clients.matchAll({
type: "window"
})
.then(function(clientList) {
for (var i = 0; i < clientList.length; i++) {
var client = clientList[i];
if (client.url == '/' && 'focus' in client)
return client.focus();
}
if (clients.openWindow) {
return clients.openWindow('/');
}
})
);
});
این مثال مرورگر را به ریشه اصلی سایت باز می کند، با تمرکز بر یک برگه با همان مبدا موجود در صورت وجود، و در غیر این صورت باز کردن یک برگه جدید.
در اینجا یک پست اختصاص داده شده است به برخی از کارهایی که می توانید با اعلان API انجام دهید .
لغو اشتراک دستگاه کاربر
شما مشترک دستگاه یک کاربر شده اید و آنها پیام های فشاری دریافت می کنند، اما چگونه می توانید اشتراک آنها را لغو کنید؟
مهمترین چیزهایی که برای لغو اشتراک یک دستگاه کاربر لازم است، فراخوانی متد unsubscribe()
در شی PushSubscription و حذف نقطه پایانی از سرورهای خود است (فقط برای اینکه پیامهای فشاری ارسال نکنید که میدانید دریافت نمیشوند). کد زیر دقیقاً این کار را انجام می دهد:
function unsubscribe() {
var pushButton = document.querySelector('.js-push-button');
pushButton.disabled = true;
navigator.serviceWorker.ready.then(function(serviceWorkerRegistration) {
// To unsubscribe from push messaging, you need get the
// subscription object, which you can call unsubscribe() on.
serviceWorkerRegistration.pushManager.getSubscription().then(
function(pushSubscription) {
// Check we have a subscription to unsubscribe
if (!pushSubscription) {
// No subscription object, so set the state
// to allow the user to subscribe to push
isPushEnabled = false;
pushButton.disabled = false;
pushButton.textContent = 'Enable Push Messages';
return;
}
var subscriptionId = pushSubscription.subscriptionId;
// TODO: Make a request to your server to remove
// the subscriptionId from your data store so you
// don't attempt to send them push messages anymore
// We have a subscription, so call unsubscribe on it
pushSubscription.unsubscribe().then(function(successful) {
pushButton.disabled = false;
pushButton.textContent = 'Enable Push Messages';
isPushEnabled = false;
}).catch(function(e) {
// We failed to unsubscribe, this can lead to
// an unusual state, so may be best to remove
// the users data from your data store and
// inform the user that you have done so
console.log('Unsubscription error: ', e);
pushButton.disabled = false;
pushButton.textContent = 'Enable Push Messages';
});
}).catch(function(e) {
console.error('Error thrown while unsubscribing from push messaging.', e);
});
});
}
به روز نگه داشتن اشتراک
اشتراک ها ممکن است بین FCM و سرور شما هماهنگ نباشد. اطمینان حاصل کنید که سرور شما بدنه پاسخ ارسال POST FCM API را تجزیه می کند error:NotRegistered
همانطور که در مستندات FCM توضیح داده شده canonical_id
.
همچنین ممکن است اشتراکها بین کارمند سرویس و سرور شما هماهنگ نباشد. برای مثال، پس از موفقیت در اشتراک/لغو اشتراک، یک اتصال شبکه ضعیف ممکن است مانع از بهروزرسانی سرورتان شود. یا ممکن است کاربر مجوز اعلانها را لغو کند، که باعث لغو اشتراک خودکار میشود. با بررسی نتایج serviceWorkerRegistration.pushManager.getSubscription()
به صورت دوره ای (مثلاً در بارگذاری صفحه) و همگام سازی آن با سرور، چنین مواردی را مدیریت کنید. همچنین اگر دیگر اشتراک ندارید، ممکن است بخواهید دوباره به طور خودکار مشترک شوید و Notification.permission == 'اعطا شده'.
در sendSubscriptionToServer()
باید در نظر بگیرید که چگونه درخواست های شبکه ناموفق را هنگام به روز رسانی endpoint
مدیریت می کنید. یک راه حل این است که وضعیت endpoint
را در یک کوکی ردیابی کنید تا مشخص شود آیا سرور شما به آخرین جزئیات نیاز دارد یا خیر.
تمام مراحل بالا منجر به اجرای کامل پیامرسانی فشاری در وب در Chrome 46 میشود. هنوز ویژگیهای مشخصی وجود دارد که کار را آسانتر میکند (مانند یک API استاندارد برای راهاندازی پیامهای فشار)، اما این نسخه به شما امکان میدهد شروع کنید. ایجاد پیامرسانی فشاری در برنامههای وب شما.
چگونه اپلیکیشن وب خود را اشکال زدایی کنیم
هنگام پیاده سازی پیام های فشار، اشکالات در یکی از دو مکان وجود دارند: صفحه شما یا کارمند خدمات شما.
اشکالات صفحه را می توان با استفاده از DevTools رفع اشکال کرد. برای اشکال زدایی مشکلات مربوط به سرویس کارگر، دو گزینه دارید:
- به chrome://inspect > Service Workers بروید. این نما اطلاعات زیادی به جز کارگران خدماتی که در حال حاضر در حال اجرا هستند ارائه نمی دهد.
- به chrome://serviceworker-internals بروید و از اینجا می توانید وضعیت سرویسکاران را مشاهده کنید و در صورت وجود خطاها را مشاهده کنید. این صفحه تا زمانی که DevTools مجموعه ای از ویژگی های مشابه را نداشته باشد موقت است.
یکی از بهترین راهنماییهایی که میتوانم به هر کسی که تازه وارد سرویسدهی میشود، ارائه کنم، استفاده از چک باکس به نام «باز کردن پنجره DevTools و توقف اجرای جاوا اسکریپت در راهاندازی سرویسکار برای اشکالزدایی» است. این چک باکس یک نقطه انفصال در شروع سرویسکار شما اضافه میکند و اجرای آن را متوقف میکند ، این به شما امکان میدهد اسکریپت سرویسکارگر خود را از سر بگیرید یا قدم بزنید و ببینید آیا با مشکلی مواجه شدید.
اگر به نظر میرسد بین FCM و رویداد فشار سرویسدهندهتان مشکلی وجود دارد، نمیتوانید کار زیادی برای رفع اشکال انجام دهید زیرا هیچ راهی برای مشاهده اینکه آیا Chrome چیزی دریافت کرده است یا خیر وجود ندارد. نکته کلیدی که باید اطمینان حاصل کرد این است که وقتی سرور شما یک تماس API برقرار می کند، پاسخ FCM موفقیت آمیز است. چیزی شبیه به این خواهد بود:
{"multicast_id":1234567890,"success":1,"failure":0,"canonical_ids":0,"results":[{"message_id":"0:1234567890"}]}
به "success": 1
پاسخ. اگر به جای آن مشکلی مشاهده کردید، نشان میدهد که چیزی در شناسه ثبت FCM درست نیست و پیام فشار به Chrome ارسال نمیشود.
اشکال زدایی کارگران سرویس در کروم برای اندروید
در حال حاضر اشکال زدایی کارگران سرویس در Chrome for Android واضح نیست. باید به chrome://inspect بروید، دستگاه خود را پیدا کنید و به دنبال موردی از فهرست با نام «Worker pid:...» بگردید که نشانی اینترنتی کارمند خدمات شما را دارد.
UX برای Push Notifications
تیم کروم سندی از بهترین روشها برای UX اعلانهای فشاری و همچنین سندی که برخی از موارد لبه هنگام کار با اعلانهای فشاری را پوشش میدهد، گردآوری کرده است.
آینده پیامرسانی فشاری در کروم و وب باز
این بخش به جزئیات کمی پیرامون برخی از بخش های خاص Chrome این پیاده سازی می پردازد که باید از آنها آگاه باشید و تفاوت آن با سایر پیاده سازی های مرورگر چیست.
پروتکل فشار وب و نقاط پایانی
زیبایی استاندارد Push API این است که شما باید بتوانید با اجرای پروتکل Web Push، نقطه پایانی را بگیرید، آنها را به سرور خود منتقل کنید و پیام های فشار ارسال کنید.
پروتکل Web Push استاندارد جدیدی است که ارائه دهندگان فشار می توانند آن را پیاده سازی کنند و به توسعه دهندگان این امکان را می دهد که نگران نباشند ارائه دهنده فشار کیست. ایده این است که با این کار نیازی به ثبت نام برای کلیدهای API و ارسال دادههای فرمتبندیشده خاص، مانند آنچه در FCM وجود دارد، نمیشود.
Chrome اولین مرورگری بود که Push API را پیاده سازی کرد و FCM از پروتکل Web Push پشتیبانی نمی کند، به همین دلیل است که Chrome به gcm_sender_id
نیاز دارد و شما باید از API استراحت برای FCM استفاده کنید.
هدف نهایی کروم حرکت به سمت استفاده از پروتکل فشار وب با کروم و FCM است.
تا آن زمان، باید نقطه پایانی "https://fcm.googleapis.com/fcm/send" را شناسایی کنید و آن را جدا از سایر نقاط پایانی مدیریت کنید، یعنی داده های بارگذاری را به روشی خاص قالب بندی کنید و کلید Authorization را اضافه کنید.
چگونه پروتکل Web Push را پیاده سازی کنیم؟
Firefox Nightly در حال حاضر روی push کار می کند و احتمالاً اولین مرورگری خواهد بود که پروتکل Web Push را پیاده سازی می کند.
سوالات متداول
مشخصات کجاست؟
https://slightlyoff.github.io/ServiceWorker/spec/service_worker/ https://w3c.github.io/push-api/ https://notifications.spec.whatwg.org/
اگر حضور من در وب منشأ چندگانه داشته باشد، یا اگر هم حضور در وب و هم حضور بومی داشته باشم، میتوانم از اعلانهای تکراری جلوگیری کنم؟
در حال حاضر راه حلی برای این موضوع وجود ندارد، اما می توانید پیشرفت را در Chromium دنبال کنید.
سناریوی ایده آل این است که نوعی شناسه برای یک دستگاه کاربر داشته باشید و سپس در سمت سرور، شناسه اشتراک برنامه بومی و برنامه وب را مطابقت دهید و تصمیم بگیرید که به کدام یک پیام فشار ارسال کنید. شما می توانید این کار را از طریق اندازه صفحه نمایش، مدل دستگاه، به اشتراک گذاری یک کلید ایجاد شده بین برنامه وب و برنامه بومی انجام دهید، اما هر رویکرد دارای جوانب مثبت و منفی است.
چرا به gcm_sender_id نیاز دارم؟
این مورد ضروری است تا Chrome، Opera for Android و مرورگر سامسونگ بتوانند از Firebase Cloud Messaging (FCM) API استفاده کنند. هدف استفاده از پروتکل Web Push زمانی است که استاندارد نهایی شد و FCM بتواند از آن پشتیبانی کند.
چرا از سوکت های وب یا رویدادهای ارسال شده توسط سرور (EventSource) استفاده نمی کنید؟
مزیت استفاده از پیام های فشاری این است که حتی اگر صفحه شما بسته شود، سرویس دهنده شما از خواب بیدار می شود و می تواند یک اعلان را نشان دهد. هنگامی که صفحه یا مرورگر بسته می شود اتصال Web Sockets و EventSource بسته می شود.
اگر به تحویل رویداد پسزمینه نیاز نداشته باشم، چه؟
اگر به تحویل پسزمینه نیاز ندارید، سوکتهای وب یک گزینه عالی هستند.
چه زمانی می توانم بدون نمایش اعلان ها (یعنی فشار پس زمینه بی صدا) از فشار استفاده کنم؟
هنوز هیچ جدول زمانی برای اینکه چه زمانی در دسترس خواهد بود وجود ندارد، اما قصد داریم همگامسازی پسزمینه را پیادهسازی کنیم و در حالی که تصمیمگیری نشده است یا مشخص نیست، بحث در مورد فعال کردن فشار بیصدا با همگامسازی پسزمینه وجود دارد.
چرا این به HTTPS نیاز دارد؟ چگونه می توانم در طول توسعه روی این موضوع کار کنم؟
کارکنان سرویس برای اطمینان از اینکه اسکریپت کارگر سرویس از مبدأ مورد نظر است و از یک حمله مرد میانی به وجود نیامده است، به منابع امن نیاز دارند. در حال حاضر، این به معنای استفاده از HTTPS در سایتهای زنده است، اگرچه لوکال هاست در طول توسعه کار خواهد کرد.
پشتیبانی مرورگر چگونه به نظر می رسد؟
کروم در نسخه پایدار خود پشتیبانی میکند و موزیلا روی فایرفاکس نایتلی کار میکند. برای اطلاعات بیشتر به اجرای اشکال Push API مراجعه کنید و میتوانید اجرای Notification آنها را در اینجا دنبال کنید.
آیا می توانم یک اعلان را بعد از یک دوره زمانی مشخص حذف کنم؟
در حال حاضر این امکان پذیر نیست، اما ما در حال برنامه ریزی برای اضافه کردن پشتیبانی برای دریافت لیستی از اعلان های قابل مشاهده در حال حاضر هستیم. اگر یک مورد استفاده برای تنظیم انقضا برای اعلان پس از نمایش داده ایجاد شده دارید، مایلیم بدانیم آن چیست، بنابراین لطفاً نظری را اضافه کنید و ما آن را به تیم Chrome بازگردانیم.
اگر فقط باید پس از یک بازه زمانی مشخص، ارسال اعلان فشاری به کاربر را متوقف کنید، و اهمیتی نمیدهید که اعلان تا چه مدت قابل مشاهده است، میتوانید از پارامتر time to live (ttl) FCM استفاده کنید، اینجا بیشتر بدانید .
محدودیتهای ارسال پیام فشاری در کروم چیست؟
در این پست چند محدودیت وجود دارد:
- استفاده کروم از CCM به عنوان یک سرویس فشار، تعدادی از الزامات اختصاصی را ایجاد می کند. ما با هم کار می کنیم تا ببینیم آیا می توان برخی از این موارد را در آینده برداشت.
- هنگام دریافت پیام فشار، باید یک اعلان نشان دهید.
- کروم روی دسکتاپ این هشدار را دارد که اگر کروم در حال اجرا نباشد، پیامهای فشاری دریافت نمیشوند. این با ChromeOS و Android که در آن پیامهای فشار همیشه دریافت میشوند متفاوت است.
آیا نباید از Permissions API استفاده کنیم؟
Permission API در کروم پیاده سازی شده است، اما لزوماً در همه مرورگرها در دسترس نیست. در اینجا می توانید اطلاعات بیشتری کسب کنید .
چرا وقتی روی اعلان کلیک میکنم، کروم برگه قبلی را باز نمیکند؟
این مشکل فقط صفحاتی را تحت تأثیر قرار می دهد که در حال حاضر توسط یک سرویس دهنده کنترل نمی شوند. در اینجا می توانید اطلاعات بیشتری کسب کنید .
اگر تا زمانی که دستگاه کاربر فشار را دریافت کرد، یک اعلان قدیمی باشد، چه؟
همیشه باید هنگام دریافت پیام فشار، یک اعلان نشان دهید. در سناریویی که میخواهید اعلان ارسال کنید اما فقط برای مدت زمان مشخصی مفید است، میتوانید از پارامتر "time_to_live" در CCM استفاده کنید تا در صورت سپری شدن زمان انقضا، FCM پیام فشار ارسال نکند.
جزئیات بیشتر را می توان در اینجا یافت .
اگر من 10 پیام فشار ارسال کنم اما بخواهم دستگاه فقط یک پیام دریافت کند چه اتفاقی می افتد؟
FCM دارای یک پارامتر «collapse_key» است که میتوانید از آن استفاده کنید تا به FCM بگویید هر پیام معلقی را که همان «collapse_key» را دارد، با پیام جدید جایگزین کند.