]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/colour.cpp
added a check which should prevent the crash of bug 555111
[wxWidgets.git] / src / gtk / colour.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: colour.cpp
3// Purpose:
4// Author: Robert Roebling
5// Id: $Id$
6// Copyright: (c) 1998 Robert Roebling
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10
11#ifdef __GNUG__
12#pragma implementation "colour.h"
13#endif
14
15#include "wx/gdicmn.h"
16
17#include <gdk/gdk.h>
18#include <gdk/gdkx.h>
19#include <gdk/gdkprivate.h>
20
21//-----------------------------------------------------------------------------
22// wxColour
23//-----------------------------------------------------------------------------
24
25class wxColourRefData: public wxObjectRefData
26{
27public:
28 wxColourRefData()
29 {
30 m_color.red = 0;
31 m_color.green = 0;
32 m_color.blue = 0;
33 m_color.pixel = 0;
34 m_colormap = (GdkColormap *) NULL;
35 m_hasPixel = FALSE;
36 }
37
38 wxColourRefData(const wxColourRefData& data)
39 : wxObjectRefData()
40 {
41 m_color = data.m_color;
42 m_colormap = data.m_colormap;
43 m_hasPixel = data.m_hasPixel;
44 }
45
46 ~wxColourRefData()
47 {
48 FreeColour();
49 }
50
51 bool operator == (const wxColourRefData& data) const
52 {
53 return (m_colormap == data.m_colormap &&
54 m_hasPixel == data.m_hasPixel &&
55 m_color.red == data.m_color.red &&
56 m_color.green == data.m_color.green &&
57 m_color.blue == data.m_color.blue &&
58 m_color.pixel == data.m_color.pixel);
59 }
60
61 void FreeColour();
62 void AllocColour( GdkColormap* cmap );
63
64 GdkColor m_color;
65 GdkColormap *m_colormap;
66 bool m_hasPixel;
67
68 friend class wxColour;
69
70 // reference counter for systems with <= 8-Bit display
71 static gushort colMapAllocCounter[ 256 ];
72};
73
74gushort wxColourRefData::colMapAllocCounter[ 256 ] =
75{
76 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
77 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
78 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
79 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
80 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
81 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
82 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
83 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
84 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
85 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
86 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
87 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
88 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
89};
90
91void wxColourRefData::FreeColour()
92{
93 if (m_colormap)
94 {
95#ifdef __WXGTK20__
96 if ((m_colormap->visual->type == GDK_VISUAL_GRAYSCALE) ||
97 (m_colormap->visual->type == GDK_VISUAL_PSEUDO_COLOR))
98#else
99 GdkColormapPrivate *private_colormap = (GdkColormapPrivate*) m_colormap;
100 if ((private_colormap->visual->type == GDK_VISUAL_GRAYSCALE) ||
101 (private_colormap->visual->type == GDK_VISUAL_PSEUDO_COLOR))
102#endif
103 {
104 int idx = m_color.pixel;
105 colMapAllocCounter[ idx ] = colMapAllocCounter[ idx ] - 1;
106
107 if (colMapAllocCounter[ idx ] == 0)
108 gdk_colormap_free_colors( m_colormap, &m_color, 1 );
109 }
110 }
111}
112
113void wxColourRefData::AllocColour( GdkColormap *cmap )
114{
115 if (m_hasPixel && (m_colormap == cmap))
116 return;
117
118 FreeColour();
119
120#ifdef __WXGTK20__
121 if ( (cmap->visual->type == GDK_VISUAL_GRAYSCALE) ||
122 (cmap->visual->type == GDK_VISUAL_PSEUDO_COLOR) )
123#else
124 GdkColormapPrivate *private_colormap = (GdkColormapPrivate*) cmap;
125 if ((private_colormap->visual->type == GDK_VISUAL_GRAYSCALE) ||
126 (private_colormap->visual->type == GDK_VISUAL_PSEUDO_COLOR))
127#endif
128 {
129 m_hasPixel = gdk_colormap_alloc_color( cmap, &m_color, FALSE, TRUE );
130 int idx = m_color.pixel;
131 colMapAllocCounter[ idx ] = colMapAllocCounter[ idx ] + 1;
132 }
133 else
134 {
135 m_hasPixel = gdk_color_alloc( cmap, &m_color );
136 }
137 m_colormap = cmap;
138}
139
140//-----------------------------------------------------------------------------
141
142#define M_COLDATA ((wxColourRefData *)m_refData)
143
144#define SHIFT (8*(sizeof(short int)-sizeof(char)))
145
146IMPLEMENT_DYNAMIC_CLASS(wxColour,wxGDIObject)
147
148wxColour::wxColour( unsigned char red, unsigned char green, unsigned char blue )
149{
150 m_refData = new wxColourRefData();
151 M_COLDATA->m_color.red = ((unsigned short)red) << SHIFT;
152 M_COLDATA->m_color.green = ((unsigned short)green) << SHIFT;
153 M_COLDATA->m_color.blue = ((unsigned short)blue) << SHIFT;
154 M_COLDATA->m_color.pixel = 0;
155}
156
157void wxColour::InitFromName( const wxString &colourName )
158{
159 wxNode *node = (wxNode *) NULL;
160 if ( (wxTheColourDatabase) && (node = wxTheColourDatabase->Find(colourName)) )
161 {
162 wxColour *col = (wxColour*)node->Data();
163 UnRef();
164 if (col) Ref( *col );
165 }
166 else
167 {
168 m_refData = new wxColourRefData();
169
170 if (!gdk_color_parse( colourName.mb_str(), &M_COLDATA->m_color ))
171 {
172 // VZ: asserts are good in general but this one is triggered by
173 // calling wxColourDatabase::FindColour() with an
174 // unrecognized colour name and this can't be avoided from the
175 // user code, so don't give it here
176 //
177 // a better solution would be to changed code in FindColour()
178
179 //wxFAIL_MSG( wxT("wxColour: couldn't find colour") );
180
181 delete m_refData;
182 m_refData = (wxObjectRefData *) NULL;
183 }
184 }
185}
186
187wxColour::~wxColour()
188{
189}
190
191bool wxColour::operator == ( const wxColour& col ) const
192{
193 if (m_refData == col.m_refData) return TRUE;
194
195 if (!m_refData || !col.m_refData) return FALSE;
196
197 GdkColor *own = &(((wxColourRefData*)m_refData)->m_color);
198 GdkColor *other = &(((wxColourRefData*)col.m_refData)->m_color);
199 if (own->red != other->red) return FALSE;
200 if (own->blue != other->blue) return FALSE;
201 if (own->green != other->green) return FALSE;
202
203 return TRUE;
204}
205
206wxObjectRefData *wxColour::CreateRefData() const
207{
208 return new wxColourRefData;
209}
210
211wxObjectRefData *wxColour::CloneRefData(const wxObjectRefData *data) const
212{
213 return new wxColourRefData(*(wxColourRefData *)data);
214}
215
216void wxColour::Set( unsigned char red, unsigned char green, unsigned char blue )
217{
218 AllocExclusive();
219
220 m_refData = new wxColourRefData();
221 M_COLDATA->m_color.red = ((unsigned short)red) << SHIFT;
222 M_COLDATA->m_color.green = ((unsigned short)green) << SHIFT;
223 M_COLDATA->m_color.blue = ((unsigned short)blue) << SHIFT;
224 M_COLDATA->m_color.pixel = 0;
225}
226
227unsigned char wxColour::Red() const
228{
229 wxCHECK_MSG( Ok(), 0, wxT("invalid colour") );
230
231 return (unsigned char)(M_COLDATA->m_color.red >> SHIFT);
232}
233
234unsigned char wxColour::Green() const
235{
236 wxCHECK_MSG( Ok(), 0, wxT("invalid colour") );
237
238 return (unsigned char)(M_COLDATA->m_color.green >> SHIFT);
239}
240
241unsigned char wxColour::Blue() const
242{
243 wxCHECK_MSG( Ok(), 0, wxT("invalid colour") );
244
245 return (unsigned char)(M_COLDATA->m_color.blue >> SHIFT);
246}
247
248void wxColour::CalcPixel( GdkColormap *cmap )
249{
250 if (!Ok()) return;
251
252 M_COLDATA->AllocColour( cmap );
253}
254
255int wxColour::GetPixel() const
256{
257 wxCHECK_MSG( Ok(), 0, wxT("invalid colour") );
258
259 return M_COLDATA->m_color.pixel;
260}
261
262GdkColor *wxColour::GetColor() const
263{
264 wxCHECK_MSG( Ok(), (GdkColor *) NULL, wxT("invalid colour") );
265
266 return &M_COLDATA->m_color;
267}
268
269