1.准备一个海康相机

从垃圾桶里翻出来一个USB口相机。

2.下载MVS 和SDK

海康机器人-机器视觉-下载中心

mvs:

sdk:

用MVS 调试一下,能连接就行。

海康威视相机,MVS连接成功,但无图像怎么办?-CSDN博客

3.打开VS,创建一个新项目

添加引用,点击浏览,找到  MvCameraControl.Net.dll。

MvCameraControl.Net.dll文件,在MVS的安装目录,路径:MVS\MVS\Development\DotNet

注意根据c#

选择win32 win64 还是anycpu。

在form1.cs中using MvCamCtrl.NET;

然后添加一些控件

定义枚举相机函数,把他放到 “寻找相机”事件中

然后,把枚举的相机 都写入 下拉框。

此阶段代码如下:

form1.cs

using System;
using System.Collections.Generic;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using MvCamCtrl.NET;namespace test_HKCamera
{public partial class Form1 : Form{MyCamera device = new MyCamera();public static MyCamera.MV_CC_DEVICE_INFO_LIST stDevList = new MyCamera.MV_CC_DEVICE_INFO_LIST();public Form1(){InitializeComponent();}public List<string> ListDevices()//枚举相机{List<string> result = new List<string>();/**/if (MyCamera.MV_OK != MyCamera.MV_CC_EnumDevices_NET(MyCamera.MV_GIGE_DEVICE | MyCamera.MV_USB_DEVICE, ref stDevList)){MessageBox.Show("枚举设备失败");return null;}if (0 == stDevList.nDeviceNum){//  MessageBox.Show("未找到设备");return null;}MyCamera.MV_CC_DEVICE_INFO stDevInfo;for (int i = 0; i < stDevList.nDeviceNum; i++){stDevInfo = (MyCamera.MV_CC_DEVICE_INFO)Marshal.PtrToStructure(stDevList.pDeviceInfo[i], typeof(MyCamera.MV_CC_DEVICE_INFO));if (stDevInfo.nTLayerType == MyCamera.MV_GIGE_DEVICE){// 网口USB设备处理IntPtr buffer = Marshal.UnsafeAddrOfPinnedArrayElement(stDevInfo.SpecialInfo.stGigEInfo, 0);MyCamera.MV_GIGE_DEVICE_INFO gigeInfo = (MyCamera.MV_GIGE_DEVICE_INFO)Marshal.PtrToStructure(buffer, typeof(MyCamera.MV_GIGE_DEVICE_INFO));result.Add(gigeInfo.chSerialNumber);}else if (stDevInfo.nTLayerType == MyCamera.MV_USB_DEVICE){// USB设备处理IntPtr buffer = Marshal.UnsafeAddrOfPinnedArrayElement(stDevInfo.SpecialInfo.stUsb3VInfo, 0);MyCamera.MV_USB3_DEVICE_INFO usbInfo = (MyCamera.MV_USB3_DEVICE_INFO)Marshal.PtrToStructure(buffer,typeof(MyCamera.MV_USB3_DEVICE_INFO));result.Add(usbInfo.chSerialNumber);}}return result;}private void button_find_Click(object sender, EventArgs e)//寻找相机按钮{List<string> lstdevices = ListDevices();if (lstdevices == null){MessageBox.Show("无相机");}else{MessageBox.Show("已找到设备");foreach (string device in lstdevices){comboBox1.Items.Add(device);//找到的设备写入下拉框}}comboBox1.Enabled = true;}private void button_takephoto_Click(object sender, EventArgs e)//拍照按钮{}}
}

form1.designer.cs

namespace test_HKCamera
{partial class Form1{/// <summary>///  Required designer variable./// </summary>private System.ComponentModel.IContainer components = null;/// <summary>///  Clean up any resources being used./// </summary>/// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>protected override void Dispose(bool disposing){if (disposing && (components != null)){components.Dispose();}base.Dispose(disposing);}#region Windows Form Designer generated code/// <summary>///  Required method for Designer support - do not modify///  the contents of this method with the code editor./// </summary>private void InitializeComponent(){button_find = new Button();comboBox1 = new ComboBox();button_takephoto = new Button();pictureBox1 = new PictureBox();((System.ComponentModel.ISupportInitialize)pictureBox1).BeginInit();SuspendLayout();// // button_find// button_find.Location = new Point(248, 178);button_find.Name = "button_find";button_find.Size = new Size(227, 98);button_find.TabIndex = 0;button_find.Text = "寻找相机";button_find.UseVisualStyleBackColor = true;button_find.Click += button_find_Click;// // comboBox1// comboBox1.FormattingEnabled = true;comboBox1.Location = new Point(169, 437);comboBox1.Name = "comboBox1";comboBox1.Size = new Size(400, 32);comboBox1.TabIndex = 1;// // button_takephoto// button_takephoto.Location = new Point(235, 668);button_takephoto.Name = "button_takephoto";button_takephoto.Size = new Size(240, 92);button_takephoto.TabIndex = 2;button_takephoto.Text = "拍照";button_takephoto.UseVisualStyleBackColor = true;button_takephoto.Click += button_takephoto_Click;// // pictureBox1// pictureBox1.Location = new Point(698, 141);pictureBox1.Name = "pictureBox1";pictureBox1.Size = new Size(800, 600);pictureBox1.TabIndex = 3;pictureBox1.TabStop = false;// // Form1// AutoScaleDimensions = new SizeF(11F, 24F);AutoScaleMode = AutoScaleMode.Font;ClientSize = new Size(1597, 918);Controls.Add(pictureBox1);Controls.Add(button_takephoto);Controls.Add(comboBox1);Controls.Add(button_find);Name = "Form1";Text = "Form1";((System.ComponentModel.ISupportInitialize)pictureBox1).EndInit();ResumeLayout(false);}#endregionprivate Button button_find;private ComboBox comboBox1;private Button button_takephoto;private PictureBox pictureBox1;}
}

运行

接下来完善拍照功能。

修改button_find_Click

遍历SDK返回的原始设备列表,保存设备信息到 _deviceList

                for (int i = 0; i < stDevList.nDeviceNum; i++)
                {
                    var devInfo = (MyCamera.MV_CC_DEVICE_INFO)Marshal.PtrToStructure(
                        stDevList.pDeviceInfo[i],
                        typeof(MyCamera.MV_CC_DEVICE_INFO)
                    );
                    _deviceList.Add(devInfo);     

增加 oneshot功能:

创建设备对象:实例化MyCamera对象,用于后续操作。

创建设备句柄:调用MV_CC_CreateDevice_NET,传入选中的设备信息,创建设备句柄。

打开设备:使用MV_CC_OpenDevice_NET以独占模式打开设备。

开始采集图像:调用MV_CC_StartGrabbing_NET启动图像采集。

获取图像参数:通过MV_CC_GetIntValue_NET获取PayloadSize,即图像数据的大小。

分配缓冲区:使用Marshal.AllocHGlobal分配非托管内存,用于存储图像数据。

获取单帧图像:调用MV_CC_GetOneFrameTimeout_NET,设置超时时间为1000毫秒,尝试获取一帧图像。

显示图像:如果成功获取图像数据,填充MV_DISPLAY_FRAME_INFO结构体,并调用MV_CC_DisplayOneFrame_NET显示图像。

释放资源:释放之前分配的非托管内存缓冲区。

finally块:确保在方法执行完毕后,无论是否发生异常,都会停止采集、关闭设备并销毁设备句柄

        public void OneShot(PictureBox pictureBox, MyCamera.MV_CC_DEVICE_INFO selectedDevice){MyCamera device = new MyCamera();int nRet = MyCamera.MV_OK;try{// 直接使用传入的设备信息nRet = device.MV_CC_CreateDevice_NET(ref selectedDevice);if (nRet != MyCamera.MV_OK){MessageBox.Show($"创建设备失败: 0x{nRet:X8}");return;}nRet = device.MV_CC_OpenDevice_NET(MyCamera.MV_ACCESS_Exclusive, 0);if (nRet != MyCamera.MV_OK){MessageBox.Show($"打开设备失败: 0x{nRet:X8}");return;}nRet = device.MV_CC_StartGrabbing_NET();if (nRet != MyCamera.MV_OK){MessageBox.Show($"开始采集失败: 0x{nRet:X8}");return;}// 获取图像参数MyCamera.MVCC_INTVALUE stParam = new MyCamera.MVCC_INTVALUE();nRet = device.MV_CC_GetIntValue_NET("PayloadSize", ref stParam);if (nRet != MyCamera.MV_OK){MessageBox.Show($"获取图像大小失败: 0x{nRet:X8}");return;}// 分配缓冲区IntPtr pBuf = Marshal.AllocHGlobal((int)stParam.nCurValue);MyCamera.MV_FRAME_OUT_INFO_EX frameInfo = new MyCamera.MV_FRAME_OUT_INFO_EX();// 获取单帧图像nRet = device.MV_CC_GetOneFrameTimeout_NET(pBuf, stParam.nCurValue, ref frameInfo, 1000);if (nRet == MyCamera.MV_OK){// 显示图像MyCamera.MV_DISPLAY_FRAME_INFO displayInfo = new MyCamera.MV_DISPLAY_FRAME_INFO{hWnd = pictureBox.Handle,pData = pBuf,nDataLen = frameInfo.nFrameLen,nWidth = frameInfo.nWidth,nHeight = frameInfo.nHeight,enPixelType = frameInfo.enPixelType};device.MV_CC_DisplayOneFrame_NET(ref displayInfo);}else{MessageBox.Show($"获取图像超时: 0x{nRet:X8}");}// 释放资源Marshal.FreeHGlobal(pBuf);}finally{// 确保资源释放if (device != null){device.MV_CC_StopGrabbing_NET();device.MV_CC_CloseDevice_NET();device.MV_CC_DestroyDevice_NET();}}}

在“拍照事件”中,传入

        private void button_takephoto_Click(object sender, EventArgs e)//拍照按钮
        {
            try {
                _selectedDeviceInfo = _deviceList[comboBox1.SelectedIndex];
            } catch {
                MessageBox.Show("请选择相机");
            }
            
            OneShot(pictureBox1, _selectedDeviceInfo); // 传递设备信息
            

        }

运行后:

图像很黑,设置一下曝光

nRet = device.MV_CC_SetFloatValue_NET("ExposureTime", 15000);//设置曝光时间

运行后:

完整代码

using System;
using System.Collections.Generic;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using MvCamCtrl.NET;namespace test_HKCamera
{public partial class Form1 : Form{//用于临时存储通过海康SDK枚举到的原始设备列表public static MyCamera.MV_CC_DEVICE_INFO_LIST stDevList = new MyCamera.MV_CC_DEVICE_INFO_LIST();//从下拉框 comboBox1 中选择的设备信息private MyCamera.MV_CC_DEVICE_INFO _selectedDeviceInfo;//保存当前枚举到的所有设备信息||下拉框 comboBox1 的选项List<MyCamera.MV_CC_DEVICE_INFO> _deviceList = new List<MyCamera.MV_CC_DEVICE_INFO>();public Form1(){InitializeComponent();}public List<string> ListDevices()//枚举相机{List<string> result = new List<string>();/**/if (MyCamera.MV_OK != MyCamera.MV_CC_EnumDevices_NET(MyCamera.MV_GIGE_DEVICE | MyCamera.MV_USB_DEVICE, ref stDevList)){MessageBox.Show("枚举设备失败");return null;}if (0 == stDevList.nDeviceNum){//  MessageBox.Show("未找到设备");return null;}MyCamera.MV_CC_DEVICE_INFO stDevInfo;for (int i = 0; i < stDevList.nDeviceNum; i++){stDevInfo = (MyCamera.MV_CC_DEVICE_INFO)Marshal.PtrToStructure(stDevList.pDeviceInfo[i], typeof(MyCamera.MV_CC_DEVICE_INFO));if (stDevInfo.nTLayerType == MyCamera.MV_GIGE_DEVICE){// 网口USB设备处理IntPtr buffer = Marshal.UnsafeAddrOfPinnedArrayElement(stDevInfo.SpecialInfo.stGigEInfo, 0);MyCamera.MV_GIGE_DEVICE_INFO gigeInfo = (MyCamera.MV_GIGE_DEVICE_INFO)Marshal.PtrToStructure(buffer, typeof(MyCamera.MV_GIGE_DEVICE_INFO));result.Add(gigeInfo.chSerialNumber);}else if (stDevInfo.nTLayerType == MyCamera.MV_USB_DEVICE){// USB设备处理IntPtr buffer = Marshal.UnsafeAddrOfPinnedArrayElement(stDevInfo.SpecialInfo.stUsb3VInfo, 0);MyCamera.MV_USB3_DEVICE_INFO usbInfo = (MyCamera.MV_USB3_DEVICE_INFO)Marshal.PtrToStructure(buffer,typeof(MyCamera.MV_USB3_DEVICE_INFO));result.Add(usbInfo.chSerialNumber);}}return result;}public void OneShot(PictureBox pictureBox, MyCamera.MV_CC_DEVICE_INFO selectedDevice){MyCamera device = new MyCamera();int nRet = MyCamera.MV_OK;try{// 直接使用传入的设备信息nRet = device.MV_CC_CreateDevice_NET(ref selectedDevice);if (nRet != MyCamera.MV_OK){MessageBox.Show($"创建设备失败: 0x{nRet:X8}");return;}nRet = device.MV_CC_OpenDevice_NET(MyCamera.MV_ACCESS_Exclusive, 0);if (nRet != MyCamera.MV_OK){MessageBox.Show($"打开设备失败: 0x{nRet:X8}");return;}nRet = device.MV_CC_StartGrabbing_NET();if (nRet != MyCamera.MV_OK){MessageBox.Show($"开始采集失败: 0x{nRet:X8}");return;}//nRet = device.MV_CC_SetFloatValue_NET("ExposureTime", 15000);//设置曝光时间// 获取图像参数MyCamera.MVCC_INTVALUE stParam = new MyCamera.MVCC_INTVALUE();nRet = device.MV_CC_GetIntValue_NET("PayloadSize", ref stParam);if (nRet != MyCamera.MV_OK){MessageBox.Show($"获取图像大小失败: 0x{nRet:X8}");return;}// 分配缓冲区IntPtr pBuf = Marshal.AllocHGlobal((int)stParam.nCurValue);MyCamera.MV_FRAME_OUT_INFO_EX frameInfo = new MyCamera.MV_FRAME_OUT_INFO_EX();// 获取单帧图像nRet = device.MV_CC_GetOneFrameTimeout_NET(pBuf, stParam.nCurValue, ref frameInfo, 1000);if (nRet == MyCamera.MV_OK){// 显示图像MyCamera.MV_DISPLAY_FRAME_INFO displayInfo = new MyCamera.MV_DISPLAY_FRAME_INFO{hWnd = pictureBox.Handle,pData = pBuf,nDataLen = frameInfo.nFrameLen,nWidth = frameInfo.nWidth,nHeight = frameInfo.nHeight,enPixelType = frameInfo.enPixelType};device.MV_CC_DisplayOneFrame_NET(ref displayInfo);}else{MessageBox.Show($"获取图像超时: 0x{nRet:X8}");}// 释放资源Marshal.FreeHGlobal(pBuf);}finally{// 确保资源释放if (device != null){device.MV_CC_StopGrabbing_NET();device.MV_CC_CloseDevice_NET();device.MV_CC_DestroyDevice_NET();}}}private void button_find_Click(object sender, EventArgs e)//寻找相机按钮{List<string> lstdevices = ListDevices();if (lstdevices == null){MessageBox.Show("无相机");}else{MessageBox.Show("已找到设备");foreach (string device in lstdevices){comboBox1.Items.Add(device);//找到的设备写入下拉框}for (int i = 0; i < stDevList.nDeviceNum; i++){var devInfo = (MyCamera.MV_CC_DEVICE_INFO)Marshal.PtrToStructure(stDevList.pDeviceInfo[i],typeof(MyCamera.MV_CC_DEVICE_INFO));_deviceList.Add(devInfo);                    }}comboBox1.Enabled = true;}private void button_takephoto_Click(object sender, EventArgs e)//拍照按钮{try {_selectedDeviceInfo = _deviceList[comboBox1.SelectedIndex];} catch {MessageBox.Show("请选择相机");}OneShot(pictureBox1, _selectedDeviceInfo); // 传递设备信息}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/918009.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/918009.shtml
英文地址,请注明出处:http://en.pswp.cn/news/918009.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端页面直接生成PDF下载文件

前言 因为要实现业务需求如下图&#xff0c;业务逻辑&#xff0c;该凭证为前端代码实现&#xff0c;为了简单方便实现下载为pdf的需求。 一、怎么在前端直接生成PDF&#xff1f; 需求描述&#xff1a;浏览器打开的这个页面&#xff0c;点击下载&#xff0c;把当前弹框页面的…

性能优化——GPU的影响

关闭MSAA 之前在查一个渲染问题&#xff0c;一开始是定位到了CPU在waitforFrame所以知道是GPU的问题但如何定义GPU的问题在哪里&#xff0c;就很麻烦。我一开始以为是drawcall的问题&#xff0c;因为我发现drawcall有350个但降低到30个后&#xff0c;依然情况没有好转。毕竟dra…

软件需求关闭前的质量评估标准是什么

在 需求关闭前&#xff0c;进行 质量评估 是确保需求被完整实现、测试充分且满足业务目标的关键步骤。以下是需求关闭前的质量评估标准&#xff0c;涵盖了功能、非功能、测试覆盖率和用户满意度等方面&#xff1a;一、功能实现的质量评估标准需求完整性&#xff1a;所有功能需求…

vscode中创建python虚拟环境的方法

文章目录框架不同python解释器vscode运行python需要的插件vscode可以改变执行python脚本的默认终端虚拟环境解释创建虚拟环境的方法python的settings.json的一些好用配置框架 python解释器->虚拟环境->vscode 不同python解释器 在一台电脑中我们可以安装多个版本的pyt…

基于 ShardingSphere 的 Spring Boot 数据加密与模糊查询实现

基于 ShardingSphere 的 Spring Boot 数据加密与模糊查询实现 在数据安全与查询便捷性并重的今天,敏感数据加密存储后如何支持灵活查询成为关键挑战。本文将聚焦ShardingSphere 在实现数据加密的同时支持模糊查询的核心能力,详细介绍基于 Spring Boot 和 ShardingSphere 的完…

计算虚拟化技术

&#x1f9e0; 一、什么是计算虚拟化&#xff1f;&#xff08;基础认识&#xff09; ✅ 基本概念&#xff1a; 计算虚拟化&#xff08;Compute Virtualization&#xff09; 是指&#xff1a;在一台物理服务器上模拟多个“虚拟计算资源”&#xff0c;每个虚拟机看起来像是一台独…

Python编程基础与实践:Python基础数据结构:列表、字典和集合

Python数据结构&#xff1a;掌握列表、字典和集合 学习目标 通过本课程的学习&#xff0c;学员将掌握Python中基本的数据结构&#xff1a;列表、字典和集合。学员将了解它们的特性、使用场景以及如何高效地使用它们来解决实际问题。 相关知识点 列表、字典和集合使用 学习…

三维偏序 -- cdq 套 cdq

似乎题解区并没有 cdq 套 cdq 的作法&#xff0c;刚好今天讲了&#xff0c;就来写一发。 题意与记号 题目讲的很清楚了。此方法并没有树状数组好想也没有其高效&#xff0c;但能很方便扩展。下文记原序列为 ddd&#xff0c;将每个点拆分成点与询问&#xff0c;内部增加一个名为…

Effective C++ 条款27: 尽量用const、enum、inline替换 #define

Effective C 条款27&#xff1a;尽量用const、enum、inline替换#define核心思想&#xff1a;使用编译器&#xff08;const, enum, inline&#xff09;替代预处理器&#xff08;#define&#xff09;&#xff0c;让编译器进行语义检查&#xff0c;避免宏替换引发的错误和调试困难…

芯谷科技--高效噪声降低解决方案压缩扩展器D5015

在无绳电话系统中&#xff0c;噪声降低是提升通话质量的关键。 D5015 压缩扩展器&#xff0c;通过集成压缩器和扩展器&#xff0c;有效降低了传输噪声&#xff0c;同时保持了信号的完整性。D5015 采用 SOP20 和 DIP20 封装形式&#xff0c;具有低电压工作、低功耗、高集成度等特…

LabVIEW车床静刚度自动测

​基于LabVIEW 平台开发车床静刚度自动测试系统&#xff0c;针对传统生产法测量中人工误差大、计算复杂、效率低等问题&#xff0c;结合误差复映规律与刚度方程&#xff0c;通过高精度硬件与软件协同&#xff0c;实现试件加工前后尺寸的在线采集、自动计算与报告生成&#xff0…

汽车流通行业4S门店生存性指标:零服吸收率

我们在做汽车4S集团的商业智能BI数据分析项目中&#xff0c;对于4S店的管理&#xff0c;大家经常会提到一个分析指标&#xff0c;叫“零服吸收率”&#xff0c;这个大概是什么意思呢&#xff1f;简单来说就是4S门店一台车都没有卖出的情况下&#xff0c;光靠售后服务部分的收益…

第一性原理科学计算服务器如何选择配置-CPU选择篇

一、 大多数人知道的 (显性因素)核心数与线程数 (Core Count & Thread Count): 重要性&#xff1a; 核心是王道。 科学计算任务&#xff08;如仿真、建模、数据分析、机器学习训练&#xff09;绝大多数都高度并行化&#xff0c;可以同时利用多个核心进行计算。选择建议&…

Java 后端 + JavaScript 前端 实现按钮级别权限控制的解决方案

Java JavaScript 前后端协同实现按钮权限控制 在后台管理系统中&#xff0c;不同用户根据角色和数据状态应展示不同的操作按钮&#xff0c;比如编辑、删除、审批、提交等操作。本文将介绍一种通过 Java 后端生成按钮权限 JSON&#xff0c;前端通过 jQuery 解析控制按钮显示的…

RAG面试内容整理-18. 向量量化与索引压缩技术(PQ, HNSW 等)

当知识库规模达到百万甚至数亿级文档时,向量索引的存储和查询效率成为关键瓶颈。向量量化是一类用较低比特表示近似向量的方法,大幅压缩内存占用并加速相似度计算。PQ(Product Quantization)是其中最著名的方法之一,被Faiss等库广泛实现。PQ的思想是将高维向量划分为多个子…

如何显示一个 Elasticsearch 索引的字段

作者&#xff1a;来自 Elastic JD Armada 学习如何使用 _mapping 和 _search API、子字段、合成 _source 和运行时字段来显示 Elasticsearch 索引的字段。 更多阅读&#xff1a; Elasticsearch&#xff1a;从搜索中获取选定的字段 fields Elasticsearch&#xff1a;inverted …

vue3父组件把一个对象整体传入子组件,还是把一个对象的多个属性分成多个参数传入

以一个对象整体传入时&#xff0c;对象的定义&#xff1a;<template><div><p>姓名: {{ userInfo.name }}</p><p>年龄: {{ userInfo.age }}</p><p>邮箱: {{ userInfo.email }}</p></div> </template> <script s…

【unitrix数间混合计算】2.1 数间混合计算模块(src/number/mod.rs)

一、源码 这段代码是一个Rust模块的声明和导出配置&#xff0c;主要用于实现"类型级数与基本类型混合计算"的功能。 //! 类型级数与基本类型混合计算// 模块声明 // -------------------------------- mod types; // 结构体定义 mod normalize; …

脱机部署k3s

离线部署 K3s 文档 1. 准备工作 操作系统准备&#xff1a;确保服务器已安装好基础操作系统&#xff08;Ubuntu、CentOS 等&#xff09;。关闭防火墙或放通端口&#xff1a;建议关闭防火墙或确保 6443、10250 等端口已开放。准备离线资源文件&#xff1a; 下载地址 k3s-airga…

[失败记录] 使用HBuilderX创建的uniapp vue3项目添加tailwindcss3的完整过程

写在前面 放弃了。。。 1&#xff09;方案1 - 参考下面的“完整步骤” - 安装成功&#xff0c;运行成功&#xff0c;但是&#xff01;好多class不生效&#xff01; 2&#xff09;方案2 - 不安装tailwindcss&#xff0c;直接使用独立的编译好的完整css文件&#xff01; …