]>
Commit | Line | Data |
---|---|---|
2fd284a4 JS |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: xpmhand.cpp | |
3 | // Purpose: wxBitmap: XPM handler and constructors | |
4 | // Author: Julian Smart | |
5 | // Modified by: | |
6 | // Created: 04/01/98 | |
7 | // RCS-ID: $Id$ | |
8 | // Copyright: (c) Julian Smart | |
9 | // Licence: wxWindows licence | |
10 | ///////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | #ifdef __GNUG__ | |
10a0bdb1 | 13 | #pragma implementation "xpmhand.h" |
2fd284a4 JS |
14 | #endif |
15 | ||
16 | // For compilers that support precompilation, includes "wx.h". | |
17 | #include "wx/wxprec.h" | |
18 | ||
19 | #ifdef __BORLANDC__ | |
10a0bdb1 | 20 | #pragma hdrstop |
2fd284a4 JS |
21 | #endif |
22 | ||
23 | #ifndef WX_PRECOMP | |
10a0bdb1 VZ |
24 | #include "wx/list.h" |
25 | #include "wx/utils.h" | |
26 | #include "wx/app.h" | |
27 | #include "wx/palette.h" | |
28 | #include "wx/dcmemory.h" | |
29 | #include "wx/bitmap.h" | |
30 | #include "wx/icon.h" | |
2fd284a4 JS |
31 | #endif |
32 | ||
33 | #include "wx/msw/private.h" | |
34 | #include "wx/log.h" | |
35 | ||
2fd284a4 | 36 | #if wxUSE_XPM_IN_MSW |
10a0bdb1 | 37 | #define FOR_MSW 1 |
9b5ae11a VZ |
38 | |
39 | // allow the user to use the system-wide xpm.h by defining | |
40 | // wxUSE_XPM_H_IN_PATH (and always use xpm.h in path if __WX_SETUP_H__ is | |
41 | // defined which means that we use configure as it always add -I../xpm to | |
42 | // the include path if needed) | |
43 | #if !defined(__WX_SETUP_H__) && !defined(wxUSE_XPM_H_IN_PATH) | |
44 | #include "../xpm/xpm.h" | |
45 | #else | |
46 | #include <xpm.h> | |
47 | #endif | |
2fd284a4 JS |
48 | #endif |
49 | ||
50 | #include "wx/xpmhand.h" | |
51 | #include "wx/msw/dib.h" | |
52 | ||
10a0bdb1 VZ |
53 | #if wxUSE_XPM_IN_MSW |
54 | ||
0d0512bd VZ |
55 | static void XpmToBitmap(wxBitmap *bitmap, |
56 | const XImage *ximage, | |
9853e3b7 | 57 | const XImage *xmask, |
0d0512bd VZ |
58 | const XpmAttributes& xpmAttr) |
59 | { | |
60 | wxBitmapRefData *refData = bitmap->GetBitmapData(); | |
61 | refData->m_hBitmap = (WXHBITMAP)ximage->bitmap; | |
62 | ||
4b7f2165 | 63 | // first set the bitmap width, height, depth... |
0d0512bd VZ |
64 | BITMAP bm; |
65 | if ( !::GetObject(GetHbitmapOf(*bitmap), sizeof(bm), (LPSTR) & bm) ) | |
66 | { | |
f6bcfd97 | 67 | wxLogLastError(wxT("GetObject(bitmap)")); |
0d0512bd VZ |
68 | } |
69 | ||
9853e3b7 GRG |
70 | refData->m_width = bm.bmWidth; |
71 | refData->m_height = bm.bmHeight; | |
72 | refData->m_depth = bm.bmPlanes * bm.bmBitsPixel; | |
0d0512bd | 73 | refData->m_numColors = xpmAttr.npixels; |
4b7f2165 | 74 | |
9853e3b7 GRG |
75 | // GRG Jan/2000, mask support |
76 | if (xmask) | |
4b7f2165 | 77 | { |
9853e3b7 | 78 | wxMask *mask = new wxMask(); |
3ca6a5f0 BP |
79 | mask->SetMaskBitmap((WXHBITMAP) wxInvertMask(xmask->bitmap, |
80 | bm.bmWidth, bm.bmHeight)); | |
4b7f2165 VZ |
81 | bitmap->SetMask(mask); |
82 | } | |
0d0512bd VZ |
83 | } |
84 | ||
10a0bdb1 VZ |
85 | #endif // wxUSE_XPM_IN_MSW |
86 | ||
2fd284a4 JS |
87 | IMPLEMENT_DYNAMIC_CLASS(wxXPMFileHandler, wxBitmapHandler) |
88 | ||
10a0bdb1 VZ |
89 | bool wxXPMFileHandler::LoadFile(wxBitmap *bitmap, |
90 | const wxString& name, | |
91 | long flags, | |
92 | int desiredWidth, int desiredHeight) | |
2fd284a4 JS |
93 | { |
94 | #if wxUSE_XPM_IN_MSW | |
9853e3b7 GRG |
95 | XImage *ximage = NULL; |
96 | XImage *xmask = NULL; | |
2fd284a4 JS |
97 | XpmAttributes xpmAttr; |
98 | HDC dc; | |
99 | ||
2fd284a4 JS |
100 | dc = CreateCompatibleDC(NULL); |
101 | if (dc) | |
102 | { | |
4b7f2165 VZ |
103 | xpmAttr.valuemask = XpmReturnPixels | XpmColorTable; |
104 | int errorStatus = XpmReadFileToImage(&dc, | |
105 | wxMBSTRINGCAST name.fn_str(), | |
106 | &ximage, | |
9853e3b7 | 107 | &xmask, |
4b7f2165 VZ |
108 | &xpmAttr); |
109 | DeleteDC(dc); | |
110 | if (errorStatus == XpmSuccess) | |
111 | { | |
9853e3b7 | 112 | XpmToBitmap(bitmap, ximage, xmask, xpmAttr); |
2fd284a4 | 113 | |
4b7f2165 VZ |
114 | XpmFree(xpmAttr.pixels); |
115 | XpmFreeAttributes(&xpmAttr); | |
116 | XImageFree(ximage); | |
9853e3b7 GRG |
117 | if (xmask) |
118 | XDestroyImage(xmask); | |
4b7f2165 | 119 | } |
0d0512bd VZ |
120 | |
121 | #if WXWIN_COMPATIBILITY_2 | |
4b7f2165 | 122 | bitmap->SetOk(errorStatus == XpmSuccess); |
0d0512bd VZ |
123 | #endif // WXWIN_COMPATIBILITY_2 |
124 | ||
4b7f2165 | 125 | return bitmap->Ok(); |
2fd284a4 | 126 | } |
4b7f2165 | 127 | #endif // wxUSE_XPM_IN_MSW |
2fd284a4 JS |
128 | |
129 | return FALSE; | |
130 | } | |
131 | ||
10a0bdb1 VZ |
132 | bool wxXPMFileHandler::SaveFile(wxBitmap *bitmap, |
133 | const wxString& name, | |
134 | int type, | |
135 | const wxPalette *palette) | |
2fd284a4 JS |
136 | { |
137 | #if wxUSE_XPM_IN_MSW | |
4b7f2165 | 138 | XImage ximage; |
9853e3b7 GRG |
139 | XImage xmask; |
140 | bool hasmask = FALSE; | |
0d0512bd | 141 | |
4b7f2165 | 142 | HDC dc = CreateCompatibleDC(NULL); |
2fd284a4 JS |
143 | if (dc) |
144 | { | |
9853e3b7 GRG |
145 | /* fill the XImage struct 'by hand' */ |
146 | wxBitmapRefData *refData = bitmap->GetBitmapData(); | |
147 | ximage.width = refData->m_width; | |
148 | ximage.height = refData->m_height; | |
149 | ximage.depth = refData->m_depth; | |
150 | ximage.bitmap = (HBITMAP)refData->m_hBitmap; | |
151 | ||
152 | // GRG Jan/2000, mask support | |
153 | hasmask = (refData->m_bitmapMask != NULL); | |
154 | if (hasmask) | |
2fd284a4 | 155 | { |
9853e3b7 GRG |
156 | /* Strangely enough, the MSW version of xpmlib is not |
157 | * coherent with itself regarding masks; we have to invert | |
158 | * the mask we get when loading, but we still must pass it | |
159 | * 'as is' when saving... | |
160 | */ | |
161 | xmask.bitmap = (HBITMAP) refData->m_bitmapMask->GetMaskBitmap(); | |
162 | xmask.width = refData->m_width; | |
163 | xmask.height = refData->m_height; | |
164 | xmask.depth = 1; | |
4b7f2165 | 165 | } |
9853e3b7 GRG |
166 | |
167 | int errorStatus = XpmWriteFileFromImage( | |
168 | &dc, | |
169 | wxMBSTRINGCAST name.fn_str(), | |
170 | &ximage, | |
171 | (hasmask? &xmask : (XImage *)NULL), | |
172 | (XpmAttributes *) NULL); | |
173 | ||
174 | DeleteDC(dc); | |
175 | ||
176 | return (errorStatus == XpmSuccess); | |
4b7f2165 VZ |
177 | } |
178 | #endif // !wxUSE_XPM_IN_MSW | |
179 | ||
2fd284a4 | 180 | return FALSE; |
2fd284a4 JS |
181 | } |
182 | ||
183 | IMPLEMENT_DYNAMIC_CLASS(wxXPMDataHandler, wxBitmapHandler) | |
184 | ||
10a0bdb1 VZ |
185 | bool wxXPMDataHandler::Create(wxBitmap *bitmap, |
186 | void *data, | |
187 | long flags, | |
188 | int width, | |
189 | int height, | |
190 | int depth) | |
2fd284a4 JS |
191 | { |
192 | #if wxUSE_XPM_IN_MSW | |
9853e3b7 GRG |
193 | XImage *ximage = NULL; |
194 | XImage *xmask = NULL; | |
2fd284a4 | 195 | XpmAttributes xpmAttr; |
2fd284a4 | 196 | |
4b7f2165 | 197 | HDC dc = CreateCompatibleDC(NULL); /* memory DC */ |
2fd284a4 JS |
198 | |
199 | if (dc) | |
200 | { | |
4b7f2165 VZ |
201 | xpmAttr.valuemask = XpmReturnInfos | XpmColorTable; |
202 | int errorStatus = XpmCreateImageFromData(&dc, (char **)data, | |
203 | &ximage, | |
9853e3b7 | 204 | &xmask, |
4b7f2165 VZ |
205 | &xpmAttr); |
206 | DeleteDC(dc); | |
2fd284a4 | 207 | |
4b7f2165 VZ |
208 | if ( errorStatus == XpmSuccess ) |
209 | { | |
9853e3b7 | 210 | XpmToBitmap(bitmap, ximage, xmask, xpmAttr); |
0d0512bd | 211 | |
4b7f2165 VZ |
212 | XpmFree(xpmAttr.pixels); |
213 | XpmFreeAttributes(&xpmAttr); | |
214 | XImageFree(ximage); // releases the malloc, but does not destroy bitmap | |
9853e3b7 GRG |
215 | if (xmask) |
216 | XDestroyImage(xmask); | |
4b7f2165 | 217 | } |
0d0512bd VZ |
218 | |
219 | #if WXWIN_COMPATIBILITY_2 | |
220 | bitmap->SetOk(errorStatus == XpmSuccess); | |
221 | #endif // WXWIN_COMPATIBILITY_2 | |
222 | ||
4b7f2165 | 223 | return bitmap->Ok(); |
2fd284a4 | 224 | } |
4b7f2165 | 225 | #endif // wxUSE_XPM_IN_MSW |
2fd284a4 JS |
226 | |
227 | return FALSE; | |
228 | } | |
229 |