【前端】Jquery拍照,通过PHP将base64编码数据转换成PNG格式,并保存图像到本地

news/2024/12/23 23:42:59 标签: 前端

目录

一、需求

二、开发语言

三、效果

四、业务逻辑:

五、web端调用摄像头

六、示例代码

1、前端

2、后端


一、需求

web端使用jquery调用摄像头拍照,并使用PHP把base64编码转换成png格式图片,下载到本地。

由于js不能指定图片存储的位置,所以需要把base64图像数据传到后台,由后台存储到指定位置。

二、开发语言

前端:JS

后端:PHP

三、效果

点击“拍摄图像”按钮,自动连续拍摄10张并上传到后台,存放到指定位置,并在前端显示预览图像

四、业务逻辑:

前端点击“拍摄图像”按钮,开始拍照,我这里的需求是连续拍摄10张

前端获取到base64编码图像数据,并发送到后端

③ 后端接收到base64编码后,转换成PNG格式并保存到指定位置

④ 图片存储成功后,返回图片路径给前端

前端显示拍摄的图像

五、web端调用摄像头

web端调用摄像头使用“webcam.js”插件,使用webcam.js调用摄像头前提是项目是https,否则没有获取摄像头的权限。

WebcamJS:jQuery移动端调用摄像头拍照插件WebcamJS

六、示例代码

1、前端

HTML:图像采集区

<style type="text/css">
	#open-power{
		transition: transform 0.3s ease-in-out; /* 添加过渡效果,使变换更平滑 */  
	  transform-origin: center center; /* 变换原点设置为图片中心 */  
	  width: 100%; /* 初始尺寸设置为容器宽度 */  
			height: auto; /* 保持图片的宽高比 */  
	}
	#imageContainer{
		  overflow: hidden; /* 隐藏超出容器的部分   */
		  position: relative; /* 如果需要相对于容器定位图片,可以添加这个属性 */
		  margin: 0 auto;

		  background:rgba(41, 123, 255, 0.06);
		  border-radius:18px;
		  border:1px dashed #297BFF;
		  cursor:pointer;
		  box-shadow:4px 4px 18px 0px rgb(0 0 0 / 8%);
		  padding: 10px;
	}
	.main__camera-power{
		background: none;
		border-radius: 0;
		border: none;
		cursor:none;
		margin: 0 auto;
		box-shadow:none; 
		padding: 0px;
	}
</style>
<div class="layui-col-md8 layui-col-sm8 layui-col-xs12">
	<div class="layui-row">
		<div class="layui-col-md12 layui-col-sm12 layui-col-xs12">
			<fieldset class="layui-elem-field">
				<legend>图像采集区</legend>
				<div class="layui-field-box">
					<!-- 图像 -->
					<div id="imageContainer">
						<div id="open-power" class="main__camera-power flex-center">
							<span class="main__camera-power--span">
								<img src="/home/images/camera.png" alt="power" />
							</span>
							<p class="main__camera-power--hint">请开启摄像头权限</p>
						</div>
					</div>
				</div>
			</fieldset>
		</div>
	</div>
</div>

HTML:拍摄图像按钮

<div class="layui-col-md6 layui-col-sm6 layui-col-xs12">
	<div class="paizhao" style=" width: 4.5rem; height: 4rem; margin: 0 1rem; background: #F0F0F0; text-align: center; box-shadow: 4px 4px 10px #888888; font-size: 0.9rem; border-radius: 10px; cursor: pointer; color: #000; font-size: 0.9rem;" >
		<i class="layui-icon layui-icon-camera-fill" style="font-size: 30px; color: #555;"></i> <br/>拍摄图像
	</div>
</div>

HTML:图片预览区域

<style>
/*	图像预览区域 */
#results {   background:#f8f8f8; }
#results > img { width: 160px; height: 120px; margin: 3px}
</style>

<div class="layui-row" style="width: 98%; margin: 1% auto; ">
	<div class="layui-col-md12 layui-col-sm12 layui-col-xs12">
		<div style="border: 1px solid #e6e6e6; height: 120px;overflow:auto;">
			<div id="results" style=""><p>Your captured images will appear here...</p></div>
		</div>
	</div>
</div>

JS:

<script src="/home/js/jquery-1.12.3.min.js"></script>
<script src="/home/js/webcam.js" type="text/javascript" charset="utf-8"></script>

// 初始化操作  设置摄像头区域 
$(function(){
	
	// 获取窗口尺寸并设置摄像头宽高为80%
	function setCameraSize() {
		var windowWidth = window.innerWidth;
		var windowHeight = window.innerHeight;
		var cameraWidth = windowWidth * 0.45;
		var cameraHeight = windowHeight * 0.68;

		$("#imageContainer").css('width',cameraWidth)
		$("#imageContainer").css('height',cameraHeight)
		// 设置摄像头宽高
		Webcam.set({
			width: cameraWidth,
			height: cameraHeight,
			jpeg_quality: 90
		});
	    // 附加摄像头到容器
		Webcam.attach('#open-power');
	}

	// 初始设置
	setCameraSize();
	// 监听窗口大小改变事件
	window.addEventListener('resize', setCameraSize);
	
});

// 拍摄图像
var c = 1  // 图像张数
var c2 = 1;	// 图像张数
var timer = null

// 开始拍照
$(".paizhao").click(function () {
	c = 1
	c2 = 1
	// 拍照前先清空div里已存在的图像
	document.getElementById('results').innerHTML = '';

	var yinpian = $('#yinpian').val();
	if (yinpian == '') {
		layer.msg('请先选择饮片名称', {time: 3000, icon:2});
		clearTimeout( timer );
		return false
	}
	layer.msg('图像正在采集并处理,请稍后...', {time: 3000, icon:0});
	tip_html = '<br/><span style="color:green">【'+yinpian+'】</span>图像正在采集并处理,请稍后...'
	$("#systip").prepend(tip_html)

	// 开始拍照
	take_snapshot();
	// 定时 每250ms拍摄一次
	timer = setInterval( take_snapshot, 250 );
})

function take_snapshot() {
	// 获取饮片名称
	var yinpian = $('#yinpian').val();
	if (yinpian == '') {
		layer.msg('请先选择饮片名称', {time: 3000, icon:2});
		clearTimeout( timer );
		return false
	}
	Webcam.snap( function(data_uri) {
		// 图片保存到本地
		saveJpg(data_uri,yinpian)
	} );
	// 图片数量+1
	c = c+1
	if (c > 10) {

		// 十张采集完成,结束采集
		clearTimeout( timer );
		timer = null;
	}
	
}

// Base64保存为jpg
function saveJpg(base64data,yinpian) {

	$.ajax({
		url:'/index/index/saveJpg',
		type:'POST',
		dataType:'JSON',
		data:{img:base64data,yinpian:yinpian},
		success:function (res) {
			console.log(res)

			var img_html = ''
			if (res.code == '200') {

				// 把返回的图像地址追加显示到图像预览区域
				var img = new Image();
				img.src = '/'+res.data
				document.getElementById('results').appendChild( img );
				console.log(img)

				c2 = c2+1
				console.log(c2)

				if (c2 > 10) {
					layer.msg('10张采集完成,请调整饮片再次采集', {time: 3000, icon:1});
					tip_html = '<br/><span style="color:green">【'+yinpian+'】</span>图像10张采集完成,请调整图像再次采集!'
					$("#systip").prepend(tip_html)
				}
			}
		}
	})
}

2、后端

接收前端传来的base64编码,把base64编码数据转存成png,并存放到指定位置

/**
* base64保存为jpg
*/
public function saveJpg()
{
    if (request()->isPost()) {

        $baseImg = trim(input('post.img'));     // base64编码
        $yinpian = trim(input('post.yinpian'));     // 饮片名称
        if (!empty($baseImg) && !empty($yinpian)) {
   
            //图片存放的路径
            $path = "uploads/images/".$yinpian.'/';
            if (!file_exists($path)) {
              mkdir($path, 0700, true); //创建目录
              chmod($path, 0700); //赋予权限
            }
            
            $uid = session('uid');
            //确保图片名唯一,防止重名产生覆盖
            $imageName = 'wx_' .$uid.'_' . rand(1000, 9000) . time(). '.jpg';
            
            //判断是否有逗号 如果有就截取后半部分
            if (strstr($baseImg,",")){
              $baseImg = explode(',',$baseImg);
              $baseImg = $baseImg[1];
            }

            //图片路径
            $imageSrc= $path . $imageName;
            //生成文件夹和图片
            $r = file_put_contents($imageSrc, base64_decode($baseImg));

            if($r){
                return apiResponse('200','图像保存成功',$imageSrc);
            }else{
                return apiResponse('110','图像保存失败');
            }
        }else{
            return apiResponse('110','初始化失败,请刷新页面');
        }
    }else{
        return apiResponse('110','非法请求');
    }
}

如果有控制摄像头放大缩小画面的需求,可参考另外一篇【PHP】控制摄像头缩放监控画面大小,并保存可视画面为图片_代码怎么实现监控视频怎么放大画面-CSDN博客


http://www.niftyadmin.cn/n/5797118.html

相关文章

Docker快速安装Tomcat

安装docker的教程&#xff0c;参考文章&#xff1a; Linux安装Docker-CSDN博客 在linux中安装Tomcat&#xff0c;步骤如下&#xff1a; 1.从远程仓库中拉取Tomcat镜像 docker pull tomcat 如果拉取很慢&#xff0c;通过更换下载镜像的地址便可解决&#xff0c;不过镜像地址可能…

50.第二阶段x86游戏实战2-lua获取本地寻路,跨地图寻路和获取当前地图id

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 本人写的内容纯属胡编乱造&#xff0c;全都是合成造假&#xff0c;仅仅只是为了娱乐&#xff0c;请不要…

[机器学习]XGBoost(1)——前置知识

XGBoost简介 XGBoost&#xff08;eXtreme Gradient Boosting&#xff09;是一种一种高效的梯度提升决策树算法&#xff0c;它通过集成多个弱学习器&#xff08;决策树&#xff09;来构建一个强学习器。 核心思想&#xff1a; XGBoost的核心思想是Boosting&#xff0c;即通过…

HarmonyOS NEXT 技术实践-实现音乐服务卡片

本项目展示了如何在 HarmonyOS Next 中实现音乐服务卡片&#xff0c;集成歌曲播放、歌词展示、歌单推荐等功能。通过使用服务卡片&#xff0c;用户无需进入完整的音乐应用即可通过简洁的界面进行播放控制和内容浏览&#xff0c;从而提高了操作的便捷性与效率。本文将详细介绍项…

403 Forbidden HTTP 响应状态码

403 Forbidden 是一种 HTTP 响应状态码&#xff0c;表示服务器理解了请求&#xff0c;但拒绝授权访问。以下是导致 403 Forbidden 错误的常见原因及解决方法&#xff1a; 1. 权限问题 原因 用户或客户端未被授权访问目标资源。 文件或目录的权限配置不正确。 解决方法 文…

HDR视频技术之八:色域映射

在之前的色调映射章节中提到&#xff1a; 在色调映射环节&#xff0c; 为了便于操作&#xff0c; 且不使图像颜色产生巨大失真&#xff0c; 色调映射算法通常会仅处理图像亮度信息&#xff0c; 将 HDR 图像亮度映射到 SDR图像亮度域中&#xff0c; 通过原 HDR 图像的颜色信息&a…

前端人脸识别,简单的活体检测(张张嘴...),vue3使用tracking.js,face.js,face-api.js实现

实现的逻辑是先检测是否有人脸&#xff0c;然后再检测是否张嘴了&#xff0c;最后生成照片传给后端比对人脸数据。本人是在工作中的项目需要考勤&#xff0c;前端需要活体检测。想到了这个简单的方法&#xff0c;在纯前端实现这些逻辑&#xff0c;当然精度不高&#xff0c;想要…

ChatGPT生成测试用例的最佳实践(四)

通常情况下还应该进行测试用例外不评审。将已完成的基于百度关键字搜索业务的功能和安全测试用例集的存放位置告知项目团队成员&#xff0c;需要预留出一定的时间&#xff0c;便于项目组研发、产品人员阅读&#xff0c;以免在项目团队测试用例评审会议上占用过多时间熟悉相关测…