Heating system interface for communication with Arduino and data graphing
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

188 lines
7.4 KiB

  1. # -*- coding: utf-8 -*-
  2. """
  3. Created on Wed Feb 15 10:44:49 2023
  4. @author: david
  5. """
  6. from threading import Thread #Funciones para hilos
  7. import collections #Colección de datos para graficarlos
  8. import matplotlib.pyplot as plt #Gráficación de datos
  9. import matplotlib.animation as animation #Animar la gráfica
  10. import time #Delays
  11. import serial #comunicación con el monitor serial de Arduino
  12. from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg #Crear una figura para insertar en tkinter
  13. import tkinter as tk #Para la interfaz
  14. #Variables a utilizar
  15. global isRun #Valor booleano de programa corriendo
  16. global on #Valor booleano del estatus del transistor
  17. on = False #Inicializa transistor apagado
  18. isReceiving = False #Valor booleano de recibiendo datos
  19. isRun = True
  20. dato1 = 0.0 #Variable del dato convertido a float
  21. serialPort = 'COM6' #Puerto al que está conectado el Arduino
  22. baudRate = 9600 #Baudios configurados en Arduino
  23. datos = 0.0 #Variable de dato a leer desde Arduino
  24. estado = "off" #Variable de estado que se enviará al serial
  25. #Función de lectura de datos en Arduino
  26. def leer_datos():
  27. time.sleep(1.0)
  28. arduino.reset_input_buffer() #limpia el buffer de datos
  29. print("leyendo")
  30. while(isRun):
  31. global isReceiving #valor booleano de recibiendo datos
  32. global dato1 #dato a recibir desde Arduino
  33. print("RECIBIENDO...")
  34. datos = arduino.readline().decode("utf-8").strip() #Lectura del dato
  35. print(datos)
  36. if(datos != ''): #Asegurar que se haya leído un dato
  37. dato1 = float(datos) #Conversión del dato en float
  38. print(dato1) #Datos en float
  39. var.set("TEMPERATURA: " + str(dato1) + " °C") #impresión del dato numérico en la interfaz
  40. isReceiving = True #Confirmación de que se ha recibido el dato
  41. #Función para iniciar la gráfica
  42. def iniciarGrafica(self, muestras,lines):
  43. global dato1
  44. data.append(dato1) #Agrega el dato a la colección
  45. lines.set_data(range(muestras), data) #Se grafica en la línea la colección de datos
  46. #Establecer conexión con Arduino
  47. def conectar_serial():
  48. global arduino #Variable que va aguardar la conexión
  49. try:
  50. arduino = serial.Serial(serialPort, baudRate) #Asignación del objeto a la variable
  51. arduino.timeout = 0.1 #Tiempo a esperar para que haya datos disponibles en el puerto serie
  52. time.sleep(0.5) #Tiempo muerto para permitir la conexión
  53. print("CONECTADO")
  54. btnStart.config(state = "normal")
  55. btnConectar.config(state = "disabled")
  56. btnManual.config(state = "normal")
  57. except:
  58. print("Error de conexión")
  59. #Hilo para la lectura de datos constante
  60. def iniciar_hilo():
  61. global thread #Variable donde guardar el hilo
  62. thread = Thread(target=leer_datos) #Creación del hilo con la función para leer los datos
  63. thread.start() #Inicio de la lectura
  64. btnStart.config(state = "disabled")
  65. btnPause.config(state = "normal")
  66. #Pestaña para guardar la gráfica
  67. def guardarGrafica():
  68. plt.savefig('miFigura.png')
  69. #Función de control Manual (ON/OFF controlado por el usuario)
  70. def control1():
  71. global on
  72. global estado
  73. global arduino
  74. if on == False: #Cuando el transistor está apagado
  75. on = True #Cambia la variable de condición
  76. btnManual.config(bg = "#55DE1E", text = "ON") #Cambia color y texto del botón
  77. estado = "on" #cambia la variable de estado
  78. arduino.write(estado.encode()) #Se envía encender al serial de Arduino
  79. else: #Cuando el transistor está encendido
  80. on = False #Cambia la variable de condición
  81. btnManual.config(bg = "red", text = "OFF") #Cambia color y texto del botón
  82. estado = "off" #Cambia la variable de estado
  83. arduino.write(estado.encode()) #Se envía apagar al serial de Arduino
  84. #Ventana para control según una temperatura dada (Automático)
  85. #def control2():
  86. # window3 = tk.Toplevel()
  87. # window3.resizable(width = 0, height = 0)
  88. # window3.title('Control ON/OFF')
  89. # window3.geometry('500x300')
  90. #Función para pausar la gráfica
  91. def pausar():
  92. anim.event_source.stop()
  93. btnResume.config(state = "normal")
  94. btnPause.config(state = "disabled")
  95. #Función para continuar graficando
  96. def reanudar():
  97. anim.event_source.start()
  98. btnResume.config(state = "disabled")
  99. btnPause.config(state = "normal")
  100. #Función para finaliar la conexión con Arduino
  101. def desconectar_serial():
  102. global isRun
  103. anim.event_source.stop()
  104. isRun = False
  105. arduino.close()
  106. btnPause.config(state = "disabled")
  107. btnResume.config(state = "disabled")
  108. #Variables a graficar
  109. muestras = 100 #Número de datos a visualizar
  110. data = collections.deque([0] * muestras, maxlen = muestras) #Rango de datos
  111. tiempoMuestreo = 100
  112. fig = plt.figure(facecolor = '0.94') #creación de la gráfica (figura)
  113. ax = plt.axes(xlim=(0,100), ylim=(-40, 150)) #Rango de ejes
  114. plt.title("Sensor 1 - Arduino")
  115. ax.set_xlabel("Muestras")
  116. ax.set_ylabel("Voltaje")
  117. lines = ax.plot([], [])[0] #Creación de la línea a graficar
  118. root= tk.Tk()
  119. root.title("Sistema de calentamiento")
  120. var = tk.StringVar()
  121. frame = tk.Frame(root, bd=2)
  122. frame.grid(column=0, row=3, columnspan=2, sticky="nsew")
  123. frame1 = tk.Frame(root)
  124. frame1.grid(column=0, row=1, columnspan=2, sticky="EW")
  125. frame2 = tk.Frame(root)
  126. frame2.grid(column=0, row=2, columnspan=2, sticky="EW")
  127. frame0 = tk.Frame(root)
  128. frame0.grid(column=0, row=0, columnspan=2, sticky="EW")
  129. root.columnconfigure(0, weight=1)
  130. root.columnconfigure(1, weight=1)
  131. #self.master.rowconfigure(0, weigh=1)
  132. #self.master.rowconfigure(1, weigh=1)
  133. #self.master.rowconfigure(2, weigh=1)
  134. root.rowconfigure(3, weigh=5)
  135. canvas = FigureCanvasTkAgg(fig, master=frame)
  136. canvas.get_tk_widget().pack(padx=0, pady=0, expand=True, fill='both')
  137. labelBlank = tk.Label(frame1, text="")
  138. labelBlank.grid(row=0, column=4, pady=2, padx=155)
  139. btnManual = tk.Button(frame1, text = "OFF", command = control1, bg = "red", state = "disabled")
  140. btnManual.grid(row=0, column=6, pady=2, padx=10)
  141. labelState = tk.Label(frame1, text="Transistor State:")
  142. labelState.grid(row=0, column=5, pady=2, padx=5)
  143. #btnOnOff = tk.Button(frame2, text = "ON/OFF", command = control2)
  144. #btnOnOff.grid(row=0, column=1, pady=2, padx=10)
  145. btnConectar = tk.Button(frame1, text = "Connect", command = conectar_serial, bg="#00F1FC")
  146. btnConectar.grid(row=0, column=0, pady=2, padx=10)
  147. btnStart = tk.Button(frame1, text = "Start", command = iniciar_hilo, bg="#008C17", state="disabled")
  148. btnStart.grid(row=0, column=1, pady=2, padx=10)
  149. btnPause = tk.Button(frame1, text = "Pause", command = pausar, bg="#E2E200", state="disabled")
  150. btnPause.grid(row=0, column=2, pady=2, padx=10)
  151. btnResume = tk.Button(frame1, text = "Resume", command = reanudar, bg="#00F428", state="disabled")
  152. btnResume.grid(row=0, column=3, pady=2, padx=10)
  153. btnDesconectar = tk.Button(frame2, text='Disconnect', command = desconectar_serial, bg="#FE5E5E")
  154. btnDesconectar.grid(row=0, column=0, pady=2, padx=10)
  155. labelData = tk.Label(frame2, textvariable=var)
  156. labelData.grid(row=0, column=4, pady=2, padx=500)
  157. barraMenu = tk.Menu(frame0)
  158. barra1 = tk.Menu(barraMenu)
  159. barra1.add_command(label="Guardar gráfica", command=guardarGrafica)
  160. barraMenu.add_cascade(label="Archivo", menu=barra1)
  161. root.config(menu=barraMenu)
  162. #Animación de la gráfica (figura, función que grafica la línea, argumentos para graficar, rango de la figura)
  163. anim = animation.FuncAnimation(fig, iniciarGrafica, fargs=(muestras, lines), interval = tiempoMuestreo, cache_frame_data=False)
  164. root.geometry('1000x600')
  165. root.mainloop()