Skip to content

Commit 517651e

Browse files
author
yangfan36
committed
webComponents
1 parent 0cf3738 commit 517651e

30 files changed

+891
-0
lines changed

ServiceWorker/example7/404.html

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8" />
5+
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
7+
<title>Intro to Service Workers</title>
8+
<!-- import google fonts -->
9+
<link rel="preconnect" href="https://fonts.gstatic.com" />
10+
<link
11+
href="https://fonts.googleapis.com/css2?family=Lato&family=Montserrat:wght@400;700&display=swap"
12+
rel="stylesheet"
13+
/>
14+
<link rel="stylesheet" href="css/main.css" />
15+
</head>
16+
<body>
17+
<header>
18+
<h1>Intro to Service Workers</h1>
19+
<h2>This is NOT the page you are looking for</h2>
20+
</header>
21+
<main>
22+
<h3>4-oh-4</h3>
23+
<p>Maybe you want to go back <a href="/">home</a>?</p>
24+
<p><img src="/img/distracted-boyfriend.jpg" /></p>
25+
</main>
26+
<script defer src="./js/app.js"></script>
27+
</body>
28+
</html>

ServiceWorker/example7/css/main.css

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
html {
2+
font-size: 20px;
3+
font-family: 'Montserrat', sans-serif;
4+
line-height: 1.5;
5+
background-color: #222;
6+
color: #eee;
7+
}
8+
body {
9+
min-height: 100vh;
10+
background-color: inherit;
11+
color: inherit;
12+
}
13+
header,
14+
main {
15+
margin: 1rem 2rem;
16+
}
17+
main {
18+
display: flex;
19+
flex-direction: row;
20+
flex-wrap: wrap;
21+
gap: 1rem;
22+
justify-content: space-around;
23+
}
24+
h1 {
25+
color: orangered;
26+
}
27+
h2 {
28+
color: orange;
29+
}
30+
31+
main img {
32+
/* max-width: 100%;
33+
min-width: 200px; */
34+
width: clamp(200px, 400px, 600px);
35+
}
1.04 MB
Loading
30.5 KB
Loading
Loading

ServiceWorker/example7/index.html

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8" />
5+
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
7+
<title>Intro to Service Workers</title>
8+
<!-- import google fonts -->
9+
<link rel="preconnect" href="https://fonts.gstatic.com" />
10+
<link
11+
href="https://fonts.googleapis.com/css2?family=Lato&family=Montserrat:wght@400;700&display=swap"
12+
rel="stylesheet"
13+
/>
14+
<link rel="stylesheet" href="css/main.css" />
15+
</head>
16+
<body>
17+
<header>
18+
<h1>Intro to Service Workers</h1>
19+
<h2>Handling 404 Errors and Offline Issues11</h2>
20+
</header>
21+
<main>
22+
<p>
23+
<a href="/nowhere.html">This link</a> takes you to a non-existant page.
24+
Same effect as being offline.
25+
</p>
26+
<p>
27+
<img src="/img/1011-800x600.jpg" alt="sample image 1" />
28+
Some water and a woman in a canoe.
29+
</p>
30+
<p>
31+
<img src="/img/1016-800x600.jpg" alt="sample image 2" />
32+
A beautiful desert landscape.
33+
</p>
34+
</main>
35+
<script defer src="./js/app.js"></script>
36+
</body>
37+
</html>

ServiceWorker/example7/js/app.js

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
const APP = {
2+
SW: null,
3+
init() {
4+
//called after DOMContentLoaded
5+
//register our service worker
6+
APP.registerSW();
7+
document.querySelector('h2').addEventListener('click', APP.addImage);
8+
},
9+
registerSW() {
10+
if ('serviceWorker' in navigator) {
11+
// Register a service worker hosted at the root of the site
12+
navigator.serviceWorker.register('/sw.js').then(
13+
(registration) => {
14+
APP.SW =
15+
registration.installing ||
16+
registration.waiting ||
17+
registration.active;
18+
},
19+
(error) => {
20+
console.log('Service worker registration failed:', error);
21+
}
22+
);
23+
} else {
24+
console.log('Service workers are not supported.');
25+
}
26+
},
27+
addImage(ev) {
28+
let img = document.createElement('img');
29+
let p = document.createElement('p');
30+
let main = document.querySelector('main');
31+
//handle the load and error events for the image
32+
img.addEventListener('load', (ev) => {
33+
p.append(img);
34+
main.insertBefore(p, main.querySelector('p'));
35+
});
36+
img.addEventListener('error', (err) => {
37+
//don't bother adding a broken image
38+
p.textContent = 'Sorry. Your new image cannot be found';
39+
main.insertBefore(p, main.querySelector('p'));
40+
});
41+
//try to load the image
42+
img.src = '/img/this-image-does-not-exist.jpg';
43+
img.alt = 'dynamically added image';
44+
},
45+
};
46+
47+
document.addEventListener('DOMContentLoaded', APP.init);

ServiceWorker/example7/package.json

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
{
2+
"name": "example2",
3+
"version": "1.0.0",
4+
"description": "",
5+
"main": "sw.js",
6+
"scripts": {
7+
"test": "echo \"Error: no test specified\" && exit 1"
8+
},
9+
"keywords": [],
10+
"author": "",
11+
"license": "ISC"
12+
}

ServiceWorker/example7/sw.js

+154
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,154 @@
1+
const version = 8;
2+
let staticName = `staticCache-${version}`;
3+
let dynamicName = `dynamicCache`;
4+
let imageName = `imageCache-${version}`;
5+
let options = {
6+
ignoreSearch: false,
7+
ignoreMethod: false,
8+
ignoreVary: false,
9+
};
10+
//starter html and css and js files
11+
let assets = ['/', '/index.html', '/css/main.css', '/js/app.js', '/404.html'];
12+
//starter images
13+
let imageAssets = ['/img/1011-800x600.jpg', '/img/distracted-boyfriend.png'];
14+
15+
self.addEventListener('install', (ev) => {
16+
// service worker has been installed.
17+
//Extendable Event
18+
console.log(`Version ${version} installed`);
19+
// build a cache
20+
ev.waitUntil(
21+
caches
22+
.open(staticName)
23+
.then((cache) => {
24+
cache.addAll(assets).then(
25+
() => {
26+
//addAll == fetch() + put()
27+
console.log(`${staticName} has been updated.`);
28+
},
29+
(err) => {
30+
console.warn(`failed to update ${staticName}.`);
31+
}
32+
);
33+
})
34+
.then(() => {
35+
caches.open(imageName).then((cache) => {
36+
cache.addAll(imageAssets).then(
37+
() => {
38+
console.log(`${imageName} has been updated.`);
39+
},
40+
(err) => {
41+
console.warn(`failed to update ${staticName}.`);
42+
}
43+
);
44+
});
45+
})
46+
);
47+
});
48+
49+
self.addEventListener('activate', (ev) => {
50+
// when the service worker has been activated to replace an old one.
51+
//Extendable Event
52+
console.log('activated');
53+
// delete old versions of caches.
54+
ev.waitUntil(
55+
caches.keys().then((keys) => {
56+
return Promise.all(
57+
keys
58+
.filter((key) => {
59+
if (key != staticName && key != imageName) {
60+
return true;
61+
}
62+
})
63+
.map((key) => caches.delete(key))
64+
).then((empties) => {
65+
//empties is an Array of boolean values.
66+
//one for each cache deleted
67+
});
68+
})
69+
);
70+
});
71+
72+
self.addEventListener('fetch', (ev) => {
73+
// Extendable Event.
74+
console.log(`MODE: ${ev.request.mode} for ${ev.request.url}`);
75+
76+
ev.respondWith(
77+
caches.match(ev.request).then((cacheRes) => {
78+
return (
79+
cacheRes ||
80+
Promise.resolve().then(() => {
81+
let opts = {
82+
mode: ev.request.mode, //cors, no-cors, same-origin, navigate
83+
cache: 'no-cache',
84+
};
85+
86+
if (!ev.request.url.startsWith(location.origin)) {
87+
//not on the same domain as my html file
88+
opts.mode = 'cors';
89+
opts.credentials = 'omit';
90+
}
91+
return fetch(ev.request.url, opts).then(
92+
(fetchResponse) => {
93+
//we got a response from the server.
94+
if (fetchResponse.ok) {
95+
return handleFetchResponse(fetchResponse, ev.request);
96+
}
97+
//not ok 404 error
98+
if (fetchResponse.status == 404) {
99+
if (ev.request.url.match(/\.html/i)) {
100+
return caches.open(staticName).then((cache) => {
101+
return cache.match('/404.html');
102+
});
103+
}
104+
if (
105+
ev.request.url.match(/\.jpg$/i) ||
106+
ev.request.url.match(/\.png$/i)
107+
) {
108+
return caches.open(imageName).then((cache) => {
109+
return cache.match('/img/distracted-boyfriend.png');
110+
});
111+
}
112+
}
113+
},
114+
(err) => {
115+
//this is the network failure
116+
//return the 404.html file if it is a request for an html file
117+
if (ev.request.url.match(/\.html/i)) {
118+
return caches.open(staticName).then((cache) => {
119+
return cache.match('/404.html');
120+
});
121+
}
122+
}
123+
);
124+
//.catch()
125+
})
126+
);
127+
}) //end of match().then()
128+
); //end of respondWith
129+
}); //end of fetch listener
130+
131+
const handleFetchResponse = (fetchResponse, request) => {
132+
let type = fetchResponse.headers.get('content-type');
133+
// console.log('handle request for', type, request.url);
134+
if (type && type.match(/^image\//i)) {
135+
//save the image in image cache
136+
console.log(`SAVE ${request.url} in image cache`);
137+
return caches.open(imageName).then((cache) => {
138+
cache.put(request, fetchResponse.clone());
139+
return fetchResponse;
140+
});
141+
} else {
142+
//save in dynamic cache - html, css, fonts, js, etc
143+
console.log(`SAVE ${request.url} in dynamic cache`);
144+
return caches.open(dynamicName).then((cache) => {
145+
cache.put(request, fetchResponse.clone());
146+
return fetchResponse;
147+
});
148+
}
149+
};
150+
151+
self.addEventListener('message', (ev) => {
152+
//message from web page ev.data.
153+
//Extendable Event
154+
});

webComponents/example1/bigbang.js

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
const template = document.createElement("template");
2+
3+
template.innerHTML = `
4+
<div>
5+
<h1>hello 我是测试</h1>
6+
<slot name="title">我是默认的插槽【标题】</slot>
7+
<slot name="list">我是默认的插槽【列表】</slot>
8+
</div>
9+
`;
10+
11+
class BigBang extends HTMLElement {
12+
constructor() {
13+
super();
14+
15+
const shadowRoot = this.attachShadow({ mode: "closed" }); // https://developer.mozilla.org/zh-CN/docs/Web/API/Element/attachShadow
16+
17+
// todo 不能自定义插入数据???
18+
// let div = document.createElement("div");
19+
// div.textContent = "Hello, BigBang!";
20+
// shadowRoot.append(div)
21+
22+
// 试试下方的方式
23+
24+
let clone = template.content.cloneNode(true);
25+
shadowRoot.append(clone);
26+
}
27+
}
28+
29+
customElements.define("big-bang", BigBang);

webComponents/example1/index.html

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8" />
5+
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
7+
<title>hello Web Components</title>
8+
<link rel="stylesheet" href="./main.css" />
9+
<script src="./main.js" type="module"></script>
10+
<script src="./bigbang.js" type="module"></script>
11+
</head>
12+
<body>
13+
<header>
14+
<h1>hello Web Components</h1>
15+
</header>
16+
<main>
17+
<h2>下面是组件展示:</h2>
18+
<hr/>
19+
<big-bang>
20+
<h2 slot="title">我是标题</h2>
21+
<ul slot="list">
22+
<li>测试1</li>
23+
<li>测试2</li>
24+
<li>测试3</li>
25+
</ul>
26+
</big-bang>
27+
</main>
28+
</body>
29+
</html>

0 commit comments

Comments
 (0)