Cope with % escaped URIs.
[rox-lib.git] / python / rox / loading.py
blob776da277c2a1e86da324b5d7f50340e870fbf352
1 """ROX applications should provide good drag-and-drop support. Use this module
2 to allow drops onto widgets in your application."""
4 import rox
5 from rox import g, alert, get_local_path, _
7 gdk = g.gdk
9 TARGET_URILIST = 0
10 TARGET_RAW = 1
12 def unescape(uri):
13 if '%' not in uri: return uri
14 import re
15 return re.sub('%[0-9a-fA-F][0-9a-fA-F]',
16 lambda match: chr(int(match.group(0)[1:], 16)),
17 uri)
19 def extract_uris(data):
20 """Convert a text/uri-list to a python list of (unescaped) URIs"""
21 lines = data.split('\r\n')
22 out = []
23 for l in lines:
24 if l == chr(0):
25 continue # (gmc adds a '\0' line)
26 if l and l[0] != '#':
27 out.append(unescape(l))
28 return out
30 def provides(context, type): return type in map(str, context.targets)
32 class RemoteFiles(Exception):
33 "Internal use"
34 def __init__(self):
35 Exception.__init__(self, _('Cannot load files from a remote machine '
36 '(multiple files, or target application/octet-stream not provided)'))
38 class XDSLoader:
39 """A mix-in class for widgets that can have files/data dropped on
40 them. Object should also be a GtkWidget."""
42 def __init__(self, types):
43 """Call this after initialising the widget.
44 Types is a list of MIME-types, or None to only accept files."""
46 targets = [('text/uri-list', 0, TARGET_URILIST)]
47 if types:
48 for type in types + ['application/octet-stream']:
49 targets.append((type, 0, TARGET_RAW))
51 self.targets = targets
52 self.xds_proxy_for(self)
54 def xds_proxy_for(self, widget):
55 "Handle drops on this widget as if they were to 'self'."
56 # (Konqueror requires ACTION_MOVE)
57 widget.drag_dest_set(g.DEST_DEFAULT_MOTION | g.DEST_DEFAULT_HIGHLIGHT,
58 self.targets,
59 gdk.ACTION_COPY | gdk.ACTION_MOVE | gdk.ACTION_PRIVATE)
61 widget.connect('drag-data-received', self.xds_data_received)
62 widget.connect('drag-drop', self.xds_drag_drop)
64 def xds_drag_drop(self, widget, context, data, info, time):
65 """Called when something is dropped on us. Decide which of the
66 offered targets to request and ask for it. xds_data_received will
67 be called when it finally arrives."""
68 target = widget.drag_dest_find_target(context, self.targets)
69 context.rox_leafname = None
70 if target is None:
71 # Error?
72 context.drop_finish(False, time)
73 else:
74 if provides(context, 'XdndDirectSave0'):
75 import saving
76 context.rox_leafname = saving._read_xds_property(context, False)
77 widget.drag_get_data(context, target, time)
78 return True
80 def xds_data_received(self, widget, context, x, y, selection, info, time):
81 "Called when we get some data. Internal."
83 if info == TARGET_RAW:
84 try:
85 self.xds_load_from_selection(selection, context.rox_leafname)
86 except:
87 context.drop_finish(False, time)
88 raise
89 context.drop_finish(True, time)
90 return 1
91 if info != TARGET_URILIST:
92 return 0
94 uris = extract_uris(selection.data)
95 if not uris:
96 alert("Nothing to load!")
97 context.drop_finish(False, time)
98 return 1
100 try:
101 try:
102 self.xds_load_uris(uris)
103 except RemoteFiles:
104 if len(uris) != 1 or not provides(context, 'application/octet-stream'):
105 raise
106 widget.drag_get_data(context, 'application/octet-stream', time)
107 return 1 # Don't do drag_finish
108 except:
109 context.drop_finish(False, time)
110 rox.report_exception()
111 else:
112 context.drop_finish(True, time)
114 return 1
116 def xds_load_uris(self, uris):
117 """Try to load each URI in the list. Override this if you can handle URIs
118 directly. The default method passes each local path to xds_load_from_file()
119 and displays an error for anything else."""
120 paths = []
121 for uri in uris:
122 path = get_local_path(uri)
123 if path:
124 paths.append(path)
125 if len(paths) < len(uris):
126 raise RemoteFiles
127 for path in paths:
128 self.xds_load_from_file(path)
130 def xds_load_from_file(self, path):
131 """Try to load this local file. Override this if you have a better way
132 to load files. The default method opens the file and calls xds_load_from_stream()."""
133 try:
134 self.xds_load_from_stream(path, None, open(path, 'rb'))
135 except:
136 rox.report_exception()
138 def xds_load_from_selection(self, selection, leafname = None):
139 """Try to load this selection (data from another application). The default
140 puts the data in a cStringIO and calls xds_load_from_stream()."""
141 if selection.data is None:
142 g.gdk.beep() # Load aborted
143 return
144 from cStringIO import StringIO
145 type = str(selection.type)
146 self.xds_load_from_stream(leafname, type, StringIO(selection.data))
148 def xds_load_from_stream(self, name, type, stream):
149 """Called when we get any data sent via drag-and-drop in any way (local
150 file or remote application transfer). You should override this and do
151 something with the data. 'name' may be None (if the data is unnamed),
152 a leafname, or a full path or URI. 'type' is the MIME type, or None if
153 unknown."""
154 alert('Got some data, but missing code to handle it!\n\n(name="%s";type="%s")'
155 % (name, type))