मैं एक अजगर स्क्रिप्ट के माध्यम से स्क्रीनशॉट लेना चाहता हूं और विनीत रूप से इसे सहेजना चाहता हूं।
मुझे केवल Linux समाधान में दिलचस्पी है, और किसी भी X आधारित वातावरण का समर्थन करना चाहिए।
मैं एक अजगर स्क्रिप्ट के माध्यम से स्क्रीनशॉट लेना चाहता हूं और विनीत रूप से इसे सहेजना चाहता हूं।
मुझे केवल Linux समाधान में दिलचस्पी है, और किसी भी X आधारित वातावरण का समर्थन करना चाहिए।
जवाबों:
यह स्क्रोट या इमेजमैजिक का उपयोग किए बिना काम करता है।
import gtk.gdk
w = gtk.gdk.get_default_root_window()
sz = w.get_size()
print "The size of the window is %d x %d" % sz
pb = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB,False,8,sz[0],sz[1])
pb = pb.get_from_drawable(w,w.get_colormap(),0,0,0,0,sz[0],sz[1])
if (pb != None):
pb.save("screenshot.png","png")
print "Screenshot saved to screenshot.png."
else:
print "Unable to get the screenshot."
से उधार http://ubuntuforums.org/showpost.php?p=2681009&postcount=5
एक कक्षा में सभी उत्तरों को संकलित करें। पीआईएल छवि आउटपुट।
#!/usr/bin/env python
# encoding: utf-8
"""
screengrab.py
Created by Alex Snet on 2011-10-10.
Copyright (c) 2011 CodeTeam. All rights reserved.
"""
import sys
import os
import Image
class screengrab:
def __init__(self):
try:
import gtk
except ImportError:
pass
else:
self.screen = self.getScreenByGtk
try:
import PyQt4
except ImportError:
pass
else:
self.screen = self.getScreenByQt
try:
import wx
except ImportError:
pass
else:
self.screen = self.getScreenByWx
try:
import ImageGrab
except ImportError:
pass
else:
self.screen = self.getScreenByPIL
def getScreenByGtk(self):
import gtk.gdk
w = gtk.gdk.get_default_root_window()
sz = w.get_size()
pb = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB,False,8,sz[0],sz[1])
pb = pb.get_from_drawable(w,w.get_colormap(),0,0,0,0,sz[0],sz[1])
if pb is None:
return False
else:
width,height = pb.get_width(),pb.get_height()
return Image.fromstring("RGB",(width,height),pb.get_pixels() )
def getScreenByQt(self):
from PyQt4.QtGui import QPixmap, QApplication
from PyQt4.Qt import QBuffer, QIODevice
import StringIO
app = QApplication(sys.argv)
buffer = QBuffer()
buffer.open(QIODevice.ReadWrite)
QPixmap.grabWindow(QApplication.desktop().winId()).save(buffer, 'png')
strio = StringIO.StringIO()
strio.write(buffer.data())
buffer.close()
del app
strio.seek(0)
return Image.open(strio)
def getScreenByPIL(self):
import ImageGrab
img = ImageGrab.grab()
return img
def getScreenByWx(self):
import wx
wx.App() # Need to create an App instance before doing anything
screen = wx.ScreenDC()
size = screen.GetSize()
bmp = wx.EmptyBitmap(size[0], size[1])
mem = wx.MemoryDC(bmp)
mem.Blit(0, 0, size[0], size[1], screen, 0, 0)
del mem # Release bitmap
#bmp.SaveFile('screenshot.png', wx.BITMAP_TYPE_PNG)
myWxImage = wx.ImageFromBitmap( myBitmap )
PilImage = Image.new( 'RGB', (myWxImage.GetWidth(), myWxImage.GetHeight()) )
PilImage.fromstring( myWxImage.GetData() )
return PilImage
if __name__ == '__main__':
s = screengrab()
screen = s.screen()
screen.show()
getScreenByWx
विधि विफल है wx._core.PyNoAppError: The wx.App object must be created first!
। मजेदार रूप से पर्याप्त है, कोड ठीक काम करता है यदि आप इसे लाइन द्वारा अजगर शेल में लाइन में प्रवेश कर रहे हैं, लेकिन एक स्क्रिप्ट में यह विफल रहता है।
getScreenByWx
आप में एक) myBitmap
द्वारा प्रतिस्थापित किया जाना चाहिए bmp
और ख) wx.App()
एक चर में सहेजें । को getScreenByGtk
प्रतिस्थापित (pb != None)
करने में pb is None
। और Qt का उपयोग न करें - आप दो बार नहीं बना सकते हैं QApplication
- आपका ऐप दूसरी बार बनाने की कोशिश में दुर्घटनाग्रस्त हो जाएगा।
बस पूर्णता के लिए: Xlib - लेकिन यह पूरी स्क्रीन को कैप्चर करते समय कुछ धीमा है:
from Xlib import display, X
import Image #PIL
W,H = 200,200
dsp = display.Display()
root = dsp.screen().root
raw = root.get_image(0, 0, W,H, X.ZPixmap, 0xffffffff)
image = Image.fromstring("RGB", (W, H), raw.data, "raw", "BGRX")
image.show()
पाइक्लिब में अड़चन-फाइलों में कुछ प्रकारों को फेंकने की कोशिश कर सकता है, और फिर साइथन का उपयोग करके इसे संकलित कर सकता है। जिससे स्पीड थोड़ी बढ़ सके।
संपादित करें: हम C में फ़ंक्शन का मूल लिख सकते हैं, और फिर इसे ctypes से अजगर में उपयोग कर सकते हैं, यहाँ कुछ ऐसा है जिसे मैंने एक साथ हैक किया है:
#include <stdio.h>
#include <X11/X.h>
#include <X11/Xlib.h>
//Compile hint: gcc -shared -O3 -lX11 -fPIC -Wl,-soname,prtscn -o prtscn.so prtscn.c
void getScreen(const int, const int, const int, const int, unsigned char *);
void getScreen(const int xx,const int yy,const int W, const int H, /*out*/ unsigned char * data)
{
Display *display = XOpenDisplay(NULL);
Window root = DefaultRootWindow(display);
XImage *image = XGetImage(display,root, xx,yy, W,H, AllPlanes, ZPixmap);
unsigned long red_mask = image->red_mask;
unsigned long green_mask = image->green_mask;
unsigned long blue_mask = image->blue_mask;
int x, y;
int ii = 0;
for (y = 0; y < H; y++) {
for (x = 0; x < W; x++) {
unsigned long pixel = XGetPixel(image,x,y);
unsigned char blue = (pixel & blue_mask);
unsigned char green = (pixel & green_mask) >> 8;
unsigned char red = (pixel & red_mask) >> 16;
data[ii + 2] = blue;
data[ii + 1] = green;
data[ii + 0] = red;
ii += 3;
}
}
XDestroyImage(image);
XDestroyWindow(display, root);
XCloseDisplay(display);
}
और फिर अजगर-फाइल:
import ctypes
import os
from PIL import Image
LibName = 'prtscn.so'
AbsLibPath = os.path.dirname(os.path.abspath(__file__)) + os.path.sep + LibName
grab = ctypes.CDLL(AbsLibPath)
def grab_screen(x1,y1,x2,y2):
w, h = x2-x1, y2-y1
size = w * h
objlength = size * 3
grab.getScreen.argtypes = []
result = (ctypes.c_ubyte*objlength)()
grab.getScreen(x1,y1, w, h, result)
return Image.frombuffer('RGB', (w, h), result, 'raw', 'RGB', 0, 1)
if __name__ == '__main__':
im = grab_screen(0,0,1440,900)
im.show()
#include <X11/Xutil.h>
इसके बजाय करना पड़ा #include <X11/Xlib.h>
। संकलन के लिए भी, मुझे -lX11
इस तरह अंत में जाना था gcc -shared -O3 -Wall -fPIC -Wl,-soname,prtscn -o prtscn.so prtscn.c -lX11
:।
यह एक X11 पर काम करता है, और शायद विंडोज पर भी (कोई, कृपया जाँच करें)। आवश्यकताएं PyQt4 :
import sys
from PyQt4.QtGui import QPixmap, QApplication
app = QApplication(sys.argv)
QPixmap.grabWindow(QApplication.desktop().winId()).save('test.png', 'png')
मेरे पास एक रैपर प्रोजेक्ट ( pyscreenshot ) है जो scrot, imagemagick, pyqt, wx और pygtk के लिए है। यदि आपके पास उनमें से एक है, तो आप इसका उपयोग कर सकते हैं। इस चर्चा से सभी समाधान शामिल हैं।
इंस्टॉल:
easy_install pyscreenshot
उदाहरण:
import pyscreenshot as ImageGrab
# fullscreen
im=ImageGrab.grab()
im.show()
# part of the screen
im=ImageGrab.grab(bbox=(10,10,500,500))
im.show()
# to file
ImageGrab.grab_to_file('im.png')
pyscreenshot.tempexport.RunProgError: No protocol specified giblib error: Can't open X display. It *is* running, yeah?" timeout_happened=False>
क्रॉस प्लेटफॉर्म सॉल्यूशन का उपयोग करते हुए wxPython :
import wx
wx.App() # Need to create an App instance before doing anything
screen = wx.ScreenDC()
size = screen.GetSize()
bmp = wx.EmptyBitmap(size[0], size[1])
mem = wx.MemoryDC(bmp)
mem.Blit(0, 0, size[0], size[1], screen, 0, 0)
del mem # Release bitmap
bmp.SaveFile('screenshot.png', wx.BITMAP_TYPE_PNG)
import ImageGrab
img = ImageGrab.grab()
img.save('test.jpg','JPEG')
इसके लिए अजगर इमेजिंग लाइब्रेरी की आवश्यकता होती है
आप इसका उपयोग कर सकते हैं
import os
os.system("import -window root screen_shot.png")
मैं लिनक्स में pyscreenshot या scrot के साथ स्क्रीनशॉट नहीं ले सका क्योंकि आउटपुट pyscreenshot
सिर्फ एक काली स्क्रीन png छवि फ़ाइल थी।
लेकिन भगवान का शुक्र है कि लिनक्स में स्क्रीनशॉट लेने के लिए कुछ भी स्थापित किए बिना एक और बहुत आसान तरीका था। बस अपनी निर्देशिका में नीचे कोड डाल दिया और साथ चलाpython demo.py
import os
os.system("gnome-screenshot --file=this_directory.png")
इसके लिए भी कई उपलब्ध विकल्प हैं gnome-screenshot --help
Application Options:
-c, --clipboard Send the grab directly to the clipboard
-w, --window Grab a window instead of the entire screen
-a, --area Grab an area of the screen instead of the entire screen
-b, --include-border Include the window border with the screenshot
-B, --remove-border Remove the window border from the screenshot
-p, --include-pointer Include the pointer with the screenshot
-d, --delay=seconds Take screenshot after specified delay [in seconds]
-e, --border-effect=effect Effect to add to the border (shadow, border, vintage or none)
-i, --interactive Interactively set options
-f, --file=filename Save screenshot directly to this file
--version Print version information and exit
--display=DISPLAY X display to use
से इस सूत्र :
import os
os.system("import -window root temp.png")
मेरे लिए यह काम ubuntu के लिए, आप इसके साथ चुनिंदा विंडो का स्क्रीनशॉट ले सकते हैं:
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gdk
from gi.repository import GdkPixbuf
import numpy as np
from Xlib.display import Display
#define the window name
window_name = 'Spotify'
#define xid of your select 'window'
def locate_window(stack,window):
disp = Display()
NET_WM_NAME = disp.intern_atom('_NET_WM_NAME')
WM_NAME = disp.intern_atom('WM_NAME')
name= []
for i, w in enumerate(stack):
win_id =w.get_xid()
window_obj = disp.create_resource_object('window', win_id)
for atom in (NET_WM_NAME, WM_NAME):
window_name=window_obj.get_full_property(atom, 0)
name.append(window_name.value)
for l in range(len(stack)):
if(name[2*l]==window):
return stack[l]
window = Gdk.get_default_root_window()
screen = window.get_screen()
stack = screen.get_window_stack()
myselectwindow = locate_window(stack,window_name)
img_pixbuf = Gdk.pixbuf_get_from_window(myselectwindow,*myselectwindow.get_geometry())
Pixbuf को सरणी में बदलना
def pixbuf_to_array(p):
w,h,c,r=(p.get_width(), p.get_height(), p.get_n_channels(), p.get_rowstride())
assert p.get_colorspace() == GdkPixbuf.Colorspace.RGB
assert p.get_bits_per_sample() == 8
if p.get_has_alpha():
assert c == 4
else:
assert c == 3
assert r >= w * c
a=np.frombuffer(p.get_pixels(),dtype=np.uint8)
if a.shape[0] == w*c*h:
return a.reshape( (h, w, c) )
else:
b=np.zeros((h,w*c),'uint8')
for j in range(h):
b[j,:]=a[r*j:r*j+w*c]
return b.reshape( (h, w, c) )
beauty_print = pixbuf_to_array(img_pixbuf)
यह एक पुराना सवाल है। मैं नए उपकरणों का उपयोग करके इसका उत्तर देना चाहूंगा।
अजगर 3 के साथ काम करता है (अजगर 2 के साथ काम करना चाहिए, लेकिन मैंने इसे परीक्षण नहीं किया है) और PyQt5।
न्यूनतम काम करने का उदाहरण। इसे अजगर शेल में कॉपी करें और परिणाम प्राप्त करें।
from PyQt5.QtWidgets import QApplication
app = QApplication([])
screen = app.primaryScreen()
screenshot = screen.grabWindow(QApplication.desktop().winId())
screenshot.save('/tmp/screenshot.png')
कोशिश करो:
#!/usr/bin/python
import gtk.gdk
import time
import random
import socket
import fcntl
import struct
import getpass
import os
import paramiko
while 1:
# generate a random time between 120 and 300 sec
random_time = random.randrange(20,25)
# wait between 120 and 300 seconds (or between 2 and 5 minutes)
print "Next picture in: %.2f minutes" % (float(random_time) / 60)
time.sleep(random_time)
w = gtk.gdk.get_default_root_window()
sz = w.get_size()
print "The size of the window is %d x %d" % sz
pb = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB,False,8,sz[0],sz[1])
pb = pb.get_from_drawable(w,w.get_colormap(),0,0,0,0,sz[0],sz[1])
ts = time.asctime( time.localtime(time.time()) )
date = time.strftime("%d-%m-%Y")
timer = time.strftime("%I:%M:%S%p")
filename = timer
filename += ".png"
if (pb != None):
username = getpass.getuser() #Get username
newpath = r'screenshots/'+username+'/'+date #screenshot save path
if not os.path.exists(newpath): os.makedirs(newpath)
saveas = os.path.join(newpath,filename)
print saveas
pb.save(saveas,"png")
else:
print "Unable to get the screenshot."
while
लूप है जो कभी बाहर नहीं निकलता (और 1
इसके बजाय उपयोग करता है True
), if (pb != None):
बस के बजाय है if pb:
, कुछ बेकार तार हैं।