EDIT FOR T KINTER:
IDE is Visual Studio Code
Traceback call is printed below scripts
TkinterTest.py
#!/usr/bin/env python3
from tkinter import *
from tkinter import ttk
import Ev3_Motor
ev3 = Ev3_Motor.Ev3_Motor()
def calculate(*args):
ev3.TestFunction("SUCCESSS YAHOOOOOO")
print("command to robot >>>>")
root = Tk()
root.title("TEST TKINTER")
mainframe = ttk.Frame(root, padding="3 3 12 12")
mainframe.grid(column=0, row=0, sticky=(N, W, E, S))
root.columnconfigure(0, weight=1)
root.rowconfigure(0, weight=1)
ttk.Button(mainframe, text="TEST BUTTON", command=calculate).grid(column=3, row=3, sticky=W)
#ttk.Label(mainframe, text="feet").grid(column=3, row=1, sticky=W)
for child in mainframe.winfo_children(): child.grid_configure(padx=5, pady=5)
root.bind('<Return>', calculate)
root.mainloop()
Ev3_Motor.py
#!/usr/bin/env python3
from ev3dev.ev3 import *
import os
import sys
from time import sleep
import shutil
import fileinput
os.system('setfont Lat15-TerminusBold14')
## FUNCTIONS ##
def __init(self):
debug_print("Constructor Ev3")
def TestFunction(randomString):
debug_print("Connection established: " + randomString)
TRACEBACK ERROR:
Starting: brickrun --directory="/home/robot/vscode-hello-python-master/Ev3" "/home/robot/vscode-hello-python-master/Ev3/TkinterTest.py"
Started.
----------
Traceback (most recent call last):
File "/usr/lib/python3.5/tkinter/__init__.py", line 36, in <module>
import _tkinter
ImportError: No module named '_tkinter'
During handling of the above exception, another exception occurred:
Traceback (most recent call last):
File "/home/robot/vscode-hello-python-master/Ev3/TkinterTest.py", line 3, in <module>
from tkinter import *
File "/usr/lib/python3.5/tkinter/__init__.py", line 38, in <module>
raise ImportError(str(msg) + ', please install the python3-tk package')
ImportError: No module named '_tkinter', please install the python3-tk package
----------
Exited with error code 1.
** ORIGINAL QUESTION: What I'm trying to do:**
I'm trying to create a program design where a UI program created with the Turtle Graphics Library in Python communicates directly with an EV3 Python Program that exists on the LEGO EV3 brick.
What I have so far:
What happened when I tried to get it to work:
It seemed like there's a clash in dependencies. Like turtle and ev3 are not compatible. What it seemed to me was that the FrontEnd.py file was trying to load into the RobotInstruction.py file and end up on the brick, which is not what I wanted.
Important Note:
Independently, these scripts work fine. For example, RobotInstruction.py can receive keyboard input to act on the motors. I just want to get that "command" firing from the graphical program instead
My first attempt at a Janky Super Inefficient Workaround:
-- THE EXISTING CODE EXCERPTS ARE ATTACHED AT THE END --
Use FrontEnd.py to write a string command to a file and have RobotInstruction.py constantly reading that file for a command and then based on the command, call the relevant function to turn motors.
What Worked:
Can successfully write to file with a command from FrontEnd.py
Can successfully read the command from the same file
BUT
It doesn't happen in real time. I'm not super familiar with file reading / writing with Python so much...so very possible that I may be doing something awkward...
My Question:
Is what I'm trying to do, possible? Can you click a turtle graphics created button to send a command to an ev3 robot? If so, how would I go about forming the CONNECTION between the 2 separate scripts?
CODE "EXCERPTS"
FrontEnd.py
def TurnTier(ButtonName):
if ButtonName == "TurnT1":
fileName = open("file1.txt", "w+")
fileName.write("TurnT1")
fileName.close()
RobotInstruction.py
while (not blnTierFound):
# file1.txt is written to from FrontEnd.py through a button click
# We are writing "TurnT1" into file1.txt
# Here we are opening the same file for reading to check for changes
fileName = open("file1.txt", "r+")
ButtonName = fileName.read()
fileName.close()
ButtonName = str(ButtonName)
if (ButtonName == "TurnT1"):
blnTierFound = True
strMotor = 'A'
# In the main part of the code
motorLeft = fncStartMotor(strMotor)
Important information:
Typically the EV3 is programmed with a block-based programming language from Lego. The default operating system is programmed with this language. In order to use a text-based programming language like Python to communicate with the robot, you must install a new operating system called ev3dev which is Linux based using a dual boot SD card. The complete setup instructions are available here. This setup is mandatory prior to running the below scripts.
After the discussion in the comments section, I've put together a solution that may work for you. This makes use of the Tkinter script in the question(which has been tested to work) and the Ev3_Motor script has been modified to include an Ev3_Motor class(which makes it easy to import the script and create an object of this class). However, this script is untested and may produce other errors as I do not have an Ev3 robot. These errors can be debugged later. Make sure Ev3_Motor.py is in the same directory as TkinterTest.py.
Ev3_Motor.py
#!/usr/bin/env python3
from ev3dev.ev3 import *
import os
import sys
from time import sleep
import shutil
import fileinput
import debug_print
os.system('setfont Lat15-TerminusBold14')
## Main Ev3 Motor Class ##
class Ev3_Motor:
def __init__(self):
debug_print("Constructor Ev3")
def TestFunction(randomString):
debug_print("Connection established: " + randomString)
TkinterTest.py
#!/usr/bin/env python3
from tkinter import *
from tkinter import ttk
import Ev3_Motor
ev3 = Ev3_Motor.Ev3_Motor()
def calculate(*args):
ev3.TestFunction("SUCCESSS YAHOOOOOO")
print("command to robot >>>>")
root = Tk()
root.title("TEST TKINTER")
mainframe = ttk.Frame(root, padding="3 3 12 12")
mainframe.grid(column=0, row=0, sticky=(N, W, E, S))
root.columnconfigure(0, weight=1)
root.rowconfigure(0, weight=1)
ttk.Button(mainframe, text="TEST BUTTON", command=calculate).grid(column=3, row=3, sticky=W)
#ttk.Label(mainframe, text="feet").grid(column=3, row=1, sticky=W)
for child in mainframe.winfo_children(): child.grid_configure(padx=5, pady=5)
root.bind('<Return>', calculate)
root.mainloop()
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With