Aktualizuj dynamicznie skalę widżetu Tkinter z portów Arduino przy użyciu pytona i firmaty


10

Mam problem z uzyskaniem wartości portów cyfrowych Arduino i ustawieniem tych wartości w skali widżetu Python Tkinter.

Używam Python i Arduino z Firmatą. Jestem w stanie uzyskać dostęp do mojej tablicy Arduino za pomocą kodu python. Na przykład w widżecie etykiet otrzymuję i ustawiam wartość portu analogowego Arduino w czasie rzeczywistym na etykietę jak w poniższym kodzie bez żadnych problemów:

import Tkinter
import pyfirmata

def onStartButtonPress():
    while True:
        if flag.get():
            analogReadLabel.config(text=str(a0.read()))
            analogReadLabel.update_idletasks()
            top.update()
        else:
            break
    board.exit()
    top.destroy()

def onExitButtonPress():
    flag.set(False)

port = 'COM7'
board = pyfirmata.Arduino(port)

it = pyfirmata.util.Iterator(board)
it.start()

a0 = board.get_pin('a:0:i')

top = Tkinter.Tk()
top.title("Reading Analog pins")

descriptionLabel = Tkinter.Label(top, text="Potentiometer input:- ")
descriptionLabel.grid(column=1, row=1)

analogReadLabel = Tkinter.Label(top, text="Press Start..")
analogReadLabel.grid(column=2, row=1)

flag = Tkinter.BooleanVar(top)
flag.set(True)

startButton = Tkinter.Button(top, text="Start", command=onStartButtonPress)
startButton.grid(column=1, row=2)

exitButton = Tkinter.Button(top, text="Exit", command=onExitButtonPress)
exitButton.grid(column=2, row=2)

top.mainloop()

Do tego momentu jest w porządku, a GUI pokazuje mi coś takiego:

Interfejs Python Gui odczytuje wartość w czasie rzeczywistym z analogu 0

Ale to, co próbuję zrobić, a czego nie mam, to:

Co próbuję zaktualizować Skalę za pomocą cyfrowych pinów

Oto kod:

import Tkinter
import pyfirmata
import serial; 

def onStartButtonPress():
    while True:
        if flag.get():
            analogReadLabel.config(text=str(d8.read()))
            analogReadLabel1.config(text=str(d9.read()))
            analogReadLabel2.config(text=str(d10.read()))
            analogReadLabel.update_idletasks()

        pos1 = d8.read()
        if pos1 == True:
            pos1 = int(pos1)
            pos1 = 0
            brightnessScale.set(pos1)

        pos2 = d9.read()
        if pos2 == True:
            pos2 = int(pos2)
            pos2 = 100
            brightnessScale.set(pos2)

       ''' and so on '''

        brightnessScale.update_idletasks()
        top.update()
    else:
        break
board.exit()
top.destroy()


def onExitButtonPress():
    flag.set(False)

port = 'COM7'
board = pyfirmata.Arduino(port)

it = pyfirmata.util.Iterator(board)
it.start()

a0 = board.get_pin('a:0:i')

d4 = board.get_pin('d:4:i')
d5 = board.get_pin('d:5:i')
d6 = board.get_pin('d:6:i')
d7 = board.get_pin('d:7:i')
d8 = board.get_pin('d:8:i')
d9 = board.get_pin('d:9:i')
d10 = board.get_pin('d:10:i')

top = Tkinter.Tk()
top.geometry("800x600")
top.title("Reading Analog pins")

descriptionLabel = Tkinter.Label(top, text="Potentiometer input:- ")
descriptionLabel.grid(column=1, row=1)

analogReadLabel = Tkinter.Label(top, text="Level 1")
analogReadLabel.grid(column=2, row=1)

analogReadLabel1 = Tkinter.Label(top, text="Level 2")
analogReadLabel1.grid(column=3, row=1)

analogReadLabel2 = Tkinter.Label(top, text="Level 3")
analogReadLabel2.grid(column=4, row=1)

brightnessScale = Tkinter.Scale(top, from_ = 500, 
                            to = 0, 
                            length = 500, 
                            width = "30",
                            tickinterval = 50, 

                            bg = "lightskyblue",
                            highlightcolor = "darkblue",
                            highlightbackground = "royalblue",
                            troughcolor = "darkblue",

                            state = Tkinter.DISABLED,
                            sliderlength = 50,
                            relief = "sunken",
                            label = "Volume do Reservatorio em %",
                            orient = Tkinter.VERTICAL)
brightnessScale.grid(column=1, row=5)
Tkinter.Label(top,text="Volume da Caixa em (%)").grid(column=1, row=6)

flag = Tkinter.BooleanVar(top)
flag.set(True)

startButton = Tkinter.Button(top, text="Start", command=onStartButtonPress)
startButton.grid(column=1, row=2)

exitButton = Tkinter.Button(top, text="Exit", command=onExitButtonPress)
exitButton.grid(column=2, row=2)

top.mainloop()

Podsumowując: Mam kontaktrony w każdym porcie cyfrowym zadeklarowanym w kodzie. Te kontaktrony są zwykle w stanie otwartym, kiedy umieszczam w pobliżu pole magnetyczne, zamyka się i uzyskuję zmianę tej wartości, jak z False na True. To, co próbuję zrobić, to zaktualizować widget skali Tkinter w czasie rzeczywistym jako etykiety, ale nie otrzymuję go.

Czy w kodzie Pythona jest jakaś sztuczka, która może rozwiązać mój problem? Ponieważ przeprowadziłem wiele badań w Internecie i nie mogłem tego zrobić, jak widać na obrazku 2, gdy zmienia się stan drugiego cyfrowego PIN, etykieta zmienia się na True, a w kodzie skali odpowiada to wartości 100, ale nadal jest na znaku 0.

Nie podoba mi się również użycie polecenia i zmiennych parametrów widżetu skali.

W skrócie, chcę zaktualizować w czasie rzeczywistym widżet skali Pythona w skali Tkinter zgodnie z wartościami z portów cyfrowych z Arduino za pomocą firmaty.

Używam Python 2.7

PS: dla kompletności:

To jest mój kod arduino (Szkic) bez firmaty, który działa zgodnie z oczekiwaniami.

int level; 
int position; 

void setup() {

pinMode(8, INPUT); 
pinMode(9, INPUT);
pinMode(10, INPUT);
pinMode(11, INPUT);
pinMode(12, INPUT);

Serial.begin(9600); // ativa a porta serial

void loop() {
digitalWrite(8, LOW); 
digitalWrite(9, LOW);
digitalWrite(10, LOW);
digitalWrite(11, LOW);
digitalWrite(12, LOW);


position = digitalRead(8); 
if ( position == HIGH) {level=0;}

position = digitalRead(9); 
if ( position == HIGH) {Level=100;}

position = digitalRead(10); 
if ( position == HIGH) {level=200;}

position = digitalRead(11); 
if ( position == HIGH) {level=300;}

position = digitalRead(12); 
if ( position == HIGH) {level=400;}

Serial.println(level); 
delay(500);

}

Mógłbym również pobrać te wartości poprzez serial w pythonie i użyć ich do aktualizacji skali w ten sposób, ale nie wiem czy w tej metodzie to też zadziała!

I to, co myślałem w Pythonie: Ale wciąż muszę się wiele nauczyć z algorytmów i struktur danych. Tylko pseudo kod:

def setScaleValue():
while True:
    if flag.get():
         pos1 = d4.read()
        if pos1 == True: #or 1 
        #Here is the problem i don't know for sure what comes in my digital port
        #If is a True bool or a 1 as HIGH value from arduino
        #And probably i don't know if it's possible to change the values of a scale in this way !!!
        #or if i'm mistaken the Cast stuff in Python
            pos1 = int(pos1)
            pos1 = 0
            brightnessScale.set(pos1)

        pos2 = d8.read()
        if pos2 == True:
            pos2 = int(pos2)
            pos2 = 100
            brightnessScale.set(pos2)

        brightnessScale.update_idletasks()
        top.update()
    else:
        break
board.exit()
top.destroy()

return


d4 = board.get_pin('d:4:i')
d5 = board.get_pin('d:5:i')
d6 = board.get_pin('d:6:i')
d7 = board.get_pin('d:7:i')
d8 = board.get_pin('d:8:i')

brightnessScale = Tkinter.Scale(top, from_ = 500, 
                            to = 0, 
                            length = 500, 
                            width = "30",
                            tickinterval = 50, 

                            bg = "lightskyblue",
                            highlightcolor = "darkblue",
                            highlightbackground = "royalblue",
                            troughcolor = "darkblue",

                            state = Tkinter.DISABLED,
                            sliderlength = 50,
                            relief = "sunken",
                            label = "Volume do Reservatorio em %",
                            orient = Tkinter.VERTICAL)
brightnessScale.grid(column=1, row=5)
Tkinter.Label(top,text="Volume da Caixa em (%)").grid(column=1, row=6)

trzeci plik jest zniekształcony, def setScaleValue (): nie jest poprawnie zdefiniowany, po definicji funkcji potrzebna jest zmiana poziomu wcięcia.
esoterik

Czy po uruchomieniu kodu możesz kliknąć przycisk Stop?
Mert Gülsoy,

Odpowiedzi:


-1

w twoim kodzie:

pos2 = d8.read()
print("pos2 is {} type {}".format(pos2, type(pos2)))  # This way, you'll know the var type you're getting on the console
    if posX == True:
        posX = int(posX)          # You then cast to an integer
        posX = 100                # You discard pos2 value by setting it to 100
        brightnessScale.set(posX) # You set the scale always to value 100

Dodając wiersz wydruku, możesz zapewnić wartość i typ zmiennej.
Następnie konieczne może być odpowiednie dostosowanie instrukcji if.
Strzeż się, co wtedy robisz, gdy te 2 wiersze potwierdzają posX, prawdopodobnie nie jest to to, co chcesz zrobić przed aktualizacją wagi.

Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.