Skip to content

Commit e0a0c0f

Browse files
dashnowordssoyaine
authored andcommitted
merge doc 19 written by dashrun (soyaine#23)
* try to finish the work of day16 * modify some details to improve the performance of readme.md as asked * finish the work of day17 * finish the work of day18 * update the readme.md * update the main readme.md, add the demo link of day 17 * 修改了contributor部分 * update readme * Revert "update readme" This reverts commit 4711357. * handle the conflict * finish the work of day19 * update the readme.md file , fix the problem of space
1 parent 35c4212 commit e0a0c0f

File tree

7 files changed

+342
-1
lines changed

7 files changed

+342
-1
lines changed

19 - Webcam Fun/README.md

Lines changed: 111 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,111 @@
1+
# 19 Webcam Fun 中文指南
2+
3+
> 本篇作者:©[大史快跑Dashrun](https://github.com/dashrun)——Chinasoft Frontend Developer
4+
5+
> 简介:[JavaScript30](https://javascript30.com)[Wes Bos](https://github.com/wesbos) 推出的一个 30 天挑战。项目免费提供了 30 个视频教程、30 个挑战的起始文档和 30 个挑战解决方案源代码。目的是帮助人们用纯 JavaScript 来写东西,不借助框架和库,也不使用编译器和引用。现在你看到的是这系列指南的第 19 篇。完整指南在 [GitHub](https://github.com/soyaine/JavaScript30),喜欢请 Star 哦♪(^∇^*)
6+
7+
> 创建时间:2017-08-31
8+
最后更新:2017-09-02
9+
10+
## 挑战任务
11+
`index-start.html`中提供了一个名为**Take Photo**的按钮,该按钮的点击事件会触发`takePhoto()`函数,并提供了一组标有RGBmin/max标记的`range`类型`input`元素,一个`canvas`元素,一个`video`元素,以及带有`strip`类名的空`div`元素。
12+
本次的编程任务:
13+
1.通过编写javascript代码,请求调用用户的网络摄像头;
14+
2.在页面上展示来自webcam的数据流信息;
15+
3.并允许用户保存展示的照片;
16+
4.及使用滑块来改变图像的色彩。
17+
18+
## 实现效果
19+
![结果展示](https://github.com/dashrun/vanilla-javascript-30/blob/master/19%20-%20Webcam%20Fun/effects.png)
20+
21+
## 相关知识
22+
1.`window.navigator`对象
23+
`window.navigator`对象上有很多有趣的属性和方法,通过调用相应的方法可以查看到有关当前脚本运行环境(多为浏览器)的相关信息,并使用一些扩展功能。对此不熟悉的开发者可以浏览[MDN](https://developer.mozilla.org/zh-CN/docs/Web/API/Navigator)中相关API的介绍,例如使用`getBattery()`可以获取电池状态。
24+
2.`navigator.getUserMedia`方法
25+
`getUserMedia`方法为我们提供了访问网络摄像头或麦克风的权限,该方法接受一个对象作为参数,通过该对象即可获得来自多媒体设备的数据。
26+
3.`canvas`标签
27+
HTML5强大的扩展功能之一,提供了丰富的图像绘制方法,也是HTML可以作为游戏开发工具的原因之一,本次开发中使用`canvas.getContext('2d')`提供的图像操作方法.
28+
4.像素数组
29+
使用`canvas`绘图上下文中的`getImageData()`获得的像素信息为一个定型数组,每个像素点的rgba色彩信息分别存放在数组中,故数据的格式为:[第一像素r值,第一像素g值,第一像素b值,第一像素a值,第二像素r值,第二像素g值......],通过各类函数公式对像素数据进行处理即可获得各类不同的滤镜效果。
30+
31+
## 基本思路
32+
33+
1.调用`navigator.getUserMedia()`方法,若调用成功则返回数据流,若调用失败则在控制台打印相关信息;
34+
2.成功调用网络摄像头后,将返回的数据对象绑定至video标签的srcObject属性(注意此处getUserMedia()方法成功调用时触发的回调函数中会传递一个`stream`对象,该对象直接赋值给video.src是没有作用的),并当流数据开始传递时,视频自动播放;
35+
3.点击`takePhoto()`函数时调用`canvas`绘图上下文中的`drawImage()`方法将视频中当前帧的图像绘制在canvas上,该方法第一个参数可以为图像或视频,其余参数与绘图区域尺寸相关(该方法有多种调用模式,感兴趣的读者可自行学习);
36+
4.滤色:在全局中保存滤色范围的上下限,每次滑块数据发生改变后,使用`canvas`绘图上下文中的`getImageData()`获得画布上指定区域内各像素点的颜色数据,数据被保存在返回对象的data属性中,通过遍历修改像素色彩数组中的数据改变图像的表现,修改后调用`putImageData()`方法将像素点重新绘制在`canvas`上。
37+
5.点击`savePhoto()`函数时调用`canvas``toDataUrl()`方法即可获得canvas中的图像数据,默认格式为png,也可修改为其他格式,生成的图像数据指定给`img.src`时即可预览图片;
38+
6.在`img`标签外添加`a`标签,并为其添加`download`属性,当点击链接时,即可将生成的图片保存至本地。
39+
40+
## 过程指南
41+
1.申请调用WebCam
42+
```js
43+
function askWebcam() {
44+
navigator.getUserMedia = navigator.getUserMedia ||
45+
navigator.webkitGetUserMedia ||
46+
navigator.mozGetUserMedia;
47+
if (navigator.getUserMedia) {
48+
navigator.getUserMedia({
49+
audio: false,
50+
video: {
51+
width: 300,
52+
height: 200
53+
}
54+
}, function(stream) {
55+
//若成功
56+
video.srcObject = stream;
57+
video.onloadedmetadata = function(e) {
58+
video.play();
59+
}
60+
}, function(err) {
61+
console.log('Error occured:' + err.name);
62+
});
63+
} else {
64+
console.log('this navigator doesn\'t support webcam!');
65+
}
66+
}
67+
```
68+
2.截图函数 `takePhoto()`,此处将原始图像数据保留一份,否则使用滤色函数处理后,被滤掉的颜色无法恢复,保存了原始图像数据后,只需要重新绘制在canvas上即可。
69+
```js
70+
function takePhoto() {
71+
ctx.drawImage(video, 0, 0, 300, 200);
72+
//将原始截图保存,
73+
origindata = ctx.getImageData(0,0,300,200);
74+
}
75+
```
76+
3.色彩过滤
77+
在所有滑块的父元素上监听`change`事件,当滑块的值发生改变时,先通过e.target.name确定是哪个颜色的范围要求发生了变化,再访问e.target.value获得变化值,通过与全局变量`filter`作比较来获得滤色值的上下限要求。
78+
重点难点解释如下:
79+
```js
80+
/*startPos表示操作像素点数据时的起点,从canvas获取到的像素数据每四个值表示一个像素点例如滑块为红色,则只需要改变像素数组中第0,4,8......个元素的值。通过target.value的首字母即可判断滤色过程应该检查的颜色*/
81+
const startPos = {'r':0,'g':1,'b':2}[target.name[0]];
82+
/*filterMin和filterMax表示相应的滤色范围上下限,若修改了红色滤色范围则取红色范围值。若修改蓝色的滤色范围,则取蓝色。checkFilter()函数将改变后的值与滤色标准`filter`进行比较,将更改滤色标准后需要调整的颜色类别(r,g,b)对应的上下限返回给结果。*/
83+
var tempFilter = checkFilter(target.name, target.value);
84+
const filterMin = tempFilter.min;
85+
const filterMax = tempFilter.max;
86+
```
87+
4.保存图片
88+
使用`canvas.toDataUrl()`方法将`canvas`画布保存为图片,默认为png格式,该数据可作为`img.src`的值,也可利用`a`标签将其下载.
89+
```js
90+
//保存图片
91+
function savePhoto() {
92+
img.src = canvas.toDataURL();
93+
a.href = canvas.toDataURL();
94+
a.setAttribute('download', 'handsome');
95+
}
96+
```
97+
## 延伸思考
98+
####如何制作一个简易的钢铁侠面板
99+
1.本次的编程任务中我们已经可以调用网络摄像头获得视频数据;
100+
2.将webcam获取到的数据绘制在canvas中时,同时将可视化面板的UI绘制在canvas中;
101+
3.视频的本质是连续播放的图片,当图片播放的速度超过每秒24张时,我们将看到连续的动画,当我们以相应的频率重绘canvas上的图像时,即可看到合成后的视频;
102+
4.如果要获得更加逼真的效果,可以做更多的图像识别方面的扩展延伸,本篇只做抛砖引玉,不再做更深入的探究。
103+
伪代码片段:
104+
```js
105+
setInterval(drawImg(),1000/24);
106+
function drawImg(){
107+
//获得来自webcam的数据并将其绘制在canvas上
108+
109+
}
110+
```
111+

19 - Webcam Fun/effects.png

205 KB
Loading
Lines changed: 58 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8">
5+
<title>Get User Media Code Along!</title>
6+
<link rel="stylesheet" href="style.css">
7+
</head>
8+
<body>
9+
10+
<div class="photobooth">
11+
<div class="controls">
12+
<button onClick="takePhoto()">Take Photo</button>
13+
<button onClick="savePhoto()">Save Photo</button>
14+
<div class="rgb">
15+
<label for="rmin">Red Min:</label>
16+
<input type="range" min=0 max=255 name="rmin" value="0">
17+
<label for="rmax">Red Max:</label>
18+
<input type="range" min=0 max=255 name="rmax" value="255">
19+
20+
<br>
21+
22+
<label for="gmin">Green Min:</label>
23+
<input type="range" min=0 max=255 name="gmin" value="0">
24+
<label for="gmax">Green Max:</label>
25+
<input type="range" min=0 max=255 name="gmax" value="255">
26+
27+
<br>
28+
29+
<label for="bmin">Blue Min:</label>
30+
<input type="range" min=0 max=255 name="bmin" value="0">
31+
<label for="bmax">Blue Max:</label>
32+
<input type="range" min=0 max=255 name="bmax" value="255">
33+
</div>
34+
</div>
35+
<div class="hasimg">
36+
<p>照片区域:</p>
37+
<canvas class="photo" height="200" width="300"></canvas>
38+
</div>
39+
<div class="hasimg">
40+
<p>实时视频区域:</p>
41+
<video class="player"></video>
42+
</div>
43+
<div class="hasimg">
44+
<p>变色画布区域:</p>
45+
<div class="strip">
46+
<a href="">
47+
<img alt="暂未保存图像" id='myimg'>
48+
</a>
49+
</div>
50+
</div>
51+
</div>
52+
53+
<audio class="snap" src="http://wesbos.com/demos/photobooth/snap.mp3" hidden></audio>
54+
55+
<script src="script.js"></script>
56+
57+
</body>
58+
</html>

19 - Webcam Fun/index-start.html

Lines changed: 45 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8">
5+
<title>Get User Media Code Along!</title>
6+
<link rel="stylesheet" href="style.css">
7+
</head>
8+
<body>
9+
10+
<div class="photobooth">
11+
<div class="controls">
12+
<button onClick="takePhoto()">Take Photo</button>
13+
<!-- <div class="rgb">
14+
<label for="rmin">Red Min:</label>
15+
<input type="range" min=0 max=255 name="rmin">
16+
<label for="rmax">Red Max:</label>
17+
<input type="range" min=0 max=255 name="rmax">
18+
19+
<br>
20+
21+
<label for="gmin">Green Min:</label>
22+
<input type="range" min=0 max=255 name="gmin">
23+
<label for="gmax">Green Max:</label>
24+
<input type="range" min=0 max=255 name="gmax">
25+
26+
<br>
27+
28+
<label for="bmin">Blue Min:</label>
29+
<input type="range" min=0 max=255 name="bmin">
30+
<label for="bmax">Blue Max:</label>
31+
<input type="range" min=0 max=255 name="bmax">
32+
</div> -->
33+
</div>
34+
35+
<canvas class="photo"></canvas>
36+
<video class="player"></video>
37+
<div class="strip"></div>
38+
</div>
39+
40+
<audio class="snap" src="http://wesbos.com/demos/photobooth/snap.mp3" hidden></audio>
41+
42+
<script src="script.js"></script>
43+
44+
</body>
45+
</html>

19 - Webcam Fun/script.js

Lines changed: 108 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
window.onload = function() {
2+
canvas = document.querySelector('canvas');
3+
video = document.querySelector('video');
4+
ctx = canvas.getContext('2d');
5+
img = document.querySelector('#myimg');
6+
slider = document.querySelector('.rgb');
7+
a = document.querySelector('a');
8+
//滤色范围记录
9+
filter = {
10+
rmin:0,
11+
rmax:255,
12+
gmin:0,
13+
gmax:255,
14+
bmin:0,
15+
bmax:255
16+
}
17+
18+
//调用摄像头数据
19+
askWebcam();
20+
21+
//绑定change事件动态修改图片颜色
22+
slider.onchange = function(e) {
23+
//先将canvas恢复至原始截图
24+
ctx.putImageData(origindata,0,0);
25+
const target = e.target;
26+
//startPos表示操作像素点数据时的起点,从canvas获取到的像素数据每四个值表示一个像素点
27+
//例如滑块为红色,则只需要改变像素数组中第0,4,8......个元素的值。
28+
const startPos = {'r':0,'g':1,'b':2}[target.name[0]];
29+
//filterMin和filterMax表示相应的滤色范围上下限,若修改了红色滤色范围则取红色范围值。
30+
//若修改蓝色的滤色范围,则取蓝色。
31+
var tempFilter = checkFilter(target.name, target.value);
32+
const filterMin = tempFilter.min;
33+
const filterMax = tempFilter.max;
34+
//从canvas获取像素数据
35+
var img = ctx.getImageData(0,0,300,200);
36+
37+
38+
var imgd = img.data;
39+
//色彩过滤
40+
for(var i=startPos, len = imgd.length; i<len ; i+=4){
41+
if (imgd[i]< filterMin){
42+
imgd[i] = filterMin;
43+
}else if(imgd[i] >filterMax){
44+
imgd[i] = filterMax;
45+
}
46+
}
47+
//将修改后的像素数据重绘制至canvas
48+
ctx.putImageData(img,0,0);
49+
img.src = canvas.toDataURL();
50+
}
51+
52+
}
53+
54+
//点击函数
55+
function takePhoto() {
56+
ctx.drawImage(video, 0, 0, 300, 200);
57+
//将原始截图保存
58+
origindata = ctx.getImageData(0,0,300,200);
59+
}
60+
61+
//保存图片
62+
function savePhoto() {
63+
img.src = canvas.toDataURL();
64+
a.href = canvas.toDataURL();
65+
a.setAttribute('download', 'handsome');
66+
}
67+
68+
//申请网络摄像头操作权限
69+
function askWebcam() {
70+
navigator.getUserMedia = navigator.getUserMedia ||
71+
navigator.webkitGetUserMedia ||
72+
navigator.mozGetUserMedia;
73+
if (navigator.getUserMedia) {
74+
navigator.getUserMedia({
75+
audio: false,
76+
video: {
77+
width: 300,
78+
height: 200
79+
}
80+
}, function(stream) {
81+
//若成功
82+
video.srcObject = stream;
83+
video.onloadedmetadata = function(e) {
84+
video.play();
85+
}
86+
}, function(err) {
87+
console.log('Error occured:' + err.name);
88+
});
89+
} else {
90+
console.log('this navigator doesn\'t support webcam!');
91+
}
92+
}
93+
94+
//滤色函数
95+
function checkFilter(name, value){
96+
var _min;
97+
var _max;
98+
var _antiname = {'rmin':'rmax','rmax':'rmin','gmin':'gmax','gmax':'gmin','bmin':'bmax','bmax':'bmin'}[name]
99+
filter[name] = value;
100+
//当下限值超过上限时,将两者对调
101+
_min = Math.min(filter[name],filter[_antiname]);
102+
_max = Math.max(filter[name],filter[_antiname]);
103+
console.log(filter);
104+
return {
105+
min: _min,
106+
max: _max
107+
}
108+
}

19 - Webcam Fun/style.css

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
html,body{
2+
padding: 0;
3+
margin: 0;
4+
height: 100%;
5+
width: 100%;
6+
}
7+
.player{
8+
height:200px;
9+
width:300px;
10+
}
11+
.hasimg{
12+
display:inline-block;
13+
vertical-align: top;
14+
}
15+
.strip{
16+
height:200px;
17+
width:300px;
18+
}

README.md

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ No | Guide | Demo
7474
参加挑战并不需要你缴纳费用或是加入什么组织,也不会有人催着你去做什么,你只需要打开电脑,然后开始思考、敲击键盘。相信内在动机的力量,我在这里给出了一些建议和心得,最适合你的方法还需要你自己去摸索。
7575

7676
## 本中文指南贡献者名单
77-
77+
7878
Name | Contribution
7979
--- | ---
8080
[@DrakeXiang](https://github.com/DrakeXiang) | No.[11](https://github.com/soyaine/JavaScript30/tree/master/11%20-%20Custom%20Video%20Player)
@@ -85,3 +85,4 @@ Name | Contribution
8585

8686
## JOIN US
8787
如果对这个系列的指南有什么改进的想法,欢迎[提 issue](https://github.com/soyaine/JavaScript30/issues),如果你也想参与写作,请看 [wiki](https://github.com/soyaine/JavaScript30/wiki/%E6%8C%87%E5%8D%97%E7%BB%93%E6%9E%84%E8%AF%B4%E6%98%8E),并联系 Soyaine。
88+

0 commit comments

Comments
 (0)