/* =========================================================================== Doom 3 GPL Source Code Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company. This file is part of the Doom 3 GPL Source Code (?Doom 3 Source Code?). Doom 3 Source Code is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Doom 3 Source Code is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Doom 3 Source Code. If not, see . In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below. If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA. =========================================================================== */ #include "../../idlib/precompiled.h" #pragma hdrstop #include "../../sys/win32/rc/Radiant_resource.h" #include "DialogColorPicker.h" #ifdef ID_DEBUG_MEMORY #undef new #undef DEBUG_NEW #define DEBUG_NEW new #endif // Old color picker class CMyColorDialog : public CColorDialog { DECLARE_DYNCREATE(CMyColorDialog); // Construction public: CMyColorDialog( COLORREF clrInit = 0, DWORD dwFlags = 0, CWnd *pParentWnd = NULL ); virtual int DoModal(); protected: enum { NCUSTCOLORS = 16 }; static COLORREF c_CustColors[NCUSTCOLORS]; static COLORREF c_LastCustColors[NCUSTCOLORS]; static bool c_NeedToInitCustColors; static void InitCustColors(); static void SaveCustColors(); // Dialog Data //{{AFX_DATA(CMyColorDialog) //}}AFX_DATA protected: // ClassWizard generate virtual function overrides //{{AFX_VIRTUAL(CMyColorDialog) virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support //}}AFX_VIRTUAL // Generated message map functions //{{AFX_MSG(CMyColorDialog) //}}AFX_MSG DECLARE_MESSAGE_MAP() }; IMPLEMENT_DYNCREATE( CMyColorDialog, CColorDialog ) bool CMyColorDialog::c_NeedToInitCustColors = true; COLORREF CMyColorDialog::c_CustColors[]; COLORREF CMyColorDialog::c_LastCustColors[]; #define SECTION _T("Custom Colors") void CMyColorDialog::InitCustColors() { for ( int i = 0; i < NCUSTCOLORS; i++) { CString entry; entry.Format( "tool_color%d", i); idCVar *cvar = cvarSystem->Find( entry ); if ( cvar ) { c_LastCustColors[i] = c_CustColors[i] = cvar->GetInteger(); } else { c_LastCustColors[i] = c_CustColors[i] = RGB( 255, 255, 255 ); } } c_NeedToInitCustColors= false; } void CMyColorDialog::SaveCustColors() { for (int i = 0; i < NCUSTCOLORS; i++) { if ( c_LastCustColors[i] != c_CustColors[i] ) { CString entry; entry.Format( "tool_color%d", i ); if ( c_CustColors[i] == RGB( 255, 255, 255 ) ) { cvarSystem->SetCVarString( entry, "" ); } else { cvarSystem->SetCVarString( entry, va( "%d", c_CustColors[i] ), CVAR_TOOL ); } c_LastCustColors[i] = c_CustColors[i]; } } } CMyColorDialog::CMyColorDialog( COLORREF clrInit, DWORD dwFlags, CWnd* pParentWnd) : CColorDialog(clrInit,dwFlags,pParentWnd) { //{{AFX_DATA_INIT(CMyColorDialog) //}}AFX_DATA_INIT if (c_NeedToInitCustColors) { InitCustColors(); } m_cc.lpCustColors = c_CustColors; } int CMyColorDialog::DoModal() { int code = CColorDialog::DoModal(); SaveCustColors(); return code; } void CMyColorDialog::DoDataExchange(CDataExchange* pDX) { // overridden (calls this base class) CColorDialog::DoDataExchange(pDX); //{{AFX_DATA_MAP(CMyColorDialog) //}}AFX_DATA_MAP } BEGIN_MESSAGE_MAP(CMyColorDialog, CColorDialog) //{{AFX_MSG_MAP(CMyColorDialog) //}}AFX_MSG_MAP END_MESSAGE_MAP() COLORREF DoOldColor(COLORREF cr) { CMyColorDialog dlg(cr, CC_FULLOPEN | CC_RGBINIT | CC_ANYCOLOR); if (dlg.DoModal() == IDOK) { return dlg.GetColor(); } return cr; } // New color picker // Original ColorPicker/DIB source by Rajiv Ramachandran // included with Permission from the author #define RADUIS 100 #define IN_NOTHING 0 #define IN_CIRCLE 1 #define IN_BRIGHT 2 #define IN_OVERBRIGHT 3 int Distance(CPoint pt1,CPoint pt2); double Slope( CPoint pt1,CPoint pt2 ) { double x,y; y = pt2.y - pt1.y; x = pt2.x - pt1.x; if( x ) { return y/x; } else { return BAD_SLOPE; } } CPoint Intersection(LineDesc l1,LineDesc l2) { CPoint pt; double x,y; if(l1.slope == l2.slope) { // Parallel lines, no intersection return CPoint(0,0); } else if(l1.slope == BAD_SLOPE ) { // First Line is vertical, eqn is x=0 // Put x = 0 in second line eqn to get y; x = l1.x; y = l2.slope * x + l2.c; } else if(l2.slope == BAD_SLOPE) { // second line is vertical Equation of line is x=0; // Put x = 0 in first line eqn to get y; x = l2.x; y = l1.slope * l2.x + l1.c; } else { y = ((l1.c * l2.slope) - (l2.c * l1.slope))/(l2.slope - l1.slope); x = (y - l1.c)/l1.slope; } return CPoint((int)x,(int)y); } double FindC(LineDesc& l) { double c; if(l.slope == BAD_SLOPE) { c = l.y; } else { c = l.y - l.slope * l.x; } return c; } CPoint PointOnLine(CPoint pt1,CPoint pt2,int len,int maxlen ) { double x,y,m,a,c,C,A; double a2,c2,m2,B; CPoint opt = pt1; CPoint pt; pt1.y *= -1; pt2.y *= -1; a = (double)len; if(pt2.x != pt1.x) { m = (double)(pt2.y - pt1.y)/(pt2.x - pt1.x); m2 = m*m; a2 = a*a; c = (double)pt1.y - m * (double)pt1.x; c2 = c*c; A = 1.0; x = pt1.x; B = 2.0 * pt1.x; x *= x; C = x - a2/(m2 + 1); x = (B + idMath::Sqrt(B*B - (4.0*A*C)))/(2.0*A); y = m*x + c; pt = CPoint((int)x,(int)y); if(Distance(pt,pt1) > maxlen || Distance(pt,pt2) > maxlen) { x = (B - idMath::Sqrt(B*B - (4.0*A*C)))/(2.0 * A); y = m*x + c; pt = CPoint((int)x,(int)y); } } else { a2 = a*a; y = idMath::Sqrt(a2); x = 0; pt = CPoint((int)x,(int)y); pt += pt1; if(Distance(pt,pt1) > maxlen || Distance(pt,pt2) > maxlen) { y = -1.0 *y; pt = CPoint((int)x,(int)y); pt+=pt1; } } pt.y *= -1; return pt; } int Distance(CPoint pt1,CPoint pt2) { double a; int x,y; y = (pt1.y - pt2.y); y *= y; x = (pt1.x - pt2.x); x *= x; a = (double)x + (double)y ; a = idMath::Sqrt(a); return (int)a; } double AngleFromPoint(CPoint pt,CPoint center) { double x,y; y = -1 * (pt.y - center.y); x = pt.x - center.x; if(x == 0 && y == 0) { return 0.0; } else { return atan2(y,x); } } CPoint PtFromAngle(double angle,double sat,CPoint center) { angle = DEG2RAD(angle); sat = TOSCALE(sat); double x,y; x = sat * cos(angle); y = sat * sin(angle); CPoint pt; pt = CPoint((int)x,(int)y); pt.y *= -1; pt += center; return pt; } RGBType HSVType::toRGB() { RGBType rgb; if(!h && !s) { rgb.r = rgb.g = rgb.b = v; } double min,max,delta,hue; max = v; delta = (max * s)/255.0; min = max - delta; hue = h; if(h > 300 || h <= 60) { rgb.r = (int)max; if(h > 300) { rgb.g = (int)min; hue = (hue - 360.0)/60.0; rgb.b = (int)((hue * delta - min) * -1); } else { rgb.b = (int)min; hue = hue / 60.0; rgb.g = (int)(hue * delta + min); } } else if(h > 60 && h < 180) { rgb.g = (int)max; if(h < 120) { rgb.b = (int)min; hue = (hue/60.0 - 2.0 ) * delta; rgb.r = (int)(min - hue); } else { rgb.r = (int)min; hue = (hue/60 - 2.0) * delta; rgb.b = (int)(min + hue); } } else { rgb.b = (int)max; if(h < 240) { rgb.r = (int)min; hue = (hue/60.0 - 4.0 ) * delta; rgb.g = (int)(min - hue); } else { rgb.g = (int)min; hue = (hue/60 - 4.0) * delta; rgb.r = (int)(min + hue); } } return rgb; } HSVType RGBType::toHSV() { HSVType hsv; double min,max,delta,temp; min = __min(r,__min(g,b)); max = __max(r,__max(g,b)); delta = max - min; hsv.v = (int)max; if(!delta) { hsv.h = hsv.s = 0; } else { temp = delta/max; hsv.s = (int)(temp*255); if(r == (int)max) { temp = (double)(g-b)/delta; } else if(g == (int)max) { temp = 2.0 + ((double)(b-r)/delta); } else { temp = 4.0 + ((double)(r-g)/delta); } temp *= 60; if(temp < 0) { temp+=360; } if(temp == 360) { temp = 0; } hsv.h = (int)temp; } return hsv; } ///////////////////////////////////////////////////////////////////////////// // CDialogColorPicker dialog CDialogColorPicker::CDialogColorPicker( COLORREF c, CWnd* pParent /*=NULL*/) : CDialog(CDialogColorPicker::IDD, pParent) { //{{AFX_DATA_INIT(CDialogColorPicker) m_overBright = 0.0f; //}}AFX_DATA_INIT Vertex = CPoint(102,108); Top = CPoint(102,9); Left = CPoint(23,147); Right = CPoint(181,147); color.r = GetRValue(c); color.g = GetGValue(c); color.b = GetBValue(c); m_OldColor = color; hsvColor = color.toHSV(); m_bInMouse = FALSE; m_bInitOver = FALSE; m_bInDrawAll = FALSE; overBright = 1.0f; UpdateParent = NULL; } CDialogColorPicker::~CDialogColorPicker() { if(m_RgbBitmap.GetSafeHandle()) { m_RgbBitmap.DeleteObject(); } if(m_HsbBitmap.GetSafeHandle()) { m_HsbBitmap.DeleteObject(); } } BEGIN_MESSAGE_MAP(CDialogColorPicker, CDialog) //{{AFX_MSG_MAP(CDialogColorPicker) ON_WM_LBUTTONDOWN() ON_WM_LBUTTONUP() ON_WM_MOUSEMOVE() ON_WM_SYSCOLORCHANGE() ON_WM_PAINT() ON_EN_CHANGE(IDC_EDIT_BLUE, OnChangeEditBlue) ON_EN_CHANGE(IDC_EDIT_GREEN, OnChangeEditGreen) ON_EN_CHANGE(IDC_EDIT_HUE, OnChangeEditHue) ON_EN_CHANGE(IDC_EDIT_RED, OnChangeEditRed) ON_EN_CHANGE(IDC_EDIT_SAT, OnChangeEditSat) ON_EN_CHANGE(IDC_EDIT_VAL, OnChangeEditVal) ON_EN_CHANGE(IDC_EDIT_OVERBRIGHT, OnChangeEditOverbright) ON_BN_CLICKED(IDC_BTN_OLDCOLOR, OnBtnColor) ON_WM_TIMER() //}}AFX_MSG_MAP END_MESSAGE_MAP() ///////////////////////////////////////////////////////////////////////////// // CDialogColorPicker message handlers void CDialogColorPicker::OnLButtonDown(UINT nFlags, CPoint point) { if(hsbRect.PtInRect(point)) { m_bInMouse = FALSE; if(InCircle(point)) { m_nMouseIn = IN_CIRCLE; } else if (InBright(point)) { m_nMouseIn = IN_BRIGHT; } else if (InOverBright(point)) { m_nMouseIn = IN_OVERBRIGHT; } else { m_nMouseIn = IN_NOTHING; } if(m_nMouseIn) { SetCapture(); TrackPoint(point); } } else if (rgbRect.PtInRect(point)) { m_nMouseIn = IN_NOTHING; if(rects[RED].PtInRect(point)) { SetCapture(); m_bInMouse = TRUE; nIndex = RED; } else if (rects[GREEN].PtInRect(point)) { SetCapture(); m_bInMouse = TRUE; nIndex = GREEN; } else if (rects[BLUE].PtInRect(point)) { SetCapture(); m_bInMouse = TRUE; nIndex = BLUE; } } CDialog::OnLButtonDown(nFlags, point); } void CDialogColorPicker::OnLButtonUp(UINT nFlags, CPoint point) { if(GetCapture() == this) { ReleaseCapture(); m_bInMouse = FALSE; } CDialog::OnLButtonUp(nFlags, point); } void CDialogColorPicker::OnMouseMove(UINT nFlags, CPoint point) { if(GetCapture() == this && m_nMouseIn) { TrackPoint(point); } else if(GetCapture() == this && m_bInMouse) { double val; BOOL bChange = FALSE; if(nIndex == RED) { if(point.y > Vertex.y) { point.y = Vertex.y; } point.x = Vertex.x; val = Distance(point,Vertex); if(val > RedLen) { val = RedLen; } CClientDC dc(this); DrawLines(&dc); val = (val/RedLen)*255; color.r = (int)val; CPoint pt; pt = PointOnLine(Vertex,Top,(color.r*RedLen)/255,RedLen); rects[RED] = CRect(pt.x - RECT_WIDTH ,pt.y-RECT_WIDTH ,pt.x+RECT_WIDTH ,pt.y+RECT_WIDTH ); CalcCuboid(); DrawLines(&dc); bChange = TRUE; } else if(nIndex == GREEN) { if(point.x > Vertex.x) { point.x = Vertex.x; } point.y = rects[GREEN].top + RECT_WIDTH; val = Distance(point,Vertex); if(val > GreenLen) { val = GreenLen; } CClientDC dc(this); DrawLines(&dc); val = (val/GreenLen)*255; color.g = (int)val; CPoint pt; pt = PointOnLine(Vertex,Left,(color.g*GreenLen)/255,GreenLen); rects[GREEN] = CRect(pt.x - RECT_WIDTH ,pt.y-RECT_WIDTH ,pt.x+RECT_WIDTH ,pt.y+RECT_WIDTH ); CalcCuboid(); DrawLines(&dc); bChange = TRUE; } else if(nIndex == BLUE) { if(point.x < Vertex.x) { point.x = Vertex.x; } point.y = rects[BLUE].top + RECT_WIDTH; val = Distance(point,Vertex); if(val > BlueLen) { val = BlueLen; } CClientDC dc(this); DrawLines(&dc); val = (val/BlueLen)*255; color.b = (int)val; CPoint pt; pt = PointOnLine(Vertex,Right,(color.b*GreenLen)/255,BlueLen); rects[BLUE] = CRect(pt.x - RECT_WIDTH ,pt.y-RECT_WIDTH ,pt.x+RECT_WIDTH ,pt.y+RECT_WIDTH ); CalcCuboid(); DrawLines(&dc); bChange = TRUE; } if(bChange) { hsvColor = color.toHSV(); SetEditVals(); CClientDC dc(this); DrawMarkers(&dc); CalcRects(); SetDIBPalette(); InvalidateRect(&brightRect,FALSE); DrawHSB(&dc); } } CDialog::OnMouseMove(nFlags, point); } void CDialogColorPicker::OnPaint() { CPaintDC dc(this); // device context for painting DrawHSB(&dc); DrawRGB(&dc); } BOOL CDialogColorPicker::OnInitDialog() { CDialog::OnInitDialog(); GetDlgItem(IDC_STATIC_RGB_RECT)->GetWindowRect(&rgbRect); GetDlgItem(IDC_STATIC_HSB_RECT)->GetWindowRect(&hsbRect); ScreenToClient(&rgbRect); ScreenToClient(&hsbRect); GetDlgItem(IDC_STATIC_NEWCOLOR)->GetWindowRect(&NewColorRect); ScreenToClient(&NewColorRect); CWindowDC dc(NULL); CSize bmSize; // Set Up HSB memDC.CreateCompatibleDC(&dc); LoadMappedBitmap(m_HsbBitmap,IDB_BITMAP_HSB,bmSize); hsbWidth = bmSize.cx; hsbHeight = bmSize.cy; hsbRect.InflateRect(-5,-5); hsbRect.top += 20; hsbRect.left += 10; m_Centre = CPoint(RADIUS,RADIUS); m_Centre += CPoint(hsbRect.left,hsbRect.top); brightRect = CRect(hsbRect.left+hsbWidth+20,hsbRect.top,hsbRect.left+hsbWidth+20+20,hsbRect.top + hsbHeight); overBrightRect = brightRect; overBrightRect.OffsetRect(brightRect.Width() + 5, 0); CreateBrightDIB(); CalcRects(); SetDIBPalette(); // Set Up RGB LoadMappedBitmap(m_RgbBitmap,IDB_BITMAP_RGB,bmSize); rgbWidth = bmSize.cx; rgbHeight = bmSize.cy; rgbRect.InflateRect(-5,-5); rgbRect.top+=10; rgbRect.left-=3; CPoint pt = CPoint(rgbRect.left,rgbRect.top); Top += pt; Left += pt; Right += pt; Vertex += pt; // TODO: Add your specialized code here and/or call the base class RedLen = Distance(Vertex,Top); GreenLen = Distance(Vertex,Left); BlueLen = Distance(Vertex,Right); CalcSlopes(); CalcCuboid(); SetSpinVals(); SetEditVals(); m_bInitOver = TRUE; SetTimer(0, 50, NULL); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } void CDialogColorPicker::DrawMarkers(CDC *pDC) { if(m_CurrentRect.Width()) { CPen *oldPen; CBrush *oldBrush; int oldMode; CRect cr = m_CurrentRect; oldPen = (CPen *)pDC->SelectStockObject(WHITE_PEN); oldBrush = (CBrush *)pDC->SelectStockObject(NULL_BRUSH); oldMode = pDC->SetROP2(R2_XORPEN); pDC->Rectangle(&cr); CPen pen; pen.CreatePen(PS_SOLID,2,RGB(255,255,255)); pDC->SelectObject(&pen); pDC->Rectangle(&brightMark); pDC->SelectObject(oldPen); pDC->SelectObject(oldBrush); pDC->SetROP2(oldMode); pen.DeleteObject(); } } BOOL CDialogColorPicker::InCircle(CPoint pt) { return Distance(pt,m_Centre) <= RADIUS; } BOOL CDialogColorPicker::InBright(CPoint pt) { return brightRect.PtInRect(pt); } BOOL CDialogColorPicker::InOverBright(CPoint pt) { return overBrightRect.PtInRect(pt); } void CDialogColorPicker::TrackPoint(CPoint pt) { if(m_nMouseIn == IN_CIRCLE) { CClientDC dc(this); DrawMarkers(&dc); hsvColor.h = (int)RAD2DEG(AngleFromPoint(pt,m_Centre)); if(hsvColor.h < 0) { hsvColor.h += 360; } hsvColor.s = (int)SCALETOMAX(Distance(pt,m_Centre)); if(hsvColor.s > 255) hsvColor.s = 255; SetDIBPalette(); CalcRects(); InvalidateRect(&brightRect,FALSE); DrawMarkers(&dc); color = hsvColor.toRGB(); SetEditVals(); DrawLines(&dc); CalcCuboid(); DrawRGB(&dc); } else if(m_nMouseIn == IN_BRIGHT) { double d; d = brightRect.bottom - pt.y; d *= 255; d /= brightRect.Height(); if(d < 0 ) d = 0; if(d > 255) d = 255; CClientDC dc(this); DrawMarkers(&dc); hsvColor.v = (int)d; CalcRects(); DrawMarkers(&dc); color = hsvColor.toRGB(); SetEditVals(); DrawLines(&dc); CalcCuboid(); DrawRGB(&dc); } } void CDialogColorPicker::CreateBrightDIB() { CDIB& d = m_BrightDIB; d.Create(brightRect.Width(),brightRect.Height(),8); for(int i=0; i < d.Height(); i++) { memset(d.GetLinePtr(i),i,d.Width()); } } void CDialogColorPicker::SetDIBPalette() { BYTE palette[768],*p; HSVType h = hsvColor; double d; d = 255.0/brightRect.Height(); p = palette; for(int i=brightRect.Height()-1; i >= 0 ;i--,p+=3) { h.v = (int)((double)i * d); RGBType rgb = h.toRGB(); p[0] = rgb.r; p[1] = rgb.g; p[2] = rgb.b; } m_BrightDIB.SetPalette(palette); } void CDialogColorPicker::CalcRects() { CPoint pt; pt = PtFromAngle(hsvColor.h,hsvColor.s,m_Centre); m_CurrentRect = CRect(pt.x - RECT_WIDTH,pt.y - RECT_WIDTH,pt.x+RECT_WIDTH,pt.y + RECT_WIDTH); int y; y = (int)(((double)hsvColor.v/255)*brightRect.Height()); y = brightRect.bottom - y; brightMark = CRect(brightRect.left - 2, y - 4, brightRect.right+2,y+4); } void CDialogColorPicker::DrawHSB(CDC *pDC) { if(m_HsbBitmap.GetSafeHandle()) { CBitmap *pOldBitmap ; pOldBitmap = (CBitmap *)memDC.SelectObject(&m_HsbBitmap); pDC->BitBlt(hsbRect.left,hsbRect.top,hsbWidth,hsbHeight,&memDC,0,0,SRCCOPY); m_BrightDIB.BitBlt(pDC->m_hDC,brightRect.left,brightRect.top,brightRect.Width(),brightRect.Height(),0,0); DrawMarkers(pDC); memDC.SelectObject(pOldBitmap); } } void CDialogColorPicker::DrawRGB(CDC *pDC) { if(m_RgbBitmap.GetSafeHandle()) { CBitmap *pOldBitmap ; pOldBitmap = (CBitmap *)memDC.SelectObject(&m_RgbBitmap); pDC->BitBlt(rgbRect.left,rgbRect.top,rgbWidth,rgbHeight,&memDC,0,0,SRCCOPY); DrawLines(pDC); memDC.SelectObject(pOldBitmap); } } void CDialogColorPicker::DrawLines(CDC *pDC) { CPoint pt[3]; pt[0] = PointOnLine(Vertex,Top,(color.r*RedLen)/255,RedLen); pt[1] = PointOnLine(Vertex,Left,(color.g*GreenLen)/255,GreenLen); pt[2] = PointOnLine(Vertex,Right,(color.b*BlueLen)/255,BlueLen); COLORREF col = RGB(255,255,255); CRect cr; for(int i = 0; i < 3; i++ ) { cr = CRect(pt[i].x - RECT_WIDTH ,pt[i].y-RECT_WIDTH ,pt[i].x+RECT_WIDTH ,pt[i].y+RECT_WIDTH ); rects[i] = cr; DrawXorRect(pDC,cr); } CPen *oldPen; int oldMode; oldPen = (CPen *)pDC->SelectStockObject(WHITE_PEN); oldMode = pDC->SetROP2(R2_XORPEN); /* Draw the following lines : 1 -2 2 -3 3 - 4 4- 5 5 -2 5 - 6 6-7 7-4 */ pDC->MoveTo(m_Cuboid[1]); pDC->LineTo(m_Cuboid[2]); pDC->LineTo(m_Cuboid[3]); pDC->LineTo(m_Cuboid[4]); pDC->LineTo(m_Cuboid[5]); pDC->LineTo(m_Cuboid[2]); pDC->MoveTo(m_Cuboid[5]); pDC->LineTo(m_Cuboid[6]); pDC->LineTo(m_Cuboid[7]); pDC->LineTo(m_Cuboid[4]); pDC->MoveTo(m_Cuboid[1]); pDC->LineTo(m_Cuboid[6]); pDC->SelectObject(oldPen); pDC->SetROP2(oldMode); DrawFilledColor(pDC,NewColorRect,color.color()); } void CDialogColorPicker::DrawXorRect(CDC *pDC,CRect& cr) { CPen pen,*oldPen; CBrush *oldBrush; int oldMode; pen.CreatePen(PS_SOLID,1,RGB(255,255,255)); oldPen = (CPen *)pDC->SelectObject(&pen); oldBrush = (CBrush *)pDC->SelectStockObject(NULL_BRUSH); oldMode =pDC->SetROP2(R2_XORPEN); pDC->Rectangle(&cr); pDC->SetROP2(oldMode); pDC->SelectObject(oldPen); pDC->SelectObject(oldBrush); pen.DeleteObject(); } void CDialogColorPicker::CalcSlopes() { lines[RED].slope = Slope(Top,Vertex); lines[GREEN].slope = Slope(Left,Vertex); lines[BLUE].slope = Slope(Right,Vertex); int i; for( i = 0; i < 3; i++ ) { lines[i].x = Vertex.x; lines[i].y = Vertex.y; lines[i].c = FindC(lines[i]); } } /* Cuboid points 0 = vertex 1 = Red Axis 2 = Red Green Intersection 3 = Green Axis 4 = Blue Green Intersection 5 = Blue Green Red Intersection 6 = Red Blue Intersection 7 = Blue Axis Draw the following lines : 1 -2 2 -3 3 - 4 4- 5 5 -2 5 - 6 6-7 7-4 */ void CDialogColorPicker::CalcCuboid() { double rLen,gLen,bLen; rLen = (double)(color.r*RedLen)/255; gLen = (double)(color.g*GreenLen)/255; bLen = (double)(color.b*BlueLen)/255; LineDesc l[12]; m_Cuboid[0] = Vertex; m_Cuboid[1] = PointOnLine(Vertex,Top,(int)rLen,RedLen); m_Cuboid[3] = PointOnLine(Vertex,Left,(int)gLen,GreenLen); m_Cuboid[7] = PointOnLine(Vertex,Right,(int)bLen,BlueLen); l[0] = lines[RED]; l[1] = lines[GREEN]; l[2] = lines[BLUE]; l[3].slope = lines[GREEN].slope; l[3].x = m_Cuboid[1].x; l[3].y = m_Cuboid[1].y; l[3].c = FindC(l[3]); l[4].slope = lines[RED].slope; l[4].x = m_Cuboid[3].x; l[4].y = m_Cuboid[3].y; l[4].c = FindC(l[4]); l[5].slope = lines[BLUE].slope; l[5].x = m_Cuboid[3].x; l[5].y = m_Cuboid[3].y; l[5].c = FindC(l[5]); l[6].slope = lines[GREEN].slope; l[6].x = m_Cuboid[7].x; l[6].y = m_Cuboid[7].y; l[6].c = FindC(l[6]); l[10].slope = lines[BLUE].slope; l[10].x = m_Cuboid[1].x; l[10].y = m_Cuboid[1].y; l[10].c = FindC(l[10]); l[11].slope = lines[RED].slope; l[11].x = m_Cuboid[7].x; l[11].y = m_Cuboid[7].y; l[11].c = FindC(l[11]); m_Cuboid[2] = Intersection(l[3],l[4]); m_Cuboid[4] = Intersection(l[5],l[6]); m_Cuboid[6] = Intersection(l[10],l[11]); l[7].slope = lines[RED].slope; l[7].x = m_Cuboid[4].x; l[7].y = m_Cuboid[4].y; l[7].c = FindC(l[7]); l[8].slope = lines[BLUE].slope; l[8].x = m_Cuboid[2].x; l[8].y = m_Cuboid[2].y; l[8].c = FindC(l[8]); m_Cuboid[5] = Intersection(l[7],l[8]); } void CDialogColorPicker::SetSpinVals() { ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_RED))->SetRange(0,255); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_GREEN))->SetRange(0,255); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_BLUE))->SetRange(0,255); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_HUE))->SetRange(0,360); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_SAT))->SetRange(0,255); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_VAL))->SetRange(0,255); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_OVERBRIGHT))->SetRange(0,1023); } void CDialogColorPicker::SetEditVals() { ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_RED))->SetPos(color.r); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_GREEN))->SetPos(color.g); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_BLUE))->SetPos(color.b); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_HUE))->SetPos(hsvColor.h); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_SAT))->SetPos(hsvColor.s); ((CSpinButtonCtrl *)GetDlgItem(IDC_SPIN_VAL))->SetPos(hsvColor.v); } void CDialogColorPicker::OnChangeEditBlue() { int b; b = GetDlgItemInt(IDC_EDIT_BLUE); if( b != color.b && m_bInitOver) { color.b = b; if(color.b < 0) color.b = 0; if(color.b > 255) color.b = 255; hsvColor = color.toHSV(); DrawAll(); } } void CDialogColorPicker::OnChangeEditGreen() { int g; g = GetDlgItemInt(IDC_EDIT_GREEN); if(g != color.g && m_bInitOver) { color.g = g; if(color.g < 0) color.g = 0; if(color.g > 255) color.g = 255; hsvColor = color.toHSV(); DrawAll(); } } void CDialogColorPicker::OnChangeEditRed() { int r; r = GetDlgItemInt(IDC_EDIT_RED); if(r != color.r && m_bInitOver) { color.r = r; if(color.r < 0) color.r = 0; if(color.r > 255) color.r = 255; hsvColor = color.toHSV(); DrawAll(); } } void CDialogColorPicker::OnChangeEditHue() { int h; h = GetDlgItemInt(IDC_EDIT_HUE); if(h != hsvColor.h && m_bInitOver) { hsvColor.h = h; if(hsvColor.h < 0) hsvColor.h = 0; if(hsvColor.h > 359) hsvColor.h = 359; color = hsvColor.toRGB(); DrawAll(); } } void CDialogColorPicker::OnChangeEditSat() { int s; s = GetDlgItemInt(IDC_EDIT_SAT); if(s != hsvColor.s && m_bInitOver) { hsvColor.s = s; if(hsvColor.s < 0) hsvColor.s = 0; if(hsvColor.s > 255) hsvColor.s = 255; color = hsvColor.toRGB(); DrawAll(); } } void CDialogColorPicker::OnChangeEditVal() { int v; v = GetDlgItemInt(IDC_EDIT_VAL); if(v != hsvColor.v && m_bInitOver) { hsvColor.v = v; if(hsvColor.v < 0) hsvColor.v = 0; if(hsvColor.v > 255) hsvColor.v = 255; color = hsvColor.toRGB(); DrawAll(); } } void CDialogColorPicker::OnChangeEditOverbright() { CString str; GetDlgItemText(IDC_EDIT_OVERBRIGHT, str); if(m_bInitOver) { overBright = atof(str); } } void CDialogColorPicker::DrawAll() { if(m_bInitOver && !m_bInDrawAll) { CClientDC dc(this); DrawMarkers(&dc); DrawLines(&dc); m_bInDrawAll = TRUE; CalcCuboid(); CalcRects(); SetDIBPalette(); DrawRGB(&dc); DrawHSB(&dc); SetEditVals(); m_bInDrawAll = FALSE; } } void CDialogColorPicker::DrawFilledColor(CDC *pDC,CRect cr,COLORREF c) { pDC->FillSolidRect(&cr,c); pDC->Draw3dRect(&cr,RGB(0,0,0),RGB(0,0,0)); cr.InflateRect(-1,-1); pDC->Draw3dRect(&cr,RGB(192,192,192),RGB(128,128,128)); } void CDialogColorPicker::LoadMappedBitmap(CBitmap& bitmap,UINT nIdResource,CSize& size) { CBitmap *pOldBitmap; if(bitmap.GetSafeHandle()) bitmap.DeleteObject(); if(bitmap.LoadBitmap(nIdResource)) { int width,height; BITMAP bmInfo; ::GetObject(bitmap.m_hObject,sizeof(bmInfo),&bmInfo); width = bmInfo.bmWidth; height = bmInfo.bmHeight; COLORREF colorWindow = ::GetSysColor(COLOR_3DFACE); COLORREF sourceColor = RGB(192,192,192); pOldBitmap = (CBitmap *)memDC.SelectObject(&bitmap); int i,j; for(i=0; i < height; i++) { for(j=0; j < width; j++) { if(memDC.GetPixel(j,i) == sourceColor) { memDC.SetPixel(j,i,colorWindow); } } } memDC.SelectObject(&pOldBitmap); size = CSize(width,height); } } void CDialogColorPicker::OnSysColorChange() { CSize size; LoadMappedBitmap(m_HsbBitmap,IDB_BITMAP_HSB,size); LoadMappedBitmap(m_RgbBitmap,IDB_BITMAP_RGB,size); } void CDialogColorPicker::OnTimer(UINT nIDEvent) { if ( UpdateParent ) { UpdateParent( color.r, color.g, color.b, 1.0f ); } } void CDialogColorPicker::OnBtnColor() { COLORREF cr = DoOldColor(GetColor()); color.r = GetRValue(cr); color.g = GetGValue(cr); color.b = GetBValue(cr); hsvColor = color.toHSV(); DrawAll(); } bool DoNewColor( int* i1, int* i2, int* i3, float *overBright, void (*Update)( float, float, float, float ) ) { COLORREF cr = (*i1) + ((*i2) <<8) + ((*i3) <<16); CDialogColorPicker dlg( cr ); //CMyColorDialog dlg(cr, CC_FULLOPEN | CC_RGBINIT | CC_ANYCOLOR); dlg.UpdateParent = Update; if ( dlg.DoModal() == IDOK ) { *i1 = (dlg.GetColor() & 255); *i2 = ((dlg.GetColor() >> 8) & 255); *i3 = ((dlg.GetColor() >> 16) & 255); *overBright = dlg.GetOverBright(); return true; } return false; }