Intro
해당 글은 manifest version 3을 기반으로 작성된 글입니다. 혹여 version 2를 이용하셨다면, version 2에서 version 3로 migration 하면서 제가 적어놓은 글이 있으니 그것을 참고 하시기 바랍니다.
Chrome Extension version migration from V2 to V3
1. chrome 확장 앱의 구성
chrome extension에서 manifest version이 3이 되어 이를 한 번 정리할 겸,
chrome 확장앱을 구성하는 component는 크게 5가지로 나눌 수 있습니다.
- Background scripts => 대게 event를 등록하는데 사용합니다. (bookmark 등록, message 등과 같은 기능)
- Content scripts => 현재 열려 있는 페이지를 기준으로 이들을 바꿀 수 있습니다.
- an options page => options page에 의해 제공되는 세부 동작을 usesr가 사용할 수 있도록 합니다.
- UI elements => 브라우저 우상단에 존재하는 아이콘 or 클릭 시 열리는 popup, 검색창, contextmenu 등과 관련된 요소를 관리합니다.
- various logic files => 추가적으로 사용할 logic 등을 포함하는 것이 가능합니다.
기본적으로 모든 요소는 HTML, CSS, Javascript를 이용해서 구성됩니다. 모든 확장 component가 필요로 되는 것은 아닙니다.
2. Manifest 만들기
모든 extension은 반드시 하나의 manifest 파일을 포함합니다. 이는 JSON 형식으로 되어있고, manifest.json이라는 이름으로 저장됩니다.
기본 형태는 다음 제시한 내용처럼 구성됩니다. required 부분에는 반드시 들어가야 하는 내용을 포함합니다.
또한, app을 구성함에 있어 거의 필수적으로 들어가야 하는 action과 icon과 같은 내용과 설명 등이 포함됩니다.
1{ 2 // required 3 "manifest_version": 3, 4 "name": "app test", 5 "version": "0.0.1", 6 7 // recommended 8 "action": {}, 9 "default_locale": "ko", 10 "description": "chrome extension test", 11 "icons": {} 12}
자세한 사항은 하단 링크를 참고해주세요.
3. Background script 구성하기
확장앱은 event에 기반을 둔 크롬 브라우저 환경을 향상 또는 변경 시키기 위한 프로그램이다. event(새로운 page로의 이동, 북마크 삭제, 탭 닫기, 등)는 browser에 의해서 등록되어집니다. 확장앱은 background (service worker) scripts를 이용하여 이러한 event를 모니터링하며, 특정 지시사항을 명시합니다.
background service worker는 필요에 의해 언제든지 load되고, 사용되지 않으면 unload됩니다.
- 확장앱이 최초로 설치되거나 업데이트 되었을 경우
- background page가 전송된 event를 들었을 경우
- 다른 script 또는 extension에서 message를 전송했을 경우
- 확장앱의 다른 view에서 runtime.getBackgroundPage를 호출한 경우
한번 load되면, service worker는 이것이 action을 수행하는 동안은 종료되지 않습니다. 따라서, service worker는 모든 view 그리고 message port가 닫힐 때까지 unload되지 않습니다.
view를 여는 것은 service worker를 불러오지는 않지만, 종료되는 것을 막을 수는 있습니다.
효율적인 background scripts는 event가 발동되기 까지는 정지상태로 존재하고, 이에 응한뒤에 종료된다.
등록
service worker를 등록하기 위해서는 manifest에 이를 명시해주어야 합니다.
아래 예제에서는 명시된 background.js 파일이 service_worker들의 main이 됩니다.
1{ 2 "manifest_version": 3, 3 ..., 4 "background": { 5 "service_worker": "background.js" 6 } 7}
구성하기
runtime.onInstalled event를 listen한다면, 확장앱 설치 시에 초기화가 가능합니다. 초기 상태를 정의할 때 이를 사용합니다.
아래 예시에서는 event를 등록하는 과정입니다. 여기서 유의해야 할 것은 event의 등록은 page의 시작 시에 동기적으로 모두 설치해주어야 한다는 것입니다. 만약, event가 발생했을 때, event를 등록하는 것과 같은 동작을 하기 위해서는 다른 방식을 이용해야 합니다.
1chrome.runtime.onInstalled.addListener(function() { 2 chrome.contextMenus.create({ 3 "id": "sampleContextMenu", 4 "title": "Sample Context Menu", 5 "contexts": ["selection"] 6 }); 7}); 8 9// This will run when a bookmark is created. 10chrome.bookmarks.onCreated.addListener(function() { 11 // do something 12});
추가적으로 요청을 filtering 하거나, trigger를 재등록하는 과정과 같은 내용은 하단 링크를 추가로 참고하기 바랍니다.
4. Content scripts 만들기
contents scripts는 web page에서 동작할 내용에 대한 내용을 담습니다. DOM을 사용하여, scripts는 현재 웹 페이지의 세부사항을 조회, 변경 또는 정보를 전달하는 것이 가능합니다.
contens scripts는 확장앱의 message 교환을 통해서 부모 확장앱에 의해 사용되는 chrome API에 접근하는 것이 가능합니다. 또한, URL을 통해서 확장앱의 파일에 접근하여, 이를 사용하는 것이 가능합니다. 기본적으로 i18n, storage, runtime(connect, getURL, id, onMessage, sendMessage, etc...) 과 같은 API에 바로 접근해서 사용하는 것이 가능합니다.
1// Code for displaying <extensionDir>/images/myimage.png: 2var imgURL = chrome.runtime.getURL("images/myimage.png"); 3document.getElementById("someImage").src = imgURL;
고립
다른 확장앱, 또는 page와 충돌을 막기 위해서 기본적으로 content script는 고립됩니다. (browser의 tab간에 서로 독립적인 것처럼)
예를 들어, 다음과 같은 코드가 있다고 가정합니다.
1<html> 2 <button id="mybutton">click me</button> 3 <script> 4 var greeting = "hello, "; 5 var button = document.getElementById("mybutton"); 6 button.person_name = "Bob"; 7 button.addEventListener("click", () => 8 alert(greeting + button.person_name + ".") 9 , false); 10 </script> 11</html>
여기에 content scripts를 이용해서 아래 코드를 inject한다면,
1var greeting = "hola, "; 2var button = document.getElementById("mybutton"); 3button.person_name = "Roberto"; 4button.addEventListener("click", () => 5 alert(greeting + button.person_name + ".") 6, false);
button을 클릭했을 때, 두 개의 alert 창을 만날 수 있습니다.
Inject scripts
content scripts는 3가지의 방법으로 삽입되어질 수 있습니다.
1. statically
manifest.json 파일에 정적으로 선언하면, 자동적으로 page가 setting될 때 실행됩니다. 이는 "content_scripts"라는 부분에 정의됩니다. 여기에는 javascript, css 등을 포함할 수 있습니다.
1{ 2"manifest_version": 3, 3... 4"content_scripts": [ 5 { 6 "matches": ["http://*.nytimes.com/*"], // 해당 injection을 수행할 URL을 명시합니다. 필수입력입니다. 7 "css": ["myStyles.css"], // 추가할 css파일 입니다. 8 "js": ["contentScript.js"] // 추가할 js파일 입니다. 9 } 10] 11}
2. dynamically
2021.04.01 시점에서는 아직 완전 제공하지는 않는 기능입니다.
host를 알지 못하거나 아는 host로 부터 script가 추가 또는 삭제될 필요가 있는 경우에 사용합니다.
chrome.scripting.registerContentScript(optionsObject, callback);
or
chrome.scripting.unregisterContentScript(idArray, callback);
3. programmatically
구체적인 상황 또는 event에 대한 반응으로 실행하기 원할 때 사용합니다.
이를 수행하기 위해서는, 해당 페이지에 대한 host의 permission이 필요합니다. 이는 확장앱의 host_permissions 부분 or 일시적으로 activeTab을 이용해서 승인을 받을 수 있습니다.
1{ 2 "manifest_version": 3, 3 ... 4 "permissions": [ 5 "activeTab" 6 ] 7}
1// 1. 파일 전체를 실행 주입 시키는 방법 2chrome.runtime.onMessage.addListener((message, callback) => { 3 if (message == "runContentScript"){ 4 chrome.scripting.executeScript({ 5 file: 'contentScript.js' 6 }); 7 } 8}); 9 10// 2. 특정 함수를 주입하는 방법 11function injectedFunction(color) { 12 document.body.style.backgroundColor = color; 13} 14 15chrome.runtime.onMessage.addListener((message, callback) => { 16 if (message == "changeColor"){ 17 chrome.scripting.executeScript({ 18 function: injectedFunction, 19 arguments: ['orange'] 20 }); 21 } 22});
추가적으로 matck 범위를 세부 정의하는 부분과 frame 및 rum time 시점 관련 사항은 아래 링크를 참조해주세요.
5. option Page 만들기
사용자에게 option을 선택할 수 있는 page를 customise하여 제공할 수 있습니다. 이는 chrome 확장앱을 관리할 수 있는 chrome://extensions에서 Detail을 눌렀을 때 보이는 option과 관련된 page입니다.
이는 필요에 따라 구현하는 것이 알맞기 때문에 링크만 달아두겠습니다.
6. UI elements 만들기
확장앱은 UI 요소를 몇 가지 제공하는 것이 가능합니다. 여기서는 일부만 소개합니다.
1. Badge
확장앱의 icon을 결정하거나, 활성화 / 비활성화 등을 구분할 때 사용됩니다. => 상세 내용은 하단 링크 참고
여기서는 icon을 설정하는 방법만 적습니다.
1{ 2 "manifest_version": 3, 3 ... 4 "icons": { 5 "16": "extension_icon16.png", // favicon 6 "32": "extension_icon32.png", // 관리창 Icon(window에서 가끔 요구함) 7 "48": "extension_icon48.png", // 관리창 Icon 8 "128": "extension_icon128.png" // chrome webstore Icon 9 } 10}
2. Popup
browser 창의 tooltip을 클릭 시에 보여주고 싶은 내용을 명시하는 것이 가능합니다.
1{ 2 "manifest_version": 3, 3 ... 4 "browser_action": { 5 "default_popup": "popup.html" 6 } 7 ... 8}
popup.html
1<html> 2 <head> 3 <title>Water Popup</title> 4 </head> 5 <body> 6 <img src='./stay_hydrated.png' id='hydrateImage'> 7 <button id='sampleSecond' value='0.1'>Sample Second</button> 8 <button id='15min' value='15'>15 Minutes</button> 9 <button id='30min' value='30'>30 Minutes</button> 10 <button id='cancelAlarm'>Cancel Alarm</button> 11 <script src="popup.js"></script> 12 </body> 13</html>
3. Contextmenu
우클릭 시에 나오는 상자에 추가 내용을 추가하는 것이 가능합니다.
1{ 2 "manifest_version": 3, 3 ... 4 "permissions": [ 5 "contextMenus" 6 ], 7 "background": { 8 "service_worker": "background.js" 9 } 10}
background.js
1const kLocales = { 2 'com.au': 'Australia', 3 'com.br': 'Brazil', 4 'ca': 'Canada', 5 'cn': 'China', 6 'fr': 'France', 7 'it': 'Italy', 8 'co.in': 'India', 9 'co.jp': 'Japan', 10 'com.ms': 'Mexico', 11 'ru': 'Russia', 12 'co.za': 'South Africa', 13 'co.uk': 'United Kingdom' 14}; 15 16chrome.runtime.onInstalled.addListener(function() { 17 for (let key of Object.keys(kLocales)) { 18 chrome.contextMenus.create({ 19 id: key, 20 title: kLocales[key], 21 type: 'normal', 22 contexts: ['selection'], 23 }); 24 } 25});
추가적으로, override page, command, 검색창 디자인 등 추가적인 요소를 보려면 아래 링크를 참조해주세요.
7. Boilerplate
만약, react, typescript에 익숙하다면, 필자가 만들어놓은 boilerplate를 추천합니다.
Comments