##// END OF EJS Templates
[merge] Merging with upstream trunk.
Fernando Perez -
r2107:3611aca7 merge
parent child Browse files
Show More
@@ -0,0 +1,56 b''
1 """ Utilities for accessing the platform's clipboard.
2 """
3
4 import subprocess
5 import sys
6
7 from IPython.ipapi import TryNext
8
9
10 def win32_clipboard_get():
11 """ Get the current clipboard's text on Windows.
12
13 Requires Mark Hammond's pywin32 extensions.
14 """
15 try:
16 import win32clipboard
17 except ImportError:
18 message = ("Getting text from the clipboard requires the pywin32 "
19 "extensions: http://sourceforge.net/projects/pywin32/")
20 raise TryNext(message)
21 win32clipboard.OpenClipboard()
22 text = win32clipboard.GetClipboardData(win32clipboard.CF_TEXT)
23 # FIXME: convert \r\n to \n?
24 win32clipboard.CloseClipboard()
25 return text
26
27 def osx_clipboard_get():
28 """ Get the clipboard's text on OS X.
29 """
30 p = subprocess.Popen(['pbpaste', '-Prefer', 'ascii'],
31 stdout=subprocess.PIPE)
32 text, stderr = p.communicate()
33 # Text comes in with old Mac \r line endings. Change them to \n.
34 text = text.replace('\r', '\n')
35 return text
36
37 def tkinter_clipboard_get():
38 """ Get the clipboard's text using Tkinter.
39
40 This is the default on systems that are not Windows or OS X. It may
41 interfere with other UI toolkits and should be replaced with an
42 implementation that uses that toolkit.
43 """
44 try:
45 import Tkinter
46 except ImportError:
47 message = ("Getting text from the clipboard on this platform "
48 "requires Tkinter.")
49 raise TryNext(message)
50 root = Tkinter.Tk()
51 root.withdraw()
52 text = root.clipboard_get()
53 root.destroy()
54 return text
55
56
@@ -3261,6 +3261,61 b' Defaulting color scheme to \'NoColor\'"""'
3261 3261 page(self.shell.pycolorize(cont),
3262 3262 screen_lines=self.shell.rc.screen_length)
3263 3263
3264 def _rerun_pasted(self):
3265 """ Rerun a previously pasted command.
3266 """
3267 b = self.user_ns.get('pasted_block', None)
3268 if b is None:
3269 raise UsageError('No previous pasted block available')
3270 print "Re-executing '%s...' (%d chars)"% (b.split('\n',1)[0], len(b))
3271 exec b in self.user_ns
3272
3273 def _get_pasted_lines(self, sentinel):
3274 """ Yield pasted lines until the user enters the given sentinel value.
3275 """
3276 from IPython import iplib
3277 print "Pasting code; enter '%s' alone on the line to stop." % sentinel
3278 while True:
3279 l = iplib.raw_input_original(':')
3280 if l == sentinel:
3281 return
3282 else:
3283 yield l
3284
3285 def _strip_pasted_lines_for_code(self, raw_lines):
3286 """ Strip non-code parts of a sequence of lines to return a block of
3287 code.
3288 """
3289 # Regular expressions that declare text we strip from the input:
3290 strip_re = [r'^\s*In \[\d+\]:', # IPython input prompt
3291 r'^\s*(\s?>)+', # Python input prompt
3292 r'^\s*\.{3,}', # Continuation prompts
3293 r'^\++',
3294 ]
3295
3296 strip_from_start = map(re.compile,strip_re)
3297
3298 lines = []
3299 for l in raw_lines:
3300 for pat in strip_from_start:
3301 l = pat.sub('',l)
3302 lines.append(l)
3303
3304 block = "\n".join(lines) + '\n'
3305 #print "block:\n",block
3306 return block
3307
3308 def _execute_block(self, block, par):
3309 """ Execute a block, or store it in a variable, per the user's request.
3310 """
3311 if not par:
3312 b = textwrap.dedent(block)
3313 self.user_ns['pasted_block'] = b
3314 exec b in self.user_ns
3315 else:
3316 self.user_ns[par] = SList(block.splitlines())
3317 print "Block assigned to '%s'" % par
3318
3264 3319 def magic_cpaste(self, parameter_s=''):
3265 3320 """Allows you to paste & execute a pre-formatted code block from clipboard.
3266 3321
@@ -3286,50 +3341,60 b' Defaulting color scheme to \'NoColor\'"""'
3286 3341 will be what was just pasted.
3287 3342
3288 3343 IPython statements (magics, shell escapes) are not supported (yet).
3344
3345 See also
3346 --------
3347 %paste: automatically pull code from clipboard.
3289 3348 """
3349
3290 3350 opts,args = self.parse_options(parameter_s,'rs:',mode='string')
3291 3351 par = args.strip()
3292 3352 if opts.has_key('r'):
3293 b = self.user_ns.get('pasted_block', None)
3294 if b is None:
3295 raise UsageError('No previous pasted block available')
3296 print "Re-executing '%s...' (%d chars)"% (b.split('\n',1)[0], len(b))
3297 exec b in self.user_ns
3353 self._rerun_pasted()
3298 3354 return
3299 3355
3300 3356 sentinel = opts.get('s','--')
3301 3357
3302 # Regular expressions that declare text we strip from the input:
3303 strip_re = [r'^\s*In \[\d+\]:', # IPython input prompt
3304 r'^\s*(\s?>)+', # Python input prompt
3305 r'^\s*\.{3,}', # Continuation prompts
3306 r'^\++',
3307 ]
3358 block = self._strip_pasted_lines_for_code(
3359 self._get_pasted_lines(sentinel))
3308 3360
3309 strip_from_start = map(re.compile,strip_re)
3361 self._execute_block(block, par)
3362
3363 def magic_paste(self, parameter_s=''):
3364 """Allows you to paste & execute a pre-formatted code block from clipboard.
3310 3365
3311 from IPython import iplib
3312 lines = []
3313 print "Pasting code; enter '%s' alone on the line to stop." % sentinel
3314 while 1:
3315 l = iplib.raw_input_original(':')
3316 if l ==sentinel:
3317 break
3318
3319 for pat in strip_from_start:
3320 l = pat.sub('',l)
3321 lines.append(l)
3322
3323 block = "\n".join(lines) + '\n'
3324 #print "block:\n",block
3325 if not par:
3326 b = textwrap.dedent(block)
3327 self.user_ns['pasted_block'] = b
3328 exec b in self.user_ns
3329 else:
3330 self.user_ns[par] = SList(block.splitlines())
3331 print "Block assigned to '%s'" % par
3332
3366 The text is pulled directly from the clipboard without user
3367 intervention.
3368
3369 The block is dedented prior to execution to enable execution of method
3370 definitions. '>' and '+' characters at the beginning of a line are
3371 ignored, to allow pasting directly from e-mails, diff files and
3372 doctests (the '...' continuation prompt is also stripped). The
3373 executed block is also assigned to variable named 'pasted_block' for
3374 later editing with '%edit pasted_block'.
3375
3376 You can also pass a variable name as an argument, e.g. '%paste foo'.
3377 This assigns the pasted block to variable 'foo' as string, without
3378 dedenting or executing it (preceding >>> and + is still stripped)
3379
3380 '%paste -r' re-executes the block previously entered by cpaste.
3381
3382 IPython statements (magics, shell escapes) are not supported (yet).
3383
3384 See also
3385 --------
3386 %cpaste: manually paste code into terminal until you mark its end.
3387 """
3388 opts,args = self.parse_options(parameter_s,'r:',mode='string')
3389 par = args.strip()
3390 if opts.has_key('r'):
3391 self._rerun_pasted()
3392 return
3393
3394 text = self.shell.hooks.clipboard_get()
3395 block = self._strip_pasted_lines_for_code(text.splitlines())
3396 self._execute_block(block, par)
3397
3333 3398 def magic_quickref(self,arg):
3334 3399 """ Show a quick reference sheet """
3335 3400 import IPython.usage
@@ -44,6 +44,7 b' somewhere in your configuration files or ipython command line.'
44 44 from IPython import ipapi
45 45
46 46 import os,bisect
47 import sys
47 48 from genutils import Term,shell
48 49 from pprint import PrettyPrinter
49 50
@@ -53,7 +54,8 b' from pprint import PrettyPrinter'
53 54 __all__ = ['editor', 'fix_error_editor', 'synchronize_with_editor', 'result_display',
54 55 'input_prefilter', 'shutdown_hook', 'late_startup_hook',
55 56 'generate_prompt', 'generate_output_prompt','shell_hook',
56 'show_in_pager','pre_prompt_hook', 'pre_runcode_hook']
57 'show_in_pager','pre_prompt_hook', 'pre_runcode_hook',
58 'clipboard_get']
57 59 # vds: <<
58 60
59 61 pformat = PrettyPrinter().pformat
@@ -243,5 +245,20 b' def pre_prompt_hook(self):'
243 245 def pre_runcode_hook(self):
244 246 """ Executed before running the (prefiltered) code in IPython """
245 247 return None
246
247 248
249 def clipboard_get(self):
250 """ Get text from the clipboard.
251 """
252 from IPython.clipboard import (osx_clipboard_get, tkinter_clipboard_get,
253 win32_clipboard_get)
254 if sys.platform == 'win32':
255 chain = [win32_clipboard_get, tkinter_clipboard_get]
256 elif sys.platform == 'darwin':
257 chain = [osx_clipboard_get, tkinter_clipboard_get]
258 else:
259 chain = [tkinter_clipboard_get]
260 dispatcher = CommandChainDispatcher()
261 for func in chain:
262 dispatcher.add(func)
263 text = dispatcher()
264 return text
@@ -252,3 +252,39 b' class TestMagicRun(object):'
252 252
253 253 def teardown(self):
254 254 self.tmpfile.close()
255
256 # Multiple tests for clipboard pasting
257 def test_paste():
258
259 def paste(txt):
260 hooks.clipboard_get = lambda : txt
261 _ip.magic('paste')
262
263 # Inject fake clipboard hook but save original so we can restore it later
264 hooks = _ip.IP.hooks
265 user_ns = _ip.user_ns
266 original_clip = hooks.clipboard_get
267
268 try:
269 # Run tests with fake clipboard function
270 user_ns.pop('x', None)
271 paste('x=1')
272 yield (nt.assert_equal, user_ns['x'], 1)
273
274 user_ns.pop('x', None)
275 paste('>>> x=2')
276 yield (nt.assert_equal, user_ns['x'], 2)
277
278 paste("""
279 >>> x = [1,2,3]
280 >>> y = []
281 >>> for i in x:
282 ... y.append(i**2)
283 ...
284 """)
285 yield (nt.assert_equal, user_ns['x'], [1,2,3])
286 yield (nt.assert_equal, user_ns['y'], [1,4,9])
287
288 finally:
289 # Restore original hook
290 hooks.clipboard_get = original_clip
1 NO CONTENT: file was removed
General Comments 0
You need to be logged in to leave comments. Login now