آموزشی که مفاهیم کارگر خدمات ترویجی را پوشش می دهد
نمای کلی
این آموزش مقدمه ای را با کارگران سرویس برنامه افزودنی Chrome ارائه می دهد. به عنوان بخشی از این آموزش، افزونه ای خواهید ساخت که به کاربران امکان می دهد با استفاده از omnibox به سرعت به صفحات مرجع Chrome API حرکت کنند. شما یاد خواهید گرفت که چگونه:
- سرویسکار خود را ثبت کنید و ماژولها را وارد کنید.
- کارمند سرویس برنامه افزودنی خود را اشکال زدایی کنید.
- مدیریت وضعیت و مدیریت رویدادها.
- رویدادهای دوره ای را فعال کنید.
- با اسکریپت های محتوا ارتباط برقرار کنید.
قبل از شروع
این راهنما فرض می کند که شما تجربه اولیه توسعه وب را دارید. برای آشنایی با توسعه برنامههای افزودنی، توصیه میکنیم برنامههای افزودنی 101 و Hello World را مرور کنید.
پسوند را بسازید
با ایجاد یک دایرکتوری جدید به نام quick-api-reference
برای نگهداری فایل های برنامه افزودنی شروع کنید یا کد منبع را از مخزن نمونه های GitHub دانلود کنید.
مرحله 1: ثبت نام کارگر خدمات
فایل مانیفست را در ریشه پروژه ایجاد کنید و کد زیر را اضافه کنید:
manifest.json:
{
"manifest_version": 3,
"name": "Open extension API reference",
"version": "1.0.0",
"icons": {
"16": "images/icon-16.png",
"128": "images/icon-128.png"
},
"background": {
"service_worker": "service-worker.js"
}
}
برنامه های افزودنی، سرویس دهنده خود را در مانیفست ثبت می کنند که فقط یک فایل جاوا اسکریپت را می گیرد. نیازی به فراخوانی navigator.serviceWorker.register()
نیست، مانند یک صفحه وب.
یک پوشه images
ایجاد کنید سپس آیکون ها را در آن دانلود کنید .
اولین مراحل آموزش زمان خواندن را بررسی کنید تا درباره متادیتاها و نمادهای برنامه افزودنی در مانیفست بیشتر بدانید.
مرحله 2: چندین ماژول سرویس کار را وارد کنید
کارگر خدمات ما دو ویژگی را پیاده سازی می کند. برای نگهداری بهتر، هر ویژگی را در یک ماژول جداگانه پیاده سازی می کنیم. ابتدا باید سرویسکار را بهعنوان یک ماژول ES در مانیفست خود اعلام کنیم، که به ما امکان میدهد ماژولها را در سرویسکار خود وارد کنیم:
manifest.json:
{
"background": {
"service_worker": "service-worker.js",
"type": "module"
},
}
فایل service-worker.js
را ایجاد کنید و دو ماژول را وارد کنید:
import './sw-omnibox.js';
import './sw-tips.js';
این فایل ها را ایجاد کنید و به هر کدام یک گزارش کنسول اضافه کنید.
sw-omnibox.js:
console.log("sw-omnibox.js");
sw-tips.js:
console.log("sw-tips.js");
برای آشنایی با روشهای دیگر وارد کردن چندین فایل در یک سرویسدهنده، وارد کردن اسکریپتها را ببینید.
اختیاری: اشکال زدایی کارگر سرویس
من توضیح خواهم داد که چگونه گزارش کارگر سرویس را پیدا کنم و بدانم چه زمانی خاتمه یافته است. ابتدا دستورالعملهای بارگیری یک برنامه افزودنی بدون بستهبندی را دنبال کنید.
پس از 30 ثانیه، "سرویسکار (غیرفعال)" را مشاهده خواهید کرد که به معنای پایان کار است. برای بررسی روی پیوند "کارگر خدمات (غیرفعال)" کلیک کنید. انیمیشن زیر این را نشان می دهد.
آیا متوجه شده اید که بازرسی کارگر سرویس آن را بیدار کرده است؟ باز کردن Service Worker در Devtools آن را فعال نگه می دارد. برای اطمینان از اینکه برنامه افزودنی شما در هنگام پایان کار سرویس دهنده شما به درستی رفتار می کند، به یاد داشته باشید که DevTools را ببندید.
اکنون، برنامه افزودنی را بشکنید تا بدانید کجا باید خطاها را پیدا کنید. یکی از راههای انجام این کار، حذف «.js» از './sw-omnibox.js'
در فایل service-worker.js
است. Chrome نمیتواند کارمند سرویس را ثبت کند.
به chrome://extensions برگردید و برنامه افزودنی را بازخوانی کنید. دو خطا خواهید دید:
Service worker registration failed. Status code: 3.
An unknown error occurred when fetching the script.
برای آشنایی بیشتر با روشهای اشکالزدایی کارمند خدمات افزودنی، به اشکالزدایی برنامههای افزودنی مراجعه کنید.
مرحله 4: حالت اولیه را تنظیم کنید
در صورت عدم نیاز، Chrome کارمندان سرویس را خاموش میکند. ما از chrome.storage
API برای تداوم وضعیت در جلسات کارگر سرویس استفاده میکنیم. برای دسترسی به فضای ذخیرهسازی، باید در مانیفست درخواست مجوز کنیم:
manifest.json:
{
...
"permissions": ["storage"],
}
ابتدا پیشنهادات پیش فرض را در حافظه ذخیره کنید. میتوانیم با گوش دادن به رویداد runtime.onInstalled()
وضعیت را زمانی که پسوند برای اولین بار نصب میشود مقداردهی کنیم:
sw-omnibox.js:
...
// Save default API suggestions
chrome.runtime.onInstalled.addListener(({ reason }) => {
if (reason === 'install') {
chrome.storage.local.set({
apiSuggestions: ['tabs', 'storage', 'scripting']
});
}
});
کارکنان سرویس دسترسی مستقیم به شی پنجره ندارند و بنابراین نمی توانند از window.localStorage
برای ذخیره مقادیر استفاده کنند. همچنین، کارگران خدماتی محیط های اجرایی کوتاه مدت هستند. آنها به طور مکرر در طول جلسه مرورگر کاربر خاتمه می یابند، که آنها را با متغیرهای جهانی ناسازگار می کند. در عوض، از chrome.storage.local
استفاده کنید که داده ها را در ماشین محلی ذخیره می کند.
مرحله 5: رویدادهای خود را ثبت کنید
همه شنوندگان رویداد باید به صورت ایستا در محدوده جهانی کارگر خدمات ثبت شوند. به عبارت دیگر، شنوندگان رویداد نباید در توابع ناهمگام قرار گیرند. به این ترتیب Chrome میتواند اطمینان حاصل کند که در صورت راهاندازی مجدد سرویسکار، همه کنترلکنندههای رویداد بازیابی میشوند.
در این مثال، میخواهیم از chrome.omnibox
API استفاده کنیم، اما ابتدا باید کلیدواژه omnibox را در مانیفست اعلام کنیم:
manifest.json:
{
...
"minimum_chrome_version": "102",
"omnibox": {
"keyword": "api"
},
}
اکنون شنوندگان رویداد omnibox را در سطح بالای اسکریپت ثبت کنید. وقتی کاربر کلمه کلیدی omnibox ( api
) را در نوار آدرس و سپس برگه یا فاصله وارد می کند، Chrome لیستی از پیشنهادات را بر اساس کلمات کلیدی موجود در فضای ذخیره سازی نمایش می دهد. رویداد onInputChanged()
که ورودی کاربر فعلی و یک شی suggestResult
را می گیرد، مسئول پر کردن این پیشنهادات است.
sw-omnibox.js:
...
const URL_CHROME_EXTENSIONS_DOC =
'https://developer.chrome.com/docs/extensions/reference/';
const NUMBER_OF_PREVIOUS_SEARCHES = 4;
// Display the suggestions after user starts typing
chrome.omnibox.onInputChanged.addListener(async (input, suggest) => {
await chrome.omnibox.setDefaultSuggestion({
description: 'Enter a Chrome API or choose from past searches'
});
const { apiSuggestions } = await chrome.storage.local.get('apiSuggestions');
const suggestions = apiSuggestions.map((api) => {
return { content: api, description: `Open chrome.${api} API` };
});
suggest(suggestions);
});
پس از اینکه کاربر پیشنهادی را انتخاب کرد، onInputEntered()
صفحه مرجع Chrome API مربوطه را باز می کند.
sw-omnibox.js:
...
// Open the reference page of the chosen API
chrome.omnibox.onInputEntered.addListener((input) => {
chrome.tabs.create({ url: URL_CHROME_EXTENSIONS_DOC + input });
// Save the latest keyword
updateHistory(input);
});
تابع updateHistory()
ورودی omnibox را می گیرد و آن را در storage.local
ذخیره می کند. به این ترتیب آخرین عبارت جستجو می تواند بعداً به عنوان پیشنهاد omnibox استفاده شود.
sw-omnibox.js:
...
async function updateHistory(input) {
const { apiSuggestions } = await chrome.storage.local.get('apiSuggestions');
apiSuggestions.unshift(input);
apiSuggestions.splice(NUMBER_OF_PREVIOUS_SEARCHES);
return chrome.storage.local.set({ apiSuggestions });
}
مرحله 6: یک رویداد تکرارشونده را تنظیم کنید
متدهای setTimeout()
یا setInterval()
معمولا برای انجام کارهای تاخیری یا دوره ای استفاده می شوند. با این حال، این APIها ممکن است از کار بیفتند، زیرا زمانبندیکننده تایمرها را با پایان کار سرویسدهنده لغو میکند. در عوض، برنامههای افزودنی میتوانند از API chrome.alarms
استفاده کنند.
با درخواست مجوز "alarms"
در مانیفست شروع کنید. علاوه بر این، برای واکشی نکات برنامه افزودنی از یک مکان میزبان راه دور، باید مجوز میزبان را درخواست کنید:
manifest.json:
{
...
"permissions": ["storage"],
"permissions": ["storage", "alarms"],
"host_permissions": ["https://extension-tips.glitch.me/*"],
}
برنامه افزودنی همه نکات را دریافت می کند، یکی را به صورت تصادفی انتخاب می کند و آن را در حافظه ذخیره می کند. ما یک زنگ هشدار ایجاد خواهیم کرد که یک بار در روز برای به روز رسانی نوک فعال می شود. وقتی Chrome را می بندید، هشدارها ذخیره نمی شوند. بنابراین باید بررسی کنیم که آلارم وجود دارد یا خیر و اگر وجود ندارد آن را ایجاد کنیم.
sw-tips.js:
// Fetch tip & save in storage
const updateTip = async () => {
const response = await fetch('https://extension-tips.glitch.me/tips.json');
const tips = await response.json();
const randomIndex = Math.floor(Math.random() * tips.length);
return chrome.storage.local.set({ tip: tips[randomIndex] });
};
const ALARM_NAME = 'tip';
// Check if alarm exists to avoid resetting the timer.
// The alarm might be removed when the browser session restarts.
async function createAlarm() {
const alarm = await chrome.alarms.get(ALARM_NAME);
if (typeof alarm === 'undefined') {
chrome.alarms.create(ALARM_NAME, {
delayInMinutes: 1,
periodInMinutes: 1440
});
updateTip();
}
}
createAlarm();
// Update tip once a day
chrome.alarms.onAlarm.addListener(updateTip);
مرحله 7: با سایر زمینه ها ارتباط برقرار کنید
برنامه های افزودنی از اسکریپت های محتوا برای خواندن و اصلاح محتوای صفحه استفاده می کنند. وقتی کاربر از صفحه مرجع Chrome API بازدید می کند، اسکریپت محتوای برنامه افزودنی صفحه را با نوک روز به روز می کند. پیامی می فرستد تا انعام روز را از کارمند خدمات درخواست کند.
با اعلام اسکریپت محتوا در مانیفست شروع کنید و الگوی تطابق مربوط به مستندات مرجع Chrome API را اضافه کنید.
manifest.json:
{
...
"content_scripts": [
{
"matches": ["https://developer.chrome.com/docs/extensions/reference/*"],
"js": ["content.js"]
}
]
}
یک فایل محتوای جدید ایجاد کنید. کد زیر پیامی را برای سرویسکار ارسال می کند و درخواست انعام می دهد. سپس، دکمه ای اضافه می کند که پاپاور حاوی نوک افزونه را باز می کند. این کد از پلتفرم وب جدید Popover API استفاده می کند.
content.js:
(async () => {
// Sends a message to the service worker and receives a tip in response
const { tip } = await chrome.runtime.sendMessage({ greeting: 'tip' });
const nav = document.querySelector('.upper-tabs > nav');
const tipWidget = createDomElement(`
<button type="button" popovertarget="tip-popover" popovertargetaction="show" style="padding: 0 12px; height: 36px;">
<span style="display: block; font: var(--devsite-link-font,500 14px/20px var(--devsite-primary-font-family));">Tip</span>
</button>
`);
const popover = createDomElement(
`<div id='tip-popover' popover style="margin: auto;">${tip}</div>`
);
document.body.append(popover);
nav.append(tipWidget);
})();
function createDomElement(html) {
const dom = new DOMParser().parseFromString(html, 'text/html');
return dom.body.firstElementChild;
}
مرحله آخر اضافه کردن یک کنترل کننده پیام به کارمند خدمات ما است که پاسخی را به اسکریپت محتوا با نکته روزانه ارسال می کند.
sw-tips.js:
...
// Send tip to content script via messaging
chrome.runtime.onMessage.addListener((message, sender, sendResponse) => {
if (message.greeting === 'tip') {
chrome.storage.local.get('tip').then(sendResponse);
return true;
}
});
تست کنید که کار می کند
بررسی کنید که ساختار فایل پروژه شما به شکل زیر باشد:
برنامه افزودنی خود را به صورت محلی بارگیری کنید
برای بارگیری یک برنامه افزودنی بدون بسته بندی در حالت توسعه دهنده، مراحل Hello world را دنبال کنید.
یک صفحه مرجع باز کنید
- کلمه کلیدی "api" را در نوار آدرس مرورگر وارد کنید.
- "tab" یا "space" را فشار دهید.
- نام کامل API را وارد کنید.
- یا از لیست جستجوهای گذشته انتخاب کنید
- صفحه جدیدی به صفحه مرجع Chrome API باز می شود.
باید به این شکل باشد:
نوک روز را باز کنید
روی دکمه Tip واقع در نوار پیمایش کلیک کنید تا نکته برنامه افزودنی باز شود.
🎯 پیشرفت های بالقوه
بر اساس آنچه امروز آموخته اید، سعی کنید یکی از موارد زیر را انجام دهید:
- روش دیگری برای اجرای پیشنهادات omnibox را کاوش کنید.
- مدال سفارشی خود را برای نمایش نکته برنامه افزودنی ایجاد کنید.
- یک صفحه اضافی به صفحات API مرجع برنامه های افزودنی وب MDN باز کنید.
به ساختن ادامه بده
به پایان رساندن این آموزش تبریک می گویم 🎉. با تکمیل سایر آموزش های مبتدی به ارتقای سطح مهارت های خود ادامه دهید:
پسوند | آنچه خواهید آموخت |
---|---|
زمان خواندن | برای درج یک عنصر در مجموعه خاصی از صفحات به طور خودکار. |
مدیر زبانه ها | برای ایجاد یک پنجره بازشو که برگه های مرورگر را مدیریت می کند. |
حالت فوکوس | برای اجرای کد در صفحه فعلی پس از کلیک بر روی اکشن افزونه. |
به کاوش ادامه دهید
برای ادامه مسیر یادگیری کارکنان خدمات توسعه، توصیه می کنیم مقالات زیر را بررسی کنید: