]>
Commit | Line | Data |
---|---|---|
1 | /////////////////////////////////////////////////////////////////////////////// | |
2 | // Name: dnd.cpp | |
3 | // Purpose: wxDropTarget class | |
4 | // Author: Robert Roebling | |
5 | // Id: $Id$ | |
6 | // Copyright: (c) 1998 Robert Roebling | |
7 | // Licence: wxWindows licence | |
8 | /////////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | #ifdef __GNUG__ | |
11 | #pragma implementation "dnd.h" | |
12 | #endif | |
13 | ||
14 | #include "wx/dnd.h" | |
15 | #include "wx/window.h" | |
16 | #include "wx/app.h" | |
17 | #include "wx/gdicmn.h" | |
18 | #include <wx/intl.h> | |
19 | ||
20 | #include "gdk/gdkprivate.h" | |
21 | ||
22 | #include <X11/Xlib.h> | |
23 | ||
24 | // ---------------------------------------------------------------------------- | |
25 | // global | |
26 | // ---------------------------------------------------------------------------- | |
27 | ||
28 | extern bool g_blockEventsOnDrag; | |
29 | ||
30 | // ---------------------------------------------------------------------------- | |
31 | // wxDropTarget | |
32 | // ---------------------------------------------------------------------------- | |
33 | ||
34 | wxDropTarget::wxDropTarget() | |
35 | { | |
36 | } | |
37 | ||
38 | wxDropTarget::~wxDropTarget() | |
39 | { | |
40 | } | |
41 | ||
42 | void wxDropTarget::UnregisterWidget( GtkWidget *widget ) | |
43 | { | |
44 | if (!widget) return; | |
45 | ||
46 | gtk_widget_dnd_drop_set( widget, FALSE, (gchar **) NULL, 0, FALSE ); | |
47 | } | |
48 | ||
49 | void wxDropTarget::RegisterWidget( GtkWidget *widget ) | |
50 | { | |
51 | wxString formats; | |
52 | int valid = 0; | |
53 | ||
54 | for ( size_t i = 0; i < GetFormatCount(); i++ ) | |
55 | { | |
56 | wxDataFormat df = GetFormat( i ); | |
57 | switch (df) | |
58 | { | |
59 | case wxDF_TEXT: | |
60 | if (i > 0) formats += ";"; | |
61 | formats += "text/plain"; | |
62 | valid++; | |
63 | break; | |
64 | case wxDF_FILENAME: | |
65 | if (i > 0) formats += ";"; | |
66 | formats += "file:ALL"; | |
67 | valid++; | |
68 | break; | |
69 | default: | |
70 | break; | |
71 | } | |
72 | } | |
73 | ||
74 | char *str = WXSTRINGCAST formats; | |
75 | ||
76 | gtk_widget_dnd_drop_set( widget, TRUE, &str, valid, FALSE ); | |
77 | } | |
78 | ||
79 | // ---------------------------------------------------------------------------- | |
80 | // wxTextDropTarget | |
81 | // ---------------------------------------------------------------------------- | |
82 | ||
83 | bool wxTextDropTarget::OnDrop( long x, long y, const void *data, size_t WXUNUSED(size) ) | |
84 | { | |
85 | OnDropText( x, y, (const char*)data ); | |
86 | return TRUE; | |
87 | } | |
88 | ||
89 | bool wxTextDropTarget::OnDropText( long x, long y, const char *psz ) | |
90 | { | |
91 | printf( "Got dropped text: %s.\n", psz ); | |
92 | printf( "At x: %d, y: %d.\n", (int)x, (int)y ); | |
93 | return TRUE; | |
94 | } | |
95 | ||
96 | size_t wxTextDropTarget::GetFormatCount() const | |
97 | { | |
98 | return 1; | |
99 | } | |
100 | ||
101 | wxDataFormat wxTextDropTarget::GetFormat(size_t WXUNUSED(n)) const | |
102 | { | |
103 | return wxDF_TEXT; | |
104 | } | |
105 | ||
106 | // ---------------------------------------------------------------------------- | |
107 | // wxFileDropTarget | |
108 | // ---------------------------------------------------------------------------- | |
109 | ||
110 | bool wxFileDropTarget::OnDropFiles( long x, long y, size_t nFiles, const char * const aszFiles[] ) | |
111 | { | |
112 | printf( "Got %d dropped files.\n", (int)nFiles ); | |
113 | printf( "At x: %d, y: %d.\n", (int)x, (int)y ); | |
114 | for (size_t i = 0; i < nFiles; i++) | |
115 | { | |
116 | printf( aszFiles[i] ); | |
117 | printf( "\n" ); | |
118 | } | |
119 | return TRUE; | |
120 | } | |
121 | ||
122 | bool wxFileDropTarget::OnDrop(long x, long y, const void *data, size_t size ) | |
123 | { | |
124 | size_t number = 0; | |
125 | char *text = (char*) data; | |
126 | for (size_t i = 0; i < size; i++) | |
127 | if (text[i] == 0) number++; | |
128 | ||
129 | if (number == 0) return TRUE; | |
130 | ||
131 | char **files = new char*[number]; | |
132 | ||
133 | text = (char*) data; | |
134 | for (size_t i = 0; i < number; i++) | |
135 | { | |
136 | files[i] = text; | |
137 | int len = strlen( text ); | |
138 | text += len+1; | |
139 | } | |
140 | ||
141 | bool ret = OnDropFiles( x, y, 1, files ); | |
142 | ||
143 | free( files ); | |
144 | ||
145 | return ret; | |
146 | } | |
147 | ||
148 | size_t wxFileDropTarget::GetFormatCount() const | |
149 | { | |
150 | return 1; | |
151 | } | |
152 | ||
153 | wxDataFormat wxFileDropTarget::GetFormat(size_t WXUNUSED(n)) const | |
154 | { | |
155 | return wxDF_FILENAME; | |
156 | } | |
157 | ||
158 | //------------------------------------------------------------------------- | |
159 | // wxDropSource | |
160 | //------------------------------------------------------------------------- | |
161 | ||
162 | //----------------------------------------------------------------------------- | |
163 | // drag request | |
164 | ||
165 | void gtk_drag_callback( GtkWidget *widget, GdkEvent *event, wxDropSource *source ) | |
166 | { | |
167 | printf( "Data requested for dropping.\n" ); | |
168 | ||
169 | wxDataObject *data = source->m_data; | |
170 | ||
171 | size_t size = data->GetDataSize(); | |
172 | char *ptr = new char[size]; | |
173 | data->GetDataHere( ptr ); | |
174 | ||
175 | gtk_widget_dnd_data_set( widget, event, ptr, size ); | |
176 | ||
177 | delete ptr; | |
178 | ||
179 | source->m_retValue = wxDragCopy; | |
180 | } | |
181 | ||
182 | wxDropSource::wxDropSource( wxWindow *win ) | |
183 | { | |
184 | g_blockEventsOnDrag = TRUE; | |
185 | ||
186 | m_window = win; | |
187 | m_widget = win->m_widget; | |
188 | if (win->m_wxwindow) m_widget = win->m_wxwindow; | |
189 | ||
190 | m_data = (wxDataObject *) NULL; | |
191 | m_retValue = wxDragCancel; | |
192 | ||
193 | m_defaultCursor = wxCursor( wxCURSOR_NO_ENTRY ); | |
194 | m_goaheadCursor = wxCursor( wxCURSOR_HAND ); | |
195 | } | |
196 | ||
197 | wxDropSource::wxDropSource( wxDataObject &data, wxWindow *win ) | |
198 | { | |
199 | g_blockEventsOnDrag = TRUE; | |
200 | ||
201 | m_window = win; | |
202 | m_widget = win->m_widget; | |
203 | if (win->m_wxwindow) m_widget = win->m_wxwindow; | |
204 | m_retValue = wxDragCancel; | |
205 | ||
206 | m_data = &data; | |
207 | ||
208 | m_defaultCursor = wxCursor( wxCURSOR_NO_ENTRY ); | |
209 | m_goaheadCursor = wxCursor( wxCURSOR_HAND ); | |
210 | } | |
211 | ||
212 | void wxDropSource::SetData( wxDataObject &data ) | |
213 | { | |
214 | m_data = &data; | |
215 | } | |
216 | ||
217 | wxDropSource::~wxDropSource(void) | |
218 | { | |
219 | // if (m_data) delete m_data; | |
220 | ||
221 | g_blockEventsOnDrag = FALSE; | |
222 | } | |
223 | ||
224 | wxDragResult wxDropSource::DoDragDrop( bool WXUNUSED(bAllowMove) ) | |
225 | { | |
226 | if (gdk_dnd.dnd_grabbed) return (wxDragResult) wxDragNone; | |
227 | if (gdk_dnd.drag_really) return (wxDragResult) wxDragNone; | |
228 | ||
229 | wxASSERT_MSG( m_data, "wxDragSource: no data" ); | |
230 | ||
231 | if (!m_data) return (wxDragResult) wxDragNone; | |
232 | if (m_data->GetDataSize() == 0) return (wxDragResult) wxDragNone; | |
233 | ||
234 | GdkWindowPrivate *wp = (GdkWindowPrivate*) m_widget->window; | |
235 | ||
236 | RegisterWindow(); | |
237 | ||
238 | gdk_dnd.drag_perhaps = TRUE; | |
239 | ||
240 | gdk_dnd.dnd_drag_start.x = 5; | |
241 | gdk_dnd.dnd_drag_start.y = 5; | |
242 | gdk_dnd.real_sw = wp; | |
243 | ||
244 | if (gdk_dnd.drag_startwindows) | |
245 | { | |
246 | g_free( gdk_dnd.drag_startwindows ); | |
247 | gdk_dnd.drag_startwindows = (GdkWindow **) NULL; | |
248 | } | |
249 | gdk_dnd.drag_numwindows = gdk_dnd.drag_really = 0; | |
250 | ||
251 | XWindowAttributes dnd_winattr; | |
252 | XGetWindowAttributes( gdk_display, wp->xwindow, &dnd_winattr ); | |
253 | wp->dnd_drag_savedeventmask = dnd_winattr.your_event_mask; | |
254 | ||
255 | gdk_dnd_drag_addwindow( m_widget->window ); | |
256 | ||
257 | GdkEventDragBegin ev; | |
258 | ev.type = GDK_DRAG_BEGIN; | |
259 | ev.window = m_widget->window; | |
260 | ev.u.allflags = 0; | |
261 | ev.u.flags.protocol_version = DND_PROTOCOL_VERSION; | |
262 | ||
263 | gdk_event_put( (GdkEvent*)&ev ); | |
264 | ||
265 | XGrabPointer( gdk_display, wp->xwindow, False, | |
266 | ButtonMotionMask | ButtonPressMask | ButtonReleaseMask | PointerMotionMask, | |
267 | GrabModeAsync, GrabModeAsync, gdk_root_window, None, CurrentTime ); | |
268 | ||
269 | gdk_dnd_set_drag_cursors( m_defaultCursor.GetCursor(), m_goaheadCursor.GetCursor() ); | |
270 | ||
271 | gdk_dnd.dnd_grabbed = TRUE; | |
272 | gdk_dnd.drag_really = 1; | |
273 | ||
274 | int x = 0; | |
275 | int y = 0; | |
276 | gdk_window_get_pointer( m_widget->window, &x, &y, (GdkModifierType *) NULL ); | |
277 | ||
278 | gdk_dnd_display_drag_cursor( x, y, FALSE, TRUE ); | |
279 | ||
280 | while (gdk_dnd.drag_really || gdk_dnd.drag_perhaps) wxYield(); | |
281 | ||
282 | UnregisterWindow(); | |
283 | ||
284 | g_blockEventsOnDrag = FALSE; | |
285 | ||
286 | return m_retValue; | |
287 | } | |
288 | ||
289 | void wxDropSource::RegisterWindow(void) | |
290 | { | |
291 | if (!m_data) return; | |
292 | ||
293 | wxString formats; | |
294 | ||
295 | wxDataFormat df = m_data->GetPreferredFormat(); | |
296 | ||
297 | switch (df) | |
298 | { | |
299 | case wxDF_TEXT: | |
300 | formats += "text/plain"; | |
301 | break; | |
302 | case wxDF_FILENAME: | |
303 | formats += "file:ALL"; | |
304 | break; | |
305 | default: | |
306 | break; | |
307 | } | |
308 | ||
309 | char *str = WXSTRINGCAST formats; | |
310 | ||
311 | gtk_widget_dnd_drag_set( m_widget, TRUE, &str, 1 ); | |
312 | ||
313 | gtk_signal_connect( GTK_OBJECT(m_widget), "drag_request_event", | |
314 | GTK_SIGNAL_FUNC(gtk_drag_callback), (gpointer)this ); | |
315 | } | |
316 | ||
317 | void wxDropSource::UnregisterWindow(void) | |
318 | { | |
319 | if (!m_widget) return; | |
320 | ||
321 | gtk_widget_dnd_drag_set( m_widget, FALSE, (gchar **) NULL, 0 ); | |
322 | ||
323 | gtk_signal_disconnect_by_data( GTK_OBJECT(m_widget), (gpointer)this ); | |
324 | } |