当前位置: 首页 > article >正文

2023-简单点-picamera2中的取消auto focus,进行手动焦距设定

Auto Focus

  • 什么是auto focus?简介
  • picamera2支持的 Auto Focus state
  • Auto
  • Manual手动模式下的重要参数 lens position
  • 未完待续。。
    • Af Wondows参数

什么是auto focus?简介

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

一个像素有两个 前置并列透镜阵列,左右可以纠正相位差

参考

picamera2支持的 Auto Focus state

简而言之,就三种模式:

	1.Manual
	2.Auto
	3.Continuous

自动对焦(AF)状态机有3种模式:

可以通过读取每张图像返回的“AfState”元数据来监控其在每种模式下的活动。

种模式是:

  • mannual手动:镜头永远不会自动移动,但“镜头位置”控制可以用来“手动”移动镜头。
    这个控制的单位是屈光度(1 /米,单位),所以零可以用来表示“无穷大”。
    “镜头位置”也可以在图像元数据中进行监控,并将指示镜头何时到达请求的位置。
  • auto自动:在这种模式下,“后触发”控制可用于启动一个自动对焦循环。可以检查与图像一起接收的“AfState”元数据,以确定它何时完成以及是否成功,尽管我们建议使用辅助函数,使用户不必实现此操作。在这种模式下,镜头也永远不会自动移动,直到它被应用程序“触发”。
  • 连续Continuous-自动对焦算法将连续运行,并在必要时自动重新对焦。应用程序可以根据需要在这些模式之间自由切换。

Auto

from picamera2 import Picamera2
from libcamera import controls
picam2 = Picamera2()
picam2.start(show_preview=True)
picam2.set_controls({"AfMode": controls.AfModeEnum.Continuous})

Manual手动模式下的重要参数 lens position

from picamera2 import Picamera2
from libcamera import controls
picam2 = Picamera2()
picam2.start(show_preview=True)
picam2.set_controls({"AfMode": controls.AfModeEnum.Manual, "LensPosition": 0.0})

lens position control( picam2.camera_controls['LensPosition'])给出三个值,即最小、最大和默认镜头位置

  • 最小值定义了 最远的焦距,
  • 最大值指定了 最近的可实现的焦距(通过取其倒数)。
  • 第三个值给出了一个“默认”值,这通常是镜头的超焦位置。

透镜位置的最小值通常为0.0(表示无穷大)。
对于最大值,一个值为10.0将表示最近的焦距为1 / 10米,或10厘米。
默认值通常可能在0.5到1.0左右,这意味着超焦距离约为1到2米。
一般来说,用户应该期望距离校准是近似的,因为它将取决于调整的准确性和用户的模块和执行校准的模块之间的变化程度。

特别注意:lensPosition 中设置的屈光度(焦距的倒数)是近似值,根本不能信任这种东西

实际上,对应关系可能是这样的

这里是引用

具体可以参考:这个

这里直接给出gui调焦的代码:

from picamera2 import Picamera2
import time
from libcamera import controls

def getVideo(duration,name="test.mp4"):
	picam2 = Picamera2()
	#picam2.set_controls({"AfMode": controls.AfModeEnum.Continuous})
	picam2.set_controls({"AfMode": controls.AfModeEnum.Manual, "LensPosition": 7.53})
	
	video_config = picam2.create_video_configuration(main={"size": (1080, 720)})
	
	picam2.configure(video_config) 
	picam2.start_and_record_video(name, duration=duration,config=video_config)


def getStill(nums):
	picam2 = Picamera2()
	picam2.set_controls({"AfMode": controls.AfModeEnum.Manual, "LensPosition": 7.53})
	#picam2.set_controls({"AfMode": controls.AfModeEnum.Continuous})
	config = picam2.create_still_configuration()
	picam2.configure(config)
	picam2.start()
	time.sleep(2)
	i = 1
	while i <= nums:
		picam2.capture_file(f"{i}.jpg")
		i += 1
	
	
def getPreview():
	from picamera2 import Picamera2
	picam2 = Picamera2()
	config = picam2.create_preview_configuration(main={"size":(1080,720)})
	picam2.configure(config)
	picam2.start(show_preview=True)
	picam2.set_controls({"AfMode": controls.AfModeEnum.Manual, 
						 "LensPosition": 7.53,
						 })
	time.sleep(100)
	picam2.stop()

def autoAdjust():
	from picamera2 import Picamera2
	from libcamera import controls
	import tkinter
	import math

	pc2  = Picamera2() 
	# 焦距调节参数
	N    = 0.08        # 最近焦点距離(m)
	M    = 0.25        # 中間焦点距離(m)
	F    = 1.0         # 最遠焦点距離(m)
	IN   = 1.0 / N
	IM   = 1.0 / M
	IF   = 1.0 / F

	# 将滑块的值转换为焦距
	def convSliderToFocusLen( t ):
	   a = ( IM - IF ) / ( 2.0 * IM - IN - IF )
	   b = a * ( 1.0 - 2.0 * a ) * ( IM - IN )
	   c = IN + b / a
	   ID = b / ( t - a ) + c   # 1/Dなので注意
	   return 1.0 / ID


	# 将焦距转换为设定距离
	def convFocusLenToSetLen( focusLen ) :
	   Px = 0.2   # 焦点距離Px
	   Py = 0.12  # 焦点距離Py
	   Qx = 1.0   # 基准焦距设定距離Qx
	   Qy = 0.16  # 基准焦距设定距離Qy
	   bP = ( Py - N ) / ( Px - N )
	   bQ = ( Qy - N ) / ( Qx - N )
	   a = ( bQ * Qx - bP * Px ) / ( bQ - bP )
	   b = ( Px - a ) * ( N - a ) * ( Py - N ) / ( N - Px )
	   c = N - b / ( N - a )
	   return b / ( focusLen - a ) + c


	# 	给相机设定手动对焦距离
def setFocusLen( focusLen ):
	   # 将焦距转换为设定距离
	   setLen = convFocusLenToSetLen( focusLen )

	   # 给照相机设定设定距离
	   pc2.set_controls( {"LensPosition" : 1.0 / setLen} )

	   # 焦点距离表示
	   label["text"] = "{:.4f}m".format( focusLen )
	   print(f"current parm:    {1.0 / setLen}")
	   

	# 滑块变更的回调
	def onChangeSlider( event ):
	   focusLen = convSliderToFocusLen( slider.get() )
	   setFocusLen( focusLen )
		  

	# 主窗口
	root = tkinter.Tk()
	root.title( "Camera manual focus test" )

	# 追加焦距调整滑块
	sliderLen = 250
	sliderWidth = 15
	slider = tkinter.Scale(
	   root,
	   orient = tkinter.HORIZONTAL,    # 水平滑块
	   from_ = 0.0, to = 1.0,          # 0〜1之间
	   resolution = 0.01,              # 0.01步宽
	   length = sliderLen,             # 滑块的长度
	   width = sliderWidth,            # 滑块的宽度
	   showvalue = 0,                  # 不显示值
	   command = onChangeSlider        #注册值改变时被调用的函数	

	initFocusPos = 0.5
	slider.set( initFocusPos );        #初始值
	slider.place( x = 30, y = 20 )

	# 焦距标示标签
	label = tkinter.Label( root )
	label.place( x = 34 + sliderLen, y = 5 + sliderWidth )

	# 相机初始化
	# 显示预览,将自动聚焦模式设置为Manual	pc2.start_preview( True )
	pc2.start()
	pc2.set_controls( {"AfMode" : controls.AfModeEnum.Manual} )

	# 循环开始
	root.mainloop()


	
if __name__ == "__main__":
	#getVideo(5) 
	#getStill(3)
	#getPreview()
	autoAdjust()

print(f"current parm: {1.0 / setLen}")
这个会给出设置的lensposition

在这里插入图片描述

滑块拖动修改参数,
实时获得结果,看上面print(f"current parm: {1.0 / setLen}")打印的结果,然后手动设置参数picam2.set_controls({"AfMode": controls.AfModeEnum.Manual, "LensPosition":参数})

未完待续。。

Af Wondows参数

设置用来定焦的图片区域
在这里插入图片描述


http://www.kler.cn/news/150041.html

相关文章:

  • 算法通关村第十二关|青铜|字符串转换整数
  • 题目标题:卫星定位(胡宁静) 暴力解法
  • php如何对比浮点数大小(bccomp函数)
  • 代码的并发问题
  • ASCII sorting
  • (亲测有效)解决windows11无法使用1500000波特率的问题
  • RabbitMQ消息模型之发布订阅Publish-Subscribe
  • docker中安装mysql,远程连接
  • 基于docker的onlyoffice使用--运行JavaSpringExample
  • 你了解vue的diff算法吗?
  • go学习之文件操作与命令行参数
  • leetcode 283. 移动零
  • JavaScript基础知识总结
  • Java 关于批量插入遇到的问题 -sqlserver
  • 配置阿里云的yum仓库
  • qt-C++笔记之主线程中使用异步逻辑来处理ROS事件循环和Qt事件循环解决相互阻塞的问题
  • 数学加速器:Python numpy.add函数全解读
  • 基于单片机的肺活量检测系统(论文+源码)
  • 树与二叉树堆:堆的意义
  • SpringBoot整合EasyExcel实现复杂Excel表格的导入导出功能
  • DNS/ICMP协议、NAT技术
  • goweb入门教程
  • Python dns域名解析(dns.resolver)
  • 【LeetCode:1670. 设计前中后队列 | 数据结构设计】
  • spring-webflux的一些概念的理解
  • 【Rust】基本的语法概念
  • 唯创知音WT588F02B-8S语音芯片:灵活更换语音内容,降低开发成本与备货压力
  • python每日一题——12最小覆盖子串
  • GoLang切片
  • Leetcode算法系列| 1. 两数之和(四种解法)