在C#中将RGB颜色转换为最接近的ACI颜色

6

我正在编写一个与dxf文件交互的程序。因此,我需要一种例程,它可以接收RGB颜色值,并返回最接近AutoCAD颜色索引(ACI)的颜色。

有没有人有相关代码或示例呢?最好是用C#编写,但这并不是必须的。

提前感谢。

4个回答

5
从某个来源(例如http://www.jtbworld.com/lisp/DisplayColorProperties.htm)获取所有ACI颜色的RGB值,并创建一个ACI颜色数组。要按索引获取ACI颜色,只需从该列表中选择颜色即可。
要进行“最接近”匹配的反向查找RGB,请对该数组进行遍历,并返回距离最小的颜色(例如通过检查3个颜色通道的平方距离:如果您的颜色为r,g,b,aci颜色为R,G,B,则距离为)。
dist = (r-R)*(r-R) + (g-G)*(g-G) + (b-B)*(b-B);

无论 ACI 阵列中哪种颜色具有最小的 dist,都是最接近 r、g、b 的匹配项。
编辑:正如已经指出的那样:RGB 距离作为视觉/感知差异并不好。要匹配视觉差异,请转换为 HSV/HSL 或者如果你很有雄心壮志,可以使用更奇特的颜色空间,比如 CIE XYZ,其中“距离”紧密代表相似性。这些天已经有好的颜色空间转换库,比如 Colorful https://www.nuget.org/packages/Colourful/

在RGB空间中查找最接近的匹配项并不适用于RGB(例如rgb(200,230,255)将匹配aci =“254”)。使用HSV模型可以获得更好的结果。 - Valimo Ral
好的,已经加上了有关颜色空间转换的注释。 - Anders Forsgren

1

以下是将RGB转换为ACI的方法

var color = Autodesk.AutoCAD.Colors.Color(r, g, b);

1

我不会像安德斯建议的那样麻烦地写一个硬编码的ACI颜色数组。你可以从每个合法索引获取一个AutoCAD Color对象,并从中提取RGB值作为System.Drawing.Color,使用ColorValue属性。

这是基于安德斯响应的其余部分的完整解决方案,将(r - R)*(r - R)替换为Math.Pow(r - R, 2),因为它似乎更清楚地表达了“距离”计算的勾股意图。

byte r = 1, g = 203, b = 103; // input color
double minDist = double.MaxValue;
short match = 0; // this will end up with our answer
for (short i = 1; i <= 255; ++i)
{
    var color = Autodesk.AutoCAD.Colors.Color.FromColorIndex(ColorMethod.ByAci, i);
    System.Drawing.Color rgb = color.ColorValue;
    double dist =
        Math.Pow(r - rgb.R, 2) +
        Math.Pow(g - rgb.G, 2) +
        Math.Pow(b - rgb.B, 2);
    if (dist < minDist)
    {
        minDist = dist;
        match = i;
    }
}

1

我不确定这个帖子/问题是否仍然有效,但我一直在网上寻找将颜色转换为ACI的方法,但没有成功。在我的情况下,我需要一种最好避免使用外部库和CAD函数的方法。

我无法提供C#的帮助。我通常使用Lazarus/Free Pascal,并经过大量尝试,我得出了一个似乎对我而言工作得非常好的函数。因此,我在此发布我的代码,以防它们对您或其他人有所帮助。

我的代码如下:

Function RGB2ACIDXFColor(MyColor : TColor) : Integer ;
Var
   OldCol, LowR, MidR, HiR : String ;
   RCol, GCol, BCol, LowCol, MidCol, HiCol : Integer ;
   StPt, HRatio, VRatio, Hemis : Integer ;
Begin
Result := 10 ;
{Break Color Component (BGR Color)}
{IntToHex & Hex2Dec are functions from Lazarus Libraries}
OldCol := IntToHex(MyColor,6) ;    
BCol := Hex2Dec(Copy(OldCol,1,2)) ;
GCol := Hex2Dec(Copy(OldCol,3,2)) ;
RCol := Hex2Dec(Copy(OldCol,5,2)) ;

{Find Color Component Priorities}
LowCol := RCol ;
LowR := 'R' ;
If (GCol < LowCol) Then
Begin
     LowCol := GCol ;
     LowR := 'G' ;
End; //If
If (BCol < LowCol) Then
Begin
     LowCol := BCol ;
     LowR := 'B' ;
End; //If

HiCol := RCol ;
HiR := 'R' ;
If (GCol > HiCol) Then
Begin
     HiCol := GCol ;
     HiR := 'G' ;
End; //If
If (BCol > HiCol) Then
Begin
     HiCol := BCol ;
     HiR := 'B' ;
End; //If

MidCol := GCol ;
MidR := 'G' ;
If ((HiR = 'G') AND (LowR = 'R')) OR
   ((HiR = 'R') AND (LowR = 'G')) Then
Begin
     MidCol := BCol ;
     MidR := 'B' ;
End; //If
If ((HiR = 'G') AND (LowR = 'B')) OR
   ((HiR = 'B') AND (LowR = 'G')) Then
Begin
     MidCol := RCol ;
     MidR := 'R' ;
End; //If

{Refer to CAD color table}
{Find Color Row}
VRatio := Round((5 * (255 - HiCol)) / 255) ;
VRatio *= 2 ;
{Find Color Hemisphere}
If (LowCol = 0) Then Hemis := 0 Else Hemis := 1 ;

{Find Color Start Column And Incrementation}
If (LowR = 'B') Then
Begin
     HRatio := Round((8 * GCol) / (GCol + RCol)) ;
     Result := 10 ;
End; //If
If (LowR = 'G') Then
Begin
     HRatio := Round((8 * RCol) / (RCol + BCol)) ;
     Result := 170 ;
End; //If
If (LowR = 'R') Then
Begin
     HRatio := Round((8 * BCol) / (BCol + GCol)) ;
     Result := 90 ;
End; //If

HRatio *= 10 ;
Result += HRatio + VRatio + Hemis ;
If (Result > 249) Then Result -= 240 ;
End; //Sub

我相信你能将其翻译成C#,希望这对某人有用。

干杯,

J-Eric J.


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接