[C++][opencv]基于opencv实现photoshop算法色相和饱和度调整
【测试环境】
vs2019
opencv==4.8.0
【效果演示】

【核心实现代码】
HSL.hpp
#ifndef OPENCV2_PS_HSL_HPP_
#define OPENCV2_PS_HSL_HPP_#include "opencv2/core.hpp"
using namespace cv;namespace cv {enum HSL_COLOR
{HSL_ALL,HSL_RED,HSL_YELLOW,HSL_GREEN,HSL_CYAN,HSL_BLUE,HSL_MAGENTA,
};/*** Class of HSL parameters for one channel*/
class HSLChannel {
public:int hue; //色度调整值, 数据范围: [-180, 180]int saturation; //饱和度调整值,数据范围: [-100, 100]int brightness; //明度调整值, 数据范围: [-100, 100]int colorIndex; //color index: 0 = RED, 1 = YELLOW, 2 = GREENfloat left_left; //hue range left-leftfloat left; //hue range leftfloat right; //hue range rightfloat right_right;//hue range right-rightbool defined; //HSLChannel();virtual ~HSLChannel();void calcDefined();void setColorIndex(int index);bool match(float hue);void adjust(int h, float *delta_hsb);
};/*** Class of HSL*/
class HSL {
public:HSL();virtual ~HSL();HSLChannel channels[7];int adjust(InputArray src, OutputArray dst);
};} /* namespace cv */#endif /* OPENCV2_PS_HSL_HPP_ */
HSL.cpp
#include "HSL.hpp"
#include "ColorSpace.hpp"
#include <math.h>#define CLIP_RANGE(value, min, max) ( (value) > (max) ? (max) : (((value) < (min)) ? (min) : (value)) )
#define COLOR_RANGE(value) CLIP_RANGE(value, 0, 255)namespace cv {HSLChannel::HSLChannel()
{hue = 0;saturation = 0;brightness = 0;defined = false;setColorIndex(0);
}HSLChannel::~HSLChannel()
{}void HSLChannel::setColorIndex(int index)
{int data[][4] = {{ 0, 0, 360, 360},{315, 345, 15, 45},{ 15, 45, 75, 105},{ 75, 105, 135, 165},{135, 165, 195, 225},{195, 225, 255, 285},{255, 285, 315, 345}};if (index < 0 ) index = 0;if (index > 6 ) index = 6;colorIndex = index;left_left = data[index][0];left = data[index][1];right = data[index][2];right_right = data[index][3];
}void HSLChannel::calcDefined()
{if ( hue != 0 || saturation != 0 || brightness != 0 ) {defined = true;return;}defined = false;
}bool HSLChannel::match(float hue)
{if ( left < right ) {if ( hue >= left_left && hue <= right_right )return true;} else {if ( hue >=left_left && hue <= 360 )return true;if ( hue >=0 && hue <= right_right )return true;}return false;
}void HSLChannel::adjust(int h, float *delta_hsb)
{if (colorIndex == 0 ) {delta_hsb[0] += hue;delta_hsb[1] += saturation;delta_hsb[2] += brightness;return;}if ( left < right ) {if ( h >= left_left && h <= right_right ) {if ( h >=left && h <= right) {delta_hsb[0] += hue;delta_hsb[1] += saturation;delta_hsb[2] += brightness;return;}if ( h >=left_left && h <= left && left > left_left) {delta_hsb[0] += hue * (h - left_left) / (left - left_left);delta_hsb[1] += saturation * (h - left_left) / (left - left_left);delta_hsb[2] += brightness * (h - left_left) / (left - left_left);return;}if ( h >=right && h <= right_right && right_right > right) {delta_hsb[0] += hue * (right_right - h) / (right_right - right);delta_hsb[1] += saturation * (right_right - h) / (right_right - right);delta_hsb[2] += brightness * (right_right - h) / (right_right - right);return;}}} else {if ( h >=left && h <= 360 ) {delta_hsb[0] += hue;delta_hsb[1] += saturation;delta_hsb[2] += brightness;return;}if ( h >=0 && h <= right ) {delta_hsb[0] += hue;delta_hsb[1] += saturation;delta_hsb[2] += brightness;return;}if ( h >=left_left && h <= left && left > left_left) {delta_hsb[0] += hue * (h - left_left) / (left - left_left);delta_hsb[1] += saturation * (h - left_left) / (left - left_left);delta_hsb[2] += brightness * (h - left_left) / (left - left_left);return;}if ( h >=right && h <= right_right && right_right > right) {delta_hsb[0] += hue * (right_right - h) / (right_right - right);delta_hsb[1] += saturation * (right_right - h) / (right_right - right);delta_hsb[2] += brightness * (right_right - h) / (right_right - right);return;}}
}//----------------------------------------------------------
//HSL classHSL::HSL()
{for (int i = 0; i < 7; i++)channels[i].setColorIndex(i);
}HSL::~HSL()
{
}int HSL::adjust(InputArray src, OutputArray dst)
{Mat input = src.getMat();if( input.empty() ) {return -1;}dst.create(src.size(), src.type());Mat output = dst.getMat();const uchar *in;uchar *out;int width = input.cols;int height = input.rows;int channel_count = input.channels();float hsb[3];float delta_hsb[3];//calculate definedfor (int i = 0; i < 7; i++)channels[i].calcDefined();//scan pixels of imagefor (int y = 0; y < height; y ++) {in = input.ptr<uchar>(y);out = output.ptr<uchar>(y);for (int x = 0; x < width; x ++) {//RGB to HSL conversionBGR2HSB(in, hsb);//adjust each channeldelta_hsb[0] = delta_hsb[1] = delta_hsb[2] = 0;for (int i = 0; i < 7; i++) {if ( channels[i].defined ) {channels[i].adjust(hsb[0], delta_hsb);}}//adjust huehsb[0] = int(hsb[0] + delta_hsb[0]) % 360;if ( hsb[0] < 0 ) hsb[0] += 360;//adjust saturationdelta_hsb[1] = CLIP_RANGE(delta_hsb[1], -100, 100);if ( delta_hsb[1] < 0) {hsb[1] = hsb[1] * (1 + delta_hsb[1] / 100.0);} else {hsb[1] = hsb[1] + ( 1 - hsb[1] ) * delta_hsb[1] / 100.0; //saturation increasehsb[2] = hsb[2] + ( 1 - hsb[2] ) * delta_hsb[1] / 100.0; //brightness increase}//adjust brightnessdelta_hsb[2] = CLIP_RANGE(delta_hsb[2], -100, 100);if ( delta_hsb[2] < 0) {hsb[2] = hsb[2] * (1 + delta_hsb[2] / 100.0);} else {hsb[2] = hsb[2] + ( 1 - hsb[2] ) * delta_hsb[2] / 100.0; //brightness increasehsb[1] = hsb[1] - hsb[1] * delta_hsb[2] / 100.0; //saturation decrease}//save to outputHSB2BGR(hsb, out);//move to next pixelin += 3;out += 3;for (int c = 0; c < channel_count - 3; c++) {*out++ = *in++;}}}return 0;
}} /* namespace cv */
【完整演示源码下载】
https://download.csdn.net/download/FL1623863129/88600796
【参考文献】
1 https://blog.csdn.net/c80486/article/details/52505546
相关文章:
[C++][opencv]基于opencv实现photoshop算法色相和饱和度调整
【测试环境】 vs2019 opencv4.8.0 【效果演示】 【核心实现代码】 HSL.hpp #ifndef OPENCV2_PS_HSL_HPP_ #define OPENCV2_PS_HSL_HPP_#include "opencv2/core.hpp" using namespace cv;namespace cv {enum HSL_COLOR {HSL_ALL,HSL_RED,HSL_YELLOW,HSL_GREEN,HS…...
Github 2024-08-16Java开源项目日报 Top10
根据Github Trendings的统计,今日(2024-08-16统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Java项目10TypeScript项目1Ruby项目1Apache Dubbo: 高性能的Java开源RPC框架 创建周期:4441 天开发语言:Java协议类型:Apache License 2.0St…...
AI学习记录 - torch 的 matmul和dot的关联,也就是点乘和点积的联系
有用大佬们点点赞 1、两个一维向量点积 ,求 词A 与 词A 之间的关联度 2、两个词向量之间求关联度,求 : 词A 与 词A 的关联度 5 词A 与 词B 的关联度 11 词B 与 词A 的关联度 11 词B 与 词B 的关联度 25 刚刚好和矩阵乘法符合: 3、什么是…...
leetcode 885. Spiral Matrix III
题目链接 You start at the cell (rStart, cStart) of an rows x cols grid facing east. The northwest corner is at the first row and column in the grid, and the southeast corner is at the last row and column. You will walk in a clockwise spiral shape to visi…...
mysql windows安装与远程连接配置
安装包在主页资源中 一、安装(此安装教程为“mysql-installer-community-5.7.41.0.msi”安装教程,安装到win10环境) 保持默认选项,点击”Next“。 点开第一行加号展开一路展开找到“MySQL Server 5,7,41 - X64”点击选中点击一下中间只想右侧的箭头看到…...
子网掩码是什么以及子网掩码相关计算
子网掩码 (Subnet Mask) 又称网络掩码 (Netmask),告知主机或路由设备,地址的哪一部分是网络号,包括子网的网络号部分,哪一部分是主机号部分。 子网掩码使用与IP地址相同的编址格式,即32 bit—4个8位组的32位长格式。…...
仿RabbitMQ实现消息队列
前言:本项目是仿照RabbitMQ并基于SpringBoot Mybatis SQLite3实现的消息队列,该项目实现了MQ的核心功能:生产者、消费者、中间人、发布、订阅等。 源码链接:仿Rabbit MQ实现消息队列 目录 前言:本项目是仿照Rabbi…...
SpringBoot教程(二十三) | SpringBoot实现分布式定时任务之xxl-job
SpringBoot教程(二十三) | SpringBoot实现分布式定时任务之xxl-job 简介一、前置条件:需要搭建调度中心1、先下载调度中心源码2、修改配置文件3、启动项目4、进行访问5、打包部署(上正式) 二、SpringBoot集成Xxl-Job1.…...
微前端架构的数据持久化策略与实践
微前端架构通过将一个大型前端应用拆分成多个小型、自治的子应用,提升了开发效率和应用的可维护性。然而,数据持久化作为应用的基础需求,在微前端架构中实现起来面临着一些挑战。本文将详细介绍在微前端架构下实现数据持久化的策略、技术和最…...
讲解 狼人杀中的买单双是什么意思
买单双这个概念通常出现在有第三方的板子中 比如 咒狐板子 丘比特板子 咒狐板子 第一天狼队只要推掉预言家 第二天就可以与咒狐协商绑票 推出其他好人 以及丘比特板子 如果拉出一个人狼链 那么如果孤独再连一个狼人 那么 狼队第一天就可以直接派人上去拿警徽,这样…...
回归分析系列5-贝叶斯回归
07贝叶斯回归 7.1 简介 贝叶斯回归将贝叶斯统计的思想应用于回归分析中,通过先验分布和似然函数来推断后验分布。在贝叶斯回归中,模型参数被视为随机变量,并且有自己的分布。通过贝叶斯公式,可以更新这些参数的分布,…...
oracle 数据中lsnrctl 是干啥的
突然发现lsnrctl stop 之后,依然可以启动数据库 就感觉怪怪的,一直以为这个是数据库的守护进程,原来不是。。。。 lsnrctl 是 Oracle 监听器控制实用程序的命令行界面工具,用于管理 Oracle Net 服务监听器。监听器是 Oracle 网络…...
Linux进程--进程地址空间
文章目录 一、进程地址空间1.想当然的内存2.实际的内存1.什么是地址空间2.地址空间和内存3.为什么要区分两种内存 一、进程地址空间 1.想当然的内存 我们在之前的学习中了解过内存的概念,所以变量都要存在内存中我们的程序才能跑起来,那么我们肯定也见…...
C语言传递指针给函数
C 语言允许您传递指针给函数,只需要简单地声明函数参数为指针类型即可。 下面的实例中,我们传递一个无符号的 long 型指针给函数,并在函数内改变这个值 实例1:获取系统的时间值 能接受指针作为参数的函数,也能接受数…...
探索 Kubernetes 持久化存储之 Rook Ceph 初窥门径
在 Kubernetes 生态系统中,持久化存储是支撑业务应用稳定运行的基石,对于维护整个系统的健壮性至关重要。对于选择自主搭建 Kubernetes 集群的运维架构师来说,挑选合适的后端持久化存储解决方案是关键的选型决策。目前,Ceph、Glus…...
今日(2024 年 8 月 13 日)科技新闻
我国成功发射卫星互联网高轨卫星:该卫星的成功发射将助力我国卫星互联网的发展。中国首台中子全散射谱仪运行 3 年成果丰硕:由粤港科技合作打造的多物理谱仪,在中国散裂中子源上运行 3 年来,已完成 300 多项用户实验。该谱仪的关键…...
Unity大场景切换进行异步加载时,如何设计加载进度条,并配置滑动条按照的曲线给定的速率滑动
一、异步加载场景的过程 1、异步加载场景用到的API LoadSceneAsync 2、异步加载的参数说明 (1)默认参数:SceneManagement.LoadSceneAsync(“SceneName”); AsyncOperation task SceneManager.LoadSceneAsync("SceneName");&a…...
Selenium + Python 自动化测试16(Python基础复习)
我们的目标是:按照这一套资料学习下来,大家可以独立完成自动化测试的任务。 上一篇我们讨论了使用模块化测试的测试模型,从某一程度也反映出熟练掌握一门编程语言的重要性。 为了后续进一步深入学习。本篇文章主要做下Python基础知识的复习。…...
2024新型数字政府综合解决方案(六)
新型数字政府综合解决方案通过融合人工智能、大数据、区块链和云计算技术,构建了一个全方位智能化的政务平台,旨在提升政府服务的效率、透明度和公众参与度。该方案实现了跨部门的数据互联互通与实时更新,利用先进的数据分析和自动化处理技术…...
vscode的C/C++环境配置和调试技巧
目录 1.背景 2.下载编译器 3.配置环境变量 4.安装C/C插件 5.写C语言代码并且编译成功 5.1文件操作 5.2对于两个窗口的解释 5.3C语言编译环境配置 6.创建执行文件 7.编译运行过程 8.写其他的代码的解决方案一 9.写其他的代码的解决方案二 10.同时编译多个.c文件 10…...
如何使用LXC实现高效容器编排:管理大规模集群的完整指南
如何使用LXC实现高效容器编排:管理大规模集群的完整指南 【免费下载链接】lxc LXC - Linux Containers 项目地址: https://gitcode.com/gh_mirrors/lx/lxc LXC(Linux Containers)是一种强大的容器技术,允许用户在单个Linux…...
3分钟部署RevokeMsgPatcher:Windows平台微信QQ消息防撤回终极指南
3分钟部署RevokeMsgPatcher:Windows平台微信QQ消息防撤回终极指南 【免费下载链接】RevokeMsgPatcher :trollface: A hex editor for WeChat/QQ/TIM - PC版微信/QQ/TIM防撤回补丁(我已经看到了,撤回也没用了) 项目地址: https:/…...
GTE文本向量中文模型保姆级教程:从环境搭建到API调用全流程
GTE文本向量中文模型保姆级教程:从环境搭建到API调用全流程 1. 环境准备与快速部署 1.1 系统要求与依赖安装 在开始之前,确保你的系统满足以下基本要求: 操作系统:推荐使用Ubuntu 18.04或更高版本Python版本:Pytho…...
别再手动调顺序了!用Vue3+Element Plus+Sortable.js给你的表格加个拖拽编辑弹窗(附完整代码)
Vue3Element PlusSortable.js打造高交互表格编辑弹窗实战 后台管理系统开发中,表格数据的顺序调整和字段管理一直是高频痛点。传统方案往往需要反复点击"上移/下移"按钮或填写表单参数,操作繁琐且体验割裂。本文将带你实现一个弹窗内一站式拖…...
Source Han Serif CN:终极开源中文字体深度技术指南
Source Han Serif CN:终极开源中文字体深度技术指南 【免费下载链接】source-han-serif-ttf Source Han Serif TTF 项目地址: https://gitcode.com/gh_mirrors/so/source-han-serif-ttf Source Han Serif CN(思源宋体)是Google与Adobe…...
5步突破:用RVC变声器从零到专业音色转换的实战指南
5步突破:用RVC变声器从零到专业音色转换的实战指南 【免费下载链接】Retrieval-based-Voice-Conversion-WebUI 语音数据小于等于10分钟也可以用来训练一个优秀的变声模型! 项目地址: https://gitcode.com/GitHub_Trending/re/Retrieval-based-Voice-Co…...
5步打造高效音乐体验:Listen1扩展的智能选择与效率提升指南
5步打造高效音乐体验:Listen1扩展的智能选择与效率提升指南 【免费下载链接】listen1_chrome_extension one for all free music in china (chrome extension, also works for firefox) 项目地址: https://gitcode.com/gh_mirrors/li/listen1_chrome_extension …...
电商数据仓库实战:从概念模型到物理模型的完整设计流程(含PostgreSQL示例)
电商数据仓库实战:从概念模型到物理模型的完整设计流程(含PostgreSQL示例) 在电商行业,数据已成为驱动业务增长的核心引擎。一个设计精良的数据仓库能够将分散的交易记录、用户行为和商品信息转化为可操作的商业洞察。本文将带您深…...
Web AR开发全指南:从技术原理到实战应用
Web AR开发全指南:从技术原理到实战应用 【免费下载链接】AR.js Image tracking, Location Based AR, Marker tracking. All on the Web. 项目地址: https://gitcode.com/gh_mirrors/arj/AR.js 随着增强现实技术的发展,Web AR开发已成为前端领域的…...
Avalonia预览器罢工了?别慌,手把手教你排查和修复‘无法加载axaml预览’的坑
Avalonia预览器崩溃自救指南:从错误日志到配置优化的全链路解决方案 当你正沉浸在Avalonia跨平台UI开发的流畅体验中,突然发现预览窗口变成一片空白,右下角弹出"无法加载axaml预览"的红色警告——这种突如其来的开发中断࿰…...
