성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Roll A Lisp In C - Reading ; https...
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>WPF Color의 기본 색 영역은 (sRGB가 아닌) scRGB</h1> <p> 재미있는 질문이 올라왔군요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 안녕하세요 rgb 계산 오차가 있는데 원인을 모르겠습니다.. ; <a target='tab' href='https://www.sysnet.pe.kr/3/0/5713'>https://www.sysnet.pe.kr/3/0/5713</a> </pre> <br /> 간단하게 코드로는 다음과 같이 재현할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Color start = Color.FromRgb(0xff, 0, <span style='color: blue; font-weight: bold'>9</span>); Color add1 = Color.FromRgb(0, 0, <span style='color: blue; font-weight: bold'>5</span>); Color add2 = Color.FromRgb(0, 0, <span style='color: blue; font-weight: bold'>6</span>); Color newColor1 = start + add1; Color newColor2 = start + add2; System.Diagnostics.Trace.WriteLine($"{newColor1.R}, {newColor1.G}, {newColor1.B}"); System.Diagnostics.Trace.WriteLine($"{newColor2.R}, {newColor2.G}, {newColor2.B}"); /* 출력 결과 255, 0, <span style='color: blue; font-weight: bold'>14</span> 255, 0, <span style='color: blue; font-weight: bold'>14</span> */ </pre> <br /> 보는 바와 같이 출력 결과가 동일합니다. 왜 이런 결과가 나올까요? ^^ 사실 WPF의 Color는 (외부적으로는 사용하기 쉬우므로) sRGB 체계를 사용하는 듯하지만 내부적으로는 이것을 scRGB 체계로 변환해 연산을 하기 때문입니다.<br /> <br /> 이를 Color.FromRgb의 소스 코드에서 알 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public static Color FromArgb(byte a, byte r, byte g, byte b) { Color result = default(Color); <span style='color: blue; font-weight: bold'>result.scRgbColor.a = (float)(int)a / 255f; result.scRgbColor.r = sRgbToScRgb(r); result.scRgbColor.g = sRgbToScRgb(g); result.scRgbColor.b = sRgbToScRgb(b);</span> <span style='color: blue; font-weight: bold'>result.context = null;</span> result.sRgbColor.a = a; result.sRgbColor.r = ScRgbTosRgb(result.scRgbColor.r); result.sRgbColor.g = ScRgbTosRgb(result.scRgbColor.g); result.sRgbColor.b = ScRgbTosRgb(result.scRgbColor.b); result.isFromScRgb = false; return result; } </pre> <br /> 보는 바와 같이 전달된 R, G, B 값을 변환해 scRgbColor에 설정하고 있고, 심지어 sRgbColor 값조차도 전달된 인자를 대입하지 않고 굳이 ScRgbTosRgb를 통해 설정하고 있습니다. 이후 더하기 연산을 할 때는,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public static Color operator +(Color color1, Color color2) { if (color1.context == null && color2.context == null) { <span style='color: blue; font-weight: bold'>return FromScRgb(color1.scRgbColor.a + color2.scRgbColor.a, color1.scRgbColor.r + color2.scRgbColor.r, color1.scRgbColor.g + color2.scRgbColor.g, color1.scRgbColor.b + color2.scRgbColor.b);</span> } if (color1.context == color2.context) { // ... context에 따른 연산... return result; } throw new ArgumentException(MS.Internal.PresentationCore.SR.Get("Color_ColorContextTypeMismatch", null)); } </pre> <br /> (context의 기본값이 null이므로) scRgbColor의 rgba 값을 더하는 연산을 합니다. 따라서 처음에 예로 든 코드를 scRGB 기준으로 다시 출력해 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > System.Diagnostics.Trace.WriteLine(newColor1); System.Diagnostics.Trace.WriteLine(newColor2); /* 출력 결과 sc#2, 1, 0, <span style='color: blue; font-weight: bold'>0.00424937764</span> sc#2, 1, 0, <span style='color: blue; font-weight: bold'>0.00455290452</span> */ </pre> <br /> 다르게 나오므로 정상적으로 더하기 연산은 된 것입니다. 단지, scRGB에서의 저런 차이가 sRGB 값으로 바꿀 때 발생하는 연산 손실로 인해 같은 것처럼 보이는 것입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > byte b1 = ScRgbTosRgb(<span style='color: blue; font-weight: bold'>0.00424937764</span>); byte b2 = ScRgbTosRgb(<span style='color: blue; font-weight: bold'>0.00455290452</span>); System.Diagnostics.Trace.WriteLine(b1); <span style='color: blue; font-weight: bold'>// 14</span> System.Diagnostics.Trace.WriteLine(b2); <span style='color: blue; font-weight: bold'>// 14</span> static byte ScRgbTosRgb(float val) { if (!((double)val > 0.0)) { return 0; } if ((double)val <= 0.0031308) { return (byte)(255f * val * 12.92f + 0.5f); } if ((double)val < 1.0) { return (byte)(255f * (1.055f * (float)Math.Pow(val, 5.0 / 12.0) - 0.055f) + 0.5f); } return byte.MaxValue; } </pre> <br /> 어쩔 수 없습니다. Color space끼리의 변환이 비가역적인 연산이기 때문에 출력에서의 R, G, B는 그에 대한 부분을 감안해야 합니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 정리해 보면, WPF가 scRGB를 따르므로 원래 연산은 이런 식으로 했어야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Color start = Color.FromScRgb(0, 1, 0, 0.0027317428f); Color add1 = Color.FromScRgb(0, 0, 0, 0.001517635f); Color add2 = Color.FromScRgb(0, 0, 0, 0.001821162f); Color newColor1 = start + add1; Color newColor2 = start + add2; System.Diagnostics.Trace.WriteLine(newColor1); System.Diagnostics.Trace.WriteLine(newColor2); /* 출력 결과 sc#0, 1, 0, 0.00424937764 sc#0, 1, 0, 0.00455290452 */ </pre> <br /> 저렇게 보면... 전혀 이상한 것이 없죠?~~~ ^^<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
8153
(왼쪽의 숫자를 입력해야 합니다.)