forked from koopaduo2/Beam-GUI
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbeam_gui.py
515 lines (472 loc) · 25.2 KB
/
beam_gui.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
#GitHub: koopaduo2
#Reddit: koopaduo
#Optical Engineer
#Custom Raspberry Pi laser image acquisition and beam profiling GUI
#The code has been developed and tested on Raspberry Pi 4B & Raspbian + arducam & raspi HQ cameras
#Version: v1.1
#This software is made available via MIT license
#GitHub project link: https://github.com/koopaduo2/Beam-GUI
#required imports
from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtCore import QThread
import numpy as np
import matplotlib.pyplot as plt
import cv2
import os
import datetime
from picamera.array import PiRGBArray
from picamera import PiCamera
import time
import math
#ignore command line warnings
import warnings
warnings.filterwarnings("ignore")
#main GUI window definitions
class Ui_MainWindow(object):
#set camera resolution which will be passed through the whole program
W, H = 2592,1944
#setup UI elements
def setupUi(self, MainWindow):
MainWindow.setObjectName("Beam GUI")
#resize the window to fit conveniently on a Raspberry Pi desktop
MainWindow.setFixedSize(935, 666)
self.centralwidget = QtWidgets.QWidget(MainWindow)
self.centralwidget.setObjectName("centralwidget")
self.tabWidget = QtWidgets.QTabWidget(self.centralwidget)
#note: Qrect input params are: x, y, wx, wy
self.tabWidget.setGeometry(QtCore.QRect(14, 8, 907, 828))
self.tabWidget.setObjectName("tabWidget")
#first tab is for "Camera" view
self.tab = QtWidgets.QWidget()
self.tab.setObjectName("tab")
self.lineEdit = QtWidgets.QLineEdit(MainWindow)
self.lineEdit.setGeometry(QtCore.QRect(156, 44, 539, 25))
self.lineEdit.setObjectName("lineEdit")
self.lineEdit.setText("Root directory: "+os.getcwd())
self.label = QtWidgets.QLabel(MainWindow)
self.label.setGeometry(QtCore.QRect(122, 45, 65, 21))
self.label.setObjectName("label")
#first pushbutton is connected to Run, which starts the image acquisition
self.pushButton = QtWidgets.QPushButton(MainWindow)
self.pushButton.setGeometry(QtCore.QRect(703, 45, 55, 23))
self.pushButton.setObjectName("pushButton")
self.textEdit_2 = QtWidgets.QTextEdit(self.tab)
self.textEdit_2.setGeometry(QtCore.QRect(52, 660, 801, 100))
self.textEdit_2.setObjectName("textEdit_2")
self.tabWidget.addTab(self.tab, "")
#second tab is for "Beam"
self.tab_2 = QtWidgets.QWidget()
self.tab_2.setObjectName("tab_2")
self.tabWidget.addTab(self.tab_2, "")
MainWindow.setCentralWidget(self.centralwidget)
self.menubar = QtWidgets.QMenuBar(MainWindow)
self.menubar.setGeometry(QtCore.QRect(0, 0, 943, 21))
self.menubar.setObjectName("menubar")
MainWindow.setMenuBar(self.menubar)
self.statusbar = QtWidgets.QStatusBar(MainWindow)
self.statusbar.setObjectName("statusbar")
MainWindow.setStatusBar(self.statusbar)
#widgets for estimated power meter
#label says Power (mW), lcd shows power, line edit allows for manual power meter cal
#alibration, and pushbutton calibrates it
self.label_P = QtWidgets.QLabel(self.tab_2)
self.label_P.setGeometry(QtCore.QRect(20,160,101,41))
self.lcdNumber_P = QtWidgets.QLCDNumber(self.tab_2)
self.lcdNumber_P.setGeometry(QtCore.QRect(20, 190, 81, 41))
self.lcdNumber_P.display("0")
self.lineEdit_P = QtWidgets.QLineEdit(self.tab_2)
self.lineEdit_P.setGeometry(QtCore.QRect(20, 110, 80, 40))
self.lineEdit_P.setText("0")
self.pushButton_P = QtWidgets.QPushButton(self.tab_2)
self.pushButton_P.setGeometry(QtCore.QRect(20, 50, 80, 40))
#label for showing number of saturated pixels
self.label_sat = QtWidgets.QLabel(self.tab_2)
self.label_sat.setFont(QtGui.QFont('Any',12))
self.label_sat.setText("# Saturated Pixels: 0")
self.label_sat.setGeometry(QtCore.QRect(210,540,250,30))
#widgets for displaying centroid and estimated beam width
#labels say d4 sigma. lcd's show computed widths
self.label_centroid = QtWidgets.QLabel(self.tab_2)
self.label_centroid.setFont(QtGui.QFont('Any',12))
self.label_centroid.setText("Centroid (x,y) = 0, 0")
self.label_centroid.setGeometry(QtCore.QRect(550, 540, 250, 30))
self.label_dx = QtWidgets.QLabel(self.tab_2)
self.label_dx.setGeometry(QtCore.QRect(20,230,101,41))
self.lcdNumber_dx = QtWidgets.QLCDNumber(self.tab_2)
self.lcdNumber_dx.setGeometry(QtCore.QRect(20, 260, 81, 41))
self.lcdNumber_dx.display(0)
self.label_dy = QtWidgets.QLabel(self.tab_2)
self.label_dy.setGeometry(QtCore.QRect(20,300,101,41))
self.lcdNumber_dy = QtWidgets.QLCDNumber(self.tab_2)
self.lcdNumber_dy.setGeometry(QtCore.QRect(20, 330, 81, 41))
self.lcdNumber_dy.display(0)
#widgets for adjustable aperture
#labels say aperture x, y, radius. line edits allow for adjustable (digital) aperture
self.label_apx = QtWidgets.QLabel(self.tab_2)
self.label_apx.setGeometry(QtCore.QRect(20,380,101,41))
self.lineEdit_apx = QtWidgets.QLineEdit(self.tab_2)
self.lineEdit_apx.setGeometry(QtCore.QRect(20, 410, 80, 40))
self.lineEdit_apx.setText(str(int(self.W / 2)))
self.label_apy = QtWidgets.QLabel(self.tab_2)
self.label_apy.setGeometry(QtCore.QRect(20,440,101,41))
self.lineEdit_apy = QtWidgets.QLineEdit(self.tab_2)
self.lineEdit_apy.setGeometry(QtCore.QRect(20, 470, 80, 40))
self.lineEdit_apy.setText(str(int(self.H / 2)))
self.label_apr = QtWidgets.QLabel(self.tab_2)
self.label_apr.setGeometry(QtCore.QRect(20,500,111,40))
self.lineEdit_apr = QtWidgets.QLineEdit(self.tab_2)
self.lineEdit_apr.setGeometry(QtCore.QRect(20, 530, 80, 40))
self.lineEdit_apr.setText(str(int(self.H / 2 - 100)))
#widgets for saving data
self.pushButton_S = QtWidgets.QPushButton(MainWindow)
self.pushButton_S.setGeometry(QtCore.QRect(765, 45, 55, 23))
#widgets for logging data continuously
self.pushButton_L = QtWidgets.QPushButton(MainWindow)
self.pushButton_L.setGeometry(QtCore.QRect(827, 45, 55, 23))
#create an image frame for raw image (downsampled)
#the image frame hosts the "Camera" tab image
#the beam frame hosts the "Beam" tab image + processing
#the cb frame hosts the colorbar (manual containment to insert colorbar). Requires cb.png in root directory
self.image_frame = QtWidgets.QLabel(self.tab)
self.beam_frame = QtWidgets.QLabel(self.tab_2)
self.cb_frame = QtWidgets.QLabel(self.tab_2)
#cb.png is required to manually place a colorbar (containment for cv2 heat map and colorbar)
colorbar = cv2.imread("cb.png")
colorbar = cv2.cvtColor(colorbar, cv2.COLOR_RGB2BGR)
self.cb_frame.move(790,49)
imGUI = QtGui.QImage(colorbar.data, colorbar.shape[1], colorbar.shape[0], \
colorbar.shape[1]*3, QtGui.QImage.Format_RGB888)
self.cb_frame.setPixmap(QtGui.QPixmap.fromImage(imGUI))
self.retranslateUi(MainWindow)
self.tabWidget.setCurrentIndex(0)
#connect the pushbuttons to start image capture, calibrate power meter, save data
self.pushButton.clicked.connect(self.run)
self.pushButton_P.clicked.connect(self.cal)
self.pushButton_S.clicked.connect(self.save)
self.pushButton_L.clicked.connect(self.log)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
#set text for GUI elements
def retranslateUi(self, MainWindow):
_translate = QtCore.QCoreApplication.translate
MainWindow.setWindowTitle(_translate("MainWindow", "Beam GUI"))
self.label.setText(_translate("MainWindow", "Info:"))
self.pushButton.setText(_translate("MainWindow", "Run"))
self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab), _translate("MainWindow", "Camera"))
self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab_2), _translate("MainWindow", "Beam"))
self.label_P.setText(_translate("MainWindow", "Power (mW)"))
self.pushButton_P.setText(_translate("MainWindow", "Cal (mW)"))
self.label_dx.setText(_translate("MainWindow", "D4σx (μm)"))
self.label_dy.setText(_translate("MainWindow", "D4σy (μm)"))
self.label_apx.setText(_translate("MainWindow", "Aperture x"))
self.label_apy.setText(_translate("MainWindow", "Aperture y"))
self.label_apr.setText(_translate("MainWindow", "Ap. Radius"))
self.pushButton_S.setText(_translate("MainWindow", "Save"))
self.pushButton_L.setText(_translate("MainWindow", "Log"))
#run image acquisition and processing thread
RUNNING = False
def run(self):
if not self.RUNNING:
self.threadA = captureThread(self, self.W, self.H)
self.threadA.start()
self.RUNNING = True
else:
self.lineEdit.setText("System already running")
def cal(self):
if not self.RUNNING:
self.lineEdit.setText("Run the system before calibrating power measurement")
else:
if self.lineEdit_P.text() != "0":
self.threadA.cal()
else:
self.lineEdit.setText("Input reading from real power meter to calibrate")
def log(self):
if self.RUNNING:
if not self.threadA.LOGGING:
self.threadA.SAVE_NOW = True
self.threadA.LOGGING = True
self.pushButton_L.setText("Stop")
else:
self.threadA.SAVE_NOW = False
self.threadA.LOGGING = False
self.pushButton_L.setText("Log")
self.lineEdit.setText("Data logging stopped")
else:
self.lineEdit.setText("Run the system before logging data")
#save images as png (can support other filetypes if needed)
#save statistics as csv
def save(self):
if self.RUNNING:
self.threadA.SAVE_NOW = True
else:
self.lineEdit.setText("Run the system before saving data")
#thread which handles live image acquisition and beam image processing
#runs separately from main GUI thread to prevent hang ups
class captureThread(QThread):
#variables which can be accessed across functions and threads
image_live = np.empty(1) #live camera image
camera = None #camera variable for PiCamera
rawCapture = None #rawCapture variable for PiCamera
MainWindow = None #MainWindow passed to thread so thread can modify UI elements
SAVE_NOW = False #flag to save all data once
LOGGING = False #flag to continuously log data
FRAMES_INIT = False #used to set camera and beam frame sizes and locations to draw images on
pix_sum, factor_P, pix_max, sat_num = 0,0,0,0 #used for power meter calibration, estimation and detecting saturation
sat_num_allowed = 20 #number of allowed saturated pixels before considering the beam profile 'saturated'
count_x,count_y,count_r = 0,0,0 #used to reset aperture values if input is left blank
mask_x, mask_y, mask_r = 1296,972,880 #mask values for digital aperture. Changes based on text input
W = 0 #camera/image width to be set
H = 0 #camera/image height to be set
pixel_um = 1.55 #multiply a pixel width by 1.55 micron to get physical width #SENSOR DEPENDENT
#initialize camera and set main window for interaction between thread and MainWindow
def __init__(self, MainWindow, W, H):
QThread.__init__(self)
#set the camera resolution
self.W, self.H = W, H
self.MainWindow = MainWindow
self.init_camera()
#capture live images and convert to beam profile
def run(self):
while(1):
self.live_image()
self.beam()
#initialize camera settings
def init_camera(self):
#initialize the PiCamera and PiRGBArray
camera = PiCamera()
camera.resolution = (self.W,self.H)
rawCapture = PiRGBArray(camera, size=(self.W,self.H))
time.sleep(0.1)
#configure settings for the camera
#these settings were chosen to minimize exposure and gain for laser profiling
#even at minimum settings, the camera will most likely need a variable ND filter
camera.awb_mode = 'off' #AWB mode
camera.awb_gains = (3.1,3.1) #manual AWB gain settings
camera.brightness = 50 #Image brightness
camera.meter_mode = 'backlit' #Metering mode
camera.exposure_mode = 'off' #Prevent auto exposure so results are consistent
camera.exposure_compensation = 0 #Exposure compensation
camera.shutter_speed = 10 #Minimum shutter speed
#vflip should be set to True since the camera sensor is lensless and will not have the image inverted
#hflip can be set to True or False depending on how you want the image to move as you physically move
#the beam spot location on the sensor. It is ultimately an alignment convention
camera.vflip = True
camera.hflip = False
camera.iso = 1
camera.saturation = 0
#If ZOOM_BOOL is set to True, the camera capture will be zoomed in to a Region of Interest
#The ROI will be the full Field of View cropped by a ratio of crop_factor
#the roi_start_x, roi_start_y are the origin (top left corner) of the ROI
ZOOM_BOOL = False
if ZOOM_BOOL:
crop_factor = 0.4
roi_start_x = (1-crop_factor)/2
roi_start_y = (1-crop_factor)/2
camera.zoom = (roi_start_x,roi_start_y,crop_factor,crop_factor)
#If CAMERA_SETTINGS is set to True, the camera settings will be printed on camera init
CAMERA_SETTINGS=True
if CAMERA_SETTINGS:
print("AWB is "+str(camera.awb_mode))
print("AWB gain is "+str(camera.awb_gains))
print("Brightness is "+str(camera.brightness))
print("Aperture is "+str(camera.exposure_compensation))
print("Shutter speed is "+str(camera.shutter_speed))
print("Camera exposure speed is "+str(camera.exposure_speed))
print("Iso is "+str(camera.iso))
print("Camera digital gain is "+str(camera.digital_gain))
print("Camera analog gain is "+str(camera.analog_gain))
print("Camera v/h flip is "+str(camera.vflip)+", "+str(camera.hflip))
print("Camera contrast is "+str(camera.contrast))
print("Camera color saturation is "+str(camera.saturation))
print("Camera meter mode is "+str(camera.meter_mode))
if ZOOM_BOOL:
print("Camera crop factor is "+str(crop_factor))
else:
print("Camera crop is disabled")
#store the camera and capture for use by other functions
self.camera = camera
self.rawCapture = rawCapture
self.MainWindow.lineEdit.setText("Camera initialized! Image processing system running")
#capture an image from the camera and store to self.image_live
def img_capture(self):
self.camera.capture(self.rawCapture, format="bgr")
self.image_live = self.rawCapture.array
self.rawCapture.truncate(0)
#take camera capture and display live on "Camera" tab
def live_image(self):
#time printouts can be used for runtime optimization which directly translates to framerate of images
#A = datetime.datetime.now()
#take a raw capture
self.img_capture()
#downsample the image by scale factor 4 to fit on the GUI screen
scale = 4
imR = cv2.resize(self.image_live, (int(self.W/scale),int(self.H/scale)))
#set the image to the proper position on the window if not already done
if not self.FRAMES_INIT:
self.MainWindow.image_frame.move(125,60)
self.MainWindow.image_frame.resize(int(self.W/scale),int(self.H/scale))
imGUI = QtGui.QImage(imR.data, imR.shape[1], imR.shape[0], imR.shape[1]*3, QtGui.QImage.Format_RGB888)
self.MainWindow.image_frame.setPixmap(QtGui.QPixmap.fromImage(imGUI))
#B = datetime.datetime.now()
#print("Live image runtime: "+str(B-A))
#calibrate the estimated power meter using a real power meter and pixel sum
def cal(self):
#define saturation as having a certain number of fully bright pixels self.sat_num_allowed
if self.sat_num > self.sat_num_allowed:
self.MainWindow.lineEdit.setText("Calibration failed! Image is saturated")
elif 0 < self.pix_max:
measured_P = float(self.MainWindow.lineEdit_P.text())
self.factor_P = measured_P / self.pix_sum
self.MainWindow.lineEdit.setText("Power meter calibrated!")
else:
self.MainWindow.lineEdit.setText("Calibration failed! No beam detected")
#convert camera image to beam profile (rainbow map) and display on GUI
#compute metrics of beam (centroid, D4σ)
def beam(self):
#time printouts can be used for runtime optimization which directly translates to framerate of images
#A = datetime.datetime.now()
#set the aperture mask values to those input by the user in the text boxes
#if the text boxes are left blank for some time, they will default
try:
self.mask_x = int(self.MainWindow.lineEdit_apx.text())
self.count_x = 0
except:
if self.count_x == 3:
self.mask_x = int(self.W / 2)
self.MainWindow.lineEdit_apx.setText(str(self.mask_x))
self.count_x += 1
try:
self.mask_y = int(self.MainWindow.lineEdit_apy.text())
self.count_y = 0
except:
if self.count_y == 3:
self.mask_y = int(self.H / 2)
self.MainWindow.lineEdit_apy.setText(str(self.mask_y))
self.count_y += 1
try:
self.mask_r = int(self.MainWindow.lineEdit_apr.text())
self.count_r = 0
except:
if self.count_r == 3:
self.mask_r = int(self.H/2 - 100)
self.MainWindow.lineEdit_apr.setText(str(self.mask_r))
self.count_r += 1
#define a mask depending on aperture
mask = np.zeros([self.H,self.W])
mask = cv2.circle(mask, (self.mask_x,self.mask_y), self.mask_r, 255, -1)
#take grayscale version of the image for intensity profiling
image = cv2.cvtColor(self.image_live, cv2.COLOR_BGR2GRAY)
#copy and mask the image based on the circular aperture mask
image_m = np.copy(image)
image_m[mask==0] = 0
#approximate the power based on the total bit count and calibration factors
self.pix_sum = np.sum(image_m)
self.pix_max = image_m.max()
P_estimated = self.pix_sum * self.factor_P
self.MainWindow.lcdNumber_P.display(round(P_estimated))
#count the number of saturated pixels
self.sat_num = len(np.where(image_m==255)[0])
self.MainWindow.label_sat.setText("# Saturated Pixels: "+str(self.sat_num))
#compute the centroid and D4σ in pixel values if image is not empty
MOM = cv2.moments(image_m)
if MOM['m00'] != 0:
centroid_x = MOM['m10']/MOM['m00']
centroid_y = MOM['m01']/MOM['m00']
#note 1 pixel has physical dimension: pixel_um * pixel_um (= 1.55 um (micron) * 1.55 um for Raspi HQ Camera module)
#With no scaling (lens) the physical beam widths are then d4x (px) * 1.55 um, d4y (px) * 1.55 um
d4x = self.pixel_um*4*math.sqrt(abs(MOM['m20']/MOM['m00'] - centroid_x**2))
d4y = self.pixel_um*4*math.sqrt(abs(MOM['m02']/MOM['m00'] - centroid_y**2))
else:
centroid_x = self.mask_x
centroid_y = self.mask_y
d4x = 0
d4y = 0
self.MainWindow.label_centroid.setText("Centroid x,y: "+str(round(centroid_x))+", "+str(round(centroid_y)))
self.MainWindow.lcdNumber_dx.display(round(d4x))
self.MainWindow.lcdNumber_dy.display(round(d4y))
#take the negative of the image in grayscale space to apply cv2 rainbow map
image_n = 255 - image
beam = cv2.applyColorMap(image_n, cv2.COLORMAP_RAINBOW)
#save all data if SAVE_NOW is flagged by save button, then reset the flag
if self.SAVE_NOW:
savepath = os.path.join(os.getcwd(), "saves")
if not os.path.exists(savepath):
os.mkdir(savepath)
timestamp = datetime.datetime.now().strftime('%Y%m%d_%H%M%S')
filename1 = "camera_"+timestamp+".png"
filename2 = "beam_"+timestamp+".png"
filename3 = "stats_"+timestamp+".csv"
filename4 = "x_profile_"+timestamp+".png"
filename5 = "y_profile_"+timestamp+".png"
cv2.imwrite(os.path.join(savepath,filename1), self.image_live) #may need to cvt color
cv2.imwrite(os.path.join(savepath,filename2), beam)
statsfile = open(os.path.join(savepath,filename3), 'w')
statsfile.write("Image width (px), height (px)\n")
statsfile.write(str(self.W)+","+str(self.H)+"\n")
statsfile.write("Centroid x (px), y (px)\n")
statsfile.write(str(centroid_x)+","+str(centroid_y)+"\n")
statsfile.write("D4σ x, y\n")
statsfile.write(str(d4x)+","+str(d4y)+"\n")
statsfile.write("Aperture x (px), y (px), radius (px)\n")
statsfile.write(str(self.mask_x)+","+str(self.mask_y)+","+str(self.mask_r)+"\n")
statsfile.write("Estimated power (mW)\n")
statsfile.write(str(P_estimated)+"\n")
statsfile.write("Gray value max (ct), sum (ct), saturated pixels (ct)\n")
statsfile.write(str(self.pix_max)+","+str(self.pix_sum)+","+str(self.sat_num)+"\n")
statsfile.close()
x_prof = image[round(centroid_y),:]
plt.plot(range(len(x_prof)),x_prof)
plt.title('Beam profile along x-axis at y-centroid')
plt.xlim(0,len(x_prof)-1)
plt.ylim(0,255)
plt.xlabel('Pixel')
plt.ylabel('Intensity')
plt.savefig(os.path.join(savepath,filename4))
plt.close('all')
y_prof = image[:,round(centroid_x)]
plt.plot(range(len(y_prof)),y_prof)
plt.title('Beam profile along y-axis at x-centroid')
plt.xlim(0,len(y_prof)-1)
plt.ylim(0,255)
plt.xlabel('Pixel')
plt.ylabel('Intensity')
plt.savefig(os.path.join(savepath,filename5))
plt.close('all')
#only stop the saving if LOGGING is not enabled
#update info bar depending on whether logging or single save
if not self.LOGGING:
self.MainWindow.lineEdit.setText("Data saved to: "+savepath)
self.SAVE_NOW = False
else:
self.MainWindow.lineEdit.setText("Data logging to: "+savepath)
#convert to int by rounding
d4x, d4y, centroid_x, centroid_y = round(d4x), round(d4y), round(centroid_x), round(centroid_y)
#apply centroid line tracking
cv2.line(beam, (centroid_x,0), (centroid_x,self.H), (0,0,0), thickness=5)
cv2.line(beam, (0,centroid_y), (self.W,centroid_y), (0,0,0), thickness=5)
#downsample the image by scale factor 4 to fit on the GUI screen
scale = 4
beam_R = cv2.resize(beam, (int(self.W/scale),int(self.H/scale)))
beam_R = cv2.cvtColor(beam_R, cv2.COLOR_BGR2RGB)
#line below is to add aperture mask circle
#image is reduced by scale times, so center coordinate is mask_x/scale, mask_y/scale; radius is mask_r/scale
beam_R = cv2.circle(beam_R, (round(self.mask_x/scale),round(self.mask_y/scale)),\
int(self.mask_r/scale), (0,0,0), 2)
#set the image to the proper position on the window if not already done
if not self.FRAMES_INIT:
self.MainWindow.beam_frame.move(125,60)
self.MainWindow.beam_frame.resize(int(self.W/scale),int(self.H/scale))
self.FRAMES_INIT = True
imGUI = QtGui.QImage(beam_R.data, beam_R.shape[1], beam_R.shape[0], \
beam_R.shape[1]*3, QtGui.QImage.Format_RGB888)
self.MainWindow.beam_frame.setPixmap(QtGui.QPixmap.fromImage(imGUI))
#B = datetime.datetime.now()
#print("Beam runtime: "+str(B-A))
#run the GUI
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
MainWindow = QtWidgets.QMainWindow()
ui = Ui_MainWindow()
ui.setupUi(MainWindow)
MainWindow.show()
sys.exit(app.exec_())