KY-040 Kodierter Drehschalter (Rotary Encoder): Unterschied zwischen den Versionen

Aus Linkerkit.de

Wechseln zu: Navigation, Suche
(Pin-Belegung)
 
(3 dazwischenliegende Versionen des gleichen Benutzers werden nicht angezeigt)
Zeile 59: Zeile 59:
 
</div>
 
</div>
 
</div>
 
</div>
</div>
+
 
 
</div>
 
</div>
  
 +
</div>
 
==Pin-Belegung==
 
==Pin-Belegung==
[[Datei:5_CLK_DT_SW_V_G.png|457x294px|none]]
+
[[Datei:5_CLK_DT_SW_V_G.png|none|457x294px]]
  
 
==Codebeispiel Arduino==
 
==Codebeispiel Arduino==
<pre class="brush:cpp">int redPin = 2;
+
Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt. Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.
int yellowPin = 3;
+
 
int greenPin = 4;
+
Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben.
int aPin = 6;
+
Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.
int bPin = 7;
+
 
int buttonPin = 5;
+
'''Für die serielle Ausgabe: Baudrate= 115200'''
int state = 0;
+
 
int longPeriod = 5000; // Time at green or red
+
<pre class="brush:cpp">// Initialisierung benötigter Variablen
int shortPeriod = 700; // Time period when changing
+
int Counter = 0;  
int targetCount = shortPeriod;
+
boolean Richtung;
int count = 0;
+
int Pin_clk_Letzter;
void setup ()
+
int Pin_clk_Aktuell;
{
+
 
  pinMode (aPin, INPUT);
+
// Definition der Eingangs-Pins
  pinMode (bPin, INPUT);
+
int pin_clk = 3;
  pinMode (buttonPin, INPUT);
+
int pin_dt = 4;  
  pinMode (redPin, OUTPUT);
+
int button_pin = 5;
  pinMode (yellowPin, OUTPUT);
+
  pinMode (greenPin, OUTPUT);
+
}
+
void setup()  
void loop ()
+
{  
{
+
  // Eingangs-Pins werden initialisiert...
  count++;
+
  pinMode (pin_clk,INPUT);
  if (digitalRead (buttonPin))
+
  pinMode (pin_dt,INPUT);
  {
+
  pinMode (button_pin,INPUT);
    setLights (HIGH, HIGH, HIGH);
+
 
  }
+
  // ...und deren Pull-Up Widerstände aktiviert
  else
+
  digitalWrite(pin_clk, true);
  {
+
  digitalWrite(pin_dt, true);
    int change = getEncoderTurn ();
+
  digitalWrite(button_pin, true);
    int newPeriod = longPeriod + (change * 1000);
+
 
    if (newPeriod  >= 1000 && newPeriod <= 10000)   {     longPeriod = newPeriod;
+
  // Initiales Auslesen des Pin_CLK
    }
+
  Pin_clk_Letzter = digitalRead(pin_clk); 
    if (count> targetCount)
+
  Serial.begin (115200);
    {
+
}  
      setState ();
+
 
      count = 0;
+
// Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden
    }
+
// Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt.
  }
+
// Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen
  delay (1);
+
// Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.
}
+
// Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben.
int getEncoderTurn ()
+
// Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.
{
+
 
  // Return -1, 0, or +1
+
void loop()
  static int oldA = LOW;
+
{  
  static int oldB = LOW;
+
  // Auslesen des aktuellen Statuses 
  int result = 0;
+
  Pin_clk_Aktuell = digitalRead(pin_clk);
  int newA = digitalRead (aPin);
+
 
  int newB = digitalRead (bPin);
+
  // Überprüfung auf Änderung
  if (newA != oldA || newB != oldB)
+
  if (Pin_clk_Aktuell != Pin_clk_Letzter)
  {
+
  {
    //Something has changed
+
    if (oldA == LOW && newA == HIGH)
+
if (digitalRead(pin_dt) != Pin_clk_Aktuell)  
    {
+
{
      result = - (oldB * 2 - 1);
+
// Pin_CLK hat sich zuerst verändert
    }
+
Counter ++;
  }
+
Richtung = true;
  oldA = newA;
+
}  
  oldB = newB;
+
  return result;
+
else  
}
+
{       // Andernfalls hat sich Pin_DT zuerst verändert
int setState ()
+
Richtung = false;
{
+
Counter--;
  if (state == 0)
+
}
  {
+
Serial.println ("Drehung erkannt: ");
    setLights (HIGH, LOW, LOW);
+
Serial.print ("Drehrichtung: ");
    targetCount = longPeriod;
+
    state = 1;
+
if (Richtung)
  }
+
{
  else if (state == 1)
+
  Serial.println ("Im Uhrzeigersinn");
  {
+
}
    setLights (HIGH, HIGH, LOW);
+
else
     targetCount = shortPeriod;
+
{
     state = 2;
+
  Serial.println("Gegen den Uhrzeigersinn");
  }
+
}
  else if (state == 2)
+
  {
+
Serial.print("Aktuelle Position: ");
    setLights (LOW, LOW, HIGH);
+
Serial.println(Counter);
    targetCount = longPeriod;
+
Serial.println("------------------------------");
    state = 3;
+
  }
+
  }
  else if (state == 3)
+
 
  {
+
  // Vorbereitung für den nächsten Druchlauf:
    setLights (LOW, HIGH, LOW);
+
  // Der Wert des aktuellen Durchlaufs ist beim nächsten Druchlauf der vorherige Wert
     targetCount = shortPeriod;
+
  Pin_clk_Letzter = Pin_clk_Aktuell;
     state = 0;
+
 
  }
+
  // Reset-Funktion um aktuelle Position zu speichern
}
+
  if (!digitalRead(button_pin) && Counter!=0)
void setLights (int red, int yellow, int green)
+
    {
{
+
      Counter = 0;
  digitalWrite (redPin, red);
+
      Serial.println("Position resettet");
  digitalWrite (yellowPin, yellow);
+
    }
  digitalWrite (greenPin, green);
+
   
}
+
}
 +
 
 +
</pre>
 +
 
 +
'''Anschlussbelegung Arduino:'''
 +
 
 +
{| style="height: 58px; padding-left: 30px;" width="228"
 +
|-
 +
||CLK
 +
||=
 +
||[Pin 3]
 +
|-
 +
||DT
 +
||=
 +
||[Pin 4]
 +
|-
 +
||Button
 +
||=
 +
||[Pin 5]
 +
|-
 +
||+
 +
||=
 +
||[Pin 5V]
 +
|-
 +
||GND
 +
||=
 +
||[Pin GND]
 +
|}
 +
 
 +
'''Beispielprogramm Download'''
 +
 
 +
[[Medium:KY-040_RotaryEncoder.zip|KY-040_RotaryEncoder.zip]]
 +
 
 +
==Codebeispiel Raspberry Pi==
 +
Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt.Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.
 +
 
 +
Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben.
 +
Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.
 +
 
 +
<pre class="brush:py">
 +
# coding=utf-8
 +
# Benoetigte Module werden importiert und eingerichtet
 +
import RPi.GPIO as GPIO
 +
import time
 +
 
 +
GPIO.setmode(GPIO.BCM)
 +
 
 +
# Hier werden die Eingangs-Pins deklariert, an dem der Sensor angeschlossen ist.
 +
PIN_CLK = 16
 +
PIN_DT = 15
 +
BUTTON_PIN = 14
 +
 
 +
GPIO.setup(PIN_CLK, GPIO.IN, pull_up_down = GPIO.PUD_UP)
 +
GPIO.setup(PIN_DT, GPIO.IN, pull_up_down = GPIO.PUD_UP)
 +
GPIO.setup(BUTTON_PIN, GPIO.IN, pull_up_down = GPIO.PUD_UP)
 +
 
 +
# Benötigte Variablen werden initialisiert
 +
Counter = 0
 +
Richtung = True
 +
PIN_CLK_LETZTER = 0
 +
PIN_CLK_AKTUELL = 0
 +
delayTime = 0.01
 +
 
 +
# Initiales Auslesen des Pin_CLK
 +
PIN_CLK_LETZTER = GPIO.input(PIN_CLK)
 +
 
 +
# Diese AusgabeFunktion wird bei Signaldetektion ausgefuehrt
 +
def ausgabeFunktion(null):
 +
     global Counter
 +
 
 +
     PIN_CLK_AKTUELL = GPIO.input(PIN_CLK)
 +
 
 +
    if PIN_CLK_AKTUELL != PIN_CLK_LETZTER:
 +
 
 +
        if GPIO.input(PIN_DT) != PIN_CLK_AKTUELL:
 +
            Counter += 1
 +
            Richtung = True;
 +
        else:
 +
            Richtung = False
 +
            Counter = Counter - 1
 +
 
 +
        print "Drehung erkannt: "
 +
 
 +
        if Richtung:
 +
            print "Drehrichtung: Im Uhrzeigersinn"
 +
        else:
 +
            print "Drehrichtung: Gegen den Uhrzeigersinn"
 +
 
 +
        print "Aktuelle Position: ", Counter
 +
        print "------------------------------"
 +
 
 +
def CounterReset(null):
 +
     global Counter
 +
 
 +
     print "Position resettet!"
 +
    print "------------------------------"
 +
    Counter = 0
 +
 
 +
# Um einen Debounce direkt zu integrieren, werden die Funktionen zur Ausgabe mittels
 +
# CallBack-Option vom GPIO Python Modul initialisiert
 +
GPIO.add_event_detect(PIN_CLK, GPIO.BOTH, callback=ausgabeFunktion, bouncetime=50)
 +
GPIO.add_event_detect(BUTTON_PIN, GPIO.FALLING, callback=CounterReset, bouncetime=50)
 +
 
 +
 
 +
print "Sensor-Test [druecken Sie STRG+C, um den Test zu beenden]"
 +
 
 +
# Hauptprogrammschleife
 +
try:
 +
        while True:
 +
            time.sleep(delayTime)
 +
 
 +
# Aufraeumarbeiten nachdem das Programm beendet wurde
 +
except KeyboardInterrupt:
 +
        GPIO.cleanup()
 +
 
 +
 
 +
</pre>
 +
'''Anschlussbelegung Raspberry Pi:'''
 +
 
 +
{| style="height: 58px; padding-left: 30px;" width="228"
 +
|-
 +
||CLK
 +
||=
 +
||GPIO16
 +
||[Pin 36]
 +
|-
 +
||DT
 +
||=
 +
||GPIO15
 +
||[Pin 10]
 +
|-
 +
||SW
 +
||=
 +
||GPIO14
 +
||[Pin 8]
 +
|-
 +
||+
 +
||=
 +
||3,3V
 +
||[Pin 1]
 +
|-
 +
||GND
 +
||=
 +
||Masse
 +
||[Pin 6]
 +
|}
 +
'''Beispielprogramm Download'''
 +
 
 +
[[Medium:KY-040_RPi_RotaryEncoder.zip|KY-040_RPi_RotaryEncoder.zip]]
 +
 
 +
Zu starten mit dem Befehl:
 +
 
 +
<pre class="brush:bash">sudo python KY-040_RPi_RotaryEncoder.py
 
</pre>
 
</pre>

Aktuelle Version vom 8. April 2016, 15:01 Uhr

Bild

ky-040.jpg

Technische Daten / Kurzbeschreibung

Die aktuelle Position des Drehschalters wird kodiert über die Ausgänge gegeben.


Kodierung

Die Idee bei einem Drehschalter/Drehgeber ist es, dass zu jedem gedrehten "Schritt", sich der Zustand jeweils immer nur einer der beiden Ausgangs-Pins ändert. Je nachdem welhcer der beiden sich zuerst geändert hat, so kann man auf die Drehrichtung schließen, wenn man auf die folgende Kodierung achtet.

Im Uhrzeigersinn [A ändert sich zuerst] -> Pin_CLK

A B
0 0
1 0
1 1
0 1
0 0


Gegen den Uhrzeigersinn [B ändert sich zuerst] -> Pin_DT

A B
0 0
0 1
1 1
1 0
0 0

Pin-Belegung

5 CLK DT SW V G.png

Codebeispiel Arduino

Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt. Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.

Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben. Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.

Für die serielle Ausgabe: Baudrate= 115200

// Initialisierung benötigter Variablen
int Counter = 0; 
boolean Richtung;
int Pin_clk_Letzter;  
int Pin_clk_Aktuell;

// Definition der Eingangs-Pins
int pin_clk = 3;  
int pin_dt = 4; 
int button_pin = 5;
 
 
void setup() 
{ 
   // Eingangs-Pins werden initialisiert...
   pinMode (pin_clk,INPUT);
   pinMode (pin_dt,INPUT);
   pinMode (button_pin,INPUT);
   
   // ...und deren Pull-Up Widerstände aktiviert
   digitalWrite(pin_clk, true);
   digitalWrite(pin_dt, true);
   digitalWrite(button_pin, true);
   
   // Initiales Auslesen des Pin_CLK
   Pin_clk_Letzter = digitalRead(pin_clk);   
   Serial.begin (115200);
 } 

// Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden
// Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt.
// Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen
// Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.
// Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben.
// Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.

void loop()
{ 
   // Auslesen des aktuellen Statuses  
   Pin_clk_Aktuell = digitalRead(pin_clk);
   
   // Überprüfung auf Änderung
   if (Pin_clk_Aktuell != Pin_clk_Letzter)
   { 
		 
		if (digitalRead(pin_dt) != Pin_clk_Aktuell) 
		{  
			// Pin_CLK hat sich zuerst verändert
			Counter ++;
			Richtung = true;
		} 
		 
		else 
		{       // Andernfalls hat sich Pin_DT zuerst verändert
			Richtung = false;
			Counter--;
		}
		Serial.println ("Drehung erkannt: ");
		Serial.print ("Drehrichtung: ");
		
		if (Richtung)
		{
		   Serial.println ("Im Uhrzeigersinn");
		}
		else
		{
		   Serial.println("Gegen den Uhrzeigersinn");
		}
		
		Serial.print("Aktuelle Position: ");
		Serial.println(Counter);
		Serial.println("------------------------------");
		 
   } 
   
   // Vorbereitung für den nächsten Druchlauf:
   // Der Wert des aktuellen Durchlaufs ist beim nächsten Druchlauf der vorherige Wert
   Pin_clk_Letzter = Pin_clk_Aktuell;
   
   // Reset-Funktion um aktuelle Position zu speichern
   if (!digitalRead(button_pin) && Counter!=0)
     {
       Counter = 0;
       Serial.println("Position resettet");
     }
     
 } 

Anschlussbelegung Arduino:

CLK = [Pin 3]
DT = [Pin 4]
Button = [Pin 5]
+ = [Pin 5V]
GND = [Pin GND]

Beispielprogramm Download

KY-040_RotaryEncoder.zip

Codebeispiel Raspberry Pi

Das Programm überprüft, falls eine Änderung der Pin-Zustände sich ereignet hat, welcher der beiden Pins sich zuerst geändert hatte, was auf die Drehrichtung schließen lässt.Diese Information erhält man, in dem man einen der beiden Pin-Werte aus einem vorherigen Durchlauf mit dem Wert des aktuellen Durchlaufs vergleicht.

Nachdem die Richtung festgestellt wurde, werden die Schritte von der Startposition an gezählt und ausgegeben. Ein Drücken auf den Knopf des Drehgebers resettet die aktuelle Position.

# coding=utf-8
# Benoetigte Module werden importiert und eingerichtet
import RPi.GPIO as GPIO
import time

GPIO.setmode(GPIO.BCM)

# Hier werden die Eingangs-Pins deklariert, an dem der Sensor angeschlossen ist.
PIN_CLK = 16
PIN_DT = 15
BUTTON_PIN = 14

GPIO.setup(PIN_CLK, GPIO.IN, pull_up_down = GPIO.PUD_UP)
GPIO.setup(PIN_DT, GPIO.IN, pull_up_down = GPIO.PUD_UP)
GPIO.setup(BUTTON_PIN, GPIO.IN, pull_up_down = GPIO.PUD_UP)

# Benötigte Variablen werden initialisiert
Counter = 0
Richtung = True
PIN_CLK_LETZTER = 0
PIN_CLK_AKTUELL = 0
delayTime = 0.01

# Initiales Auslesen des Pin_CLK
PIN_CLK_LETZTER = GPIO.input(PIN_CLK)

# Diese AusgabeFunktion wird bei Signaldetektion ausgefuehrt
def ausgabeFunktion(null):
    global Counter

    PIN_CLK_AKTUELL = GPIO.input(PIN_CLK)

    if PIN_CLK_AKTUELL != PIN_CLK_LETZTER:

        if GPIO.input(PIN_DT) != PIN_CLK_AKTUELL:
            Counter += 1
            Richtung = True;
        else:
            Richtung = False
            Counter = Counter - 1

        print "Drehung erkannt: "

        if Richtung:
            print "Drehrichtung: Im Uhrzeigersinn"
        else:
            print "Drehrichtung: Gegen den Uhrzeigersinn"

        print "Aktuelle Position: ", Counter
        print "------------------------------"

def CounterReset(null):
    global Counter

    print "Position resettet!"
    print "------------------------------"
    Counter = 0

# Um einen Debounce direkt zu integrieren, werden die Funktionen zur Ausgabe mittels
# CallBack-Option vom GPIO Python Modul initialisiert
GPIO.add_event_detect(PIN_CLK, GPIO.BOTH, callback=ausgabeFunktion, bouncetime=50)
GPIO.add_event_detect(BUTTON_PIN, GPIO.FALLING, callback=CounterReset, bouncetime=50)


print "Sensor-Test [druecken Sie STRG+C, um den Test zu beenden]"

# Hauptprogrammschleife
try:
        while True:
            time.sleep(delayTime)

# Aufraeumarbeiten nachdem das Programm beendet wurde
except KeyboardInterrupt:
        GPIO.cleanup()


Anschlussbelegung Raspberry Pi:

CLK = GPIO16 [Pin 36]
DT = GPIO15 [Pin 10]
SW = GPIO14 [Pin 8]
+ = 3,3V [Pin 1]
GND = Masse [Pin 6]

Beispielprogramm Download

KY-040_RPi_RotaryEncoder.zip

Zu starten mit dem Befehl:

sudo python KY-040_RPi_RotaryEncoder.py