| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- ##############################################
- # $Id: 20_FRM_SERVO.pm 5927 2014-05-21 21:56:37Z ntruchsess $
- ##############################################
- package main;
- use strict;
- use warnings;
- #add FHEM/lib to @INC if it's not allready included. Should rather be in fhem.pl than here though...
- BEGIN {
- if (!grep(/FHEM\/lib$/,@INC)) {
- foreach my $inc (grep(/FHEM$/,@INC)) {
- push @INC,$inc."/lib";
- };
- };
- };
- use Device::Firmata::Constants qw/ :all /;
- #####################################
- my %sets = (
- "angle" => "",
- );
- sub
- FRM_SERVO_Initialize($)
- {
- my ($hash) = @_;
- $hash->{SetFn} = "FRM_SERVO_Set";
- $hash->{DefFn} = "FRM_Client_Define";
- $hash->{InitFn} = "FRM_SERVO_Init";
- $hash->{UndefFn} = "FRM_Client_Undef";
- $hash->{AttrFn} = "FRM_SERVO_Attr";
-
- $hash->{AttrList} = "min-pulse max-pulse IODev $main::readingFnAttributes";
- main::LoadModule("FRM");
- }
- sub
- FRM_SERVO_Init($$)
- {
- my ($hash,$args) = @_;
- my $ret = FRM_Init_Pin_Client($hash,$args,PIN_SERVO);
- return $ret if (defined $ret);
- eval {
- my $firmata = FRM_Client_FirmataDevice($hash);
- $hash->{resolution}=$firmata->{metadata}{servo_resolutions}{$hash->{PIN}} if (defined $firmata->{metadata}{servo_resolutions});
- FRM_SERVO_apply_attribute($hash,"max-pulse"); #sets min-pulse as well
- };
- return FRM_Catch($@) if $@;
- main::readingsSingleUpdate($hash,"state","Initialized",1);
- return undef;
- }
- sub
- FRM_SERVO_Attr($$$$) {
- my ($command,$name,$attribute,$value) = @_;
- my $hash = $main::defs{$name};
- eval {
- if ($command eq "set") {
- ARGUMENT_HANDLER: {
- $attribute eq "IODev" and do {
- if ($main::init_done and (!defined ($hash->{IODev}) or $hash->{IODev}->{NAME} ne $value)) {
- FRM_Client_AssignIOPort($hash,$value);
- FRM_Init_Client($hash) if (defined ($hash->{IODev}));
- }
- last;
- };
- ($attribute eq "min-pulse" || $attribute eq "max-pulse") and do {
- if ($main::init_done) {
- $main::attr{$name}{$attribute}=$value;
- FRM_SERVO_apply_attribute($hash,$attribute);
- }
- last;
- };
- }
- }
- };
- my $ret = FRM_Catch($@) if $@;
- if ($ret) {
- $hash->{STATE} = "error setting $attribute to $value: ".$ret;
- return "cannot $command attribute $attribute to $value for $name: ".$ret;
- }
- return undef;
- }
- sub FRM_SERVO_apply_attribute {
- my ($hash,$attribute) = @_;
- if ( $attribute eq "min-pulse" || $attribute eq "max-pulse" ) {
- my $name = $hash->{NAME};
- # defaults are taken from: http://arduino.cc/en/Reference/ServoAttach
- FRM_Client_FirmataDevice($hash)->servo_config($hash->{PIN},{min_pulse => main::AttrVal($name,"min-pulse",544), max_pulse => main::AttrVal($name,"max-pulse",2400)});
- }
- }
- sub
- FRM_SERVO_Set($@)
- {
- my ($hash, @a) = @_;
- return "Need at least one parameters" if(@a < 2);
- return "Unknown argument $a[1], choose one of " . join(" ", sort keys %sets)
- if(!defined($sets{$a[1]}));
- my $command = $a[1];
- my $value = $a[2];
- eval {
- FRM_Client_FirmataDevice($hash)->servo_write($hash->{PIN},$value);
- main::readingsSingleUpdate($hash,"state",$value, 1);
- };
- return $@;
- }
- 1;
- =pod
- =begin html
- <a name="FRM_SERVO"></a>
- <h3>FRM_SERVO</h3>
- <ul>
- represents a pin of an <a href="http://www.arduino.cc">Arduino</a> running <a href="http://www.firmata.org">Firmata</a>
- configured to drive a pwm-controlled servo-motor.<br>
- The value set will be drive the shaft of the servo to the specified angle. see <a href="http://arduino.cc/en/Reference/ServoWrite">Servo.write</a> for values and range<br>
- Requires a defined <a href="#FRM">FRM</a>-device to work.<br><br>
-
- <a name="FRM_SERVOdefine"></a>
- <b>Define</b>
- <ul>
- <code>define <name> FRM_SERVO <pin></code> <br>
- Defines the FRM_SERVO device. <pin>> is the arduino-pin to use.
- </ul>
-
- <br>
- <a name="FRM_SERVOset"></a>
- <b>Set</b><br>
- <ul>
- <code>set <name> angle <value></code><br>sets the angle of the servo-motors shaft to the value specified (in degrees).<br>
- </ul>
- <a name="FRM_SERVOget"></a>
- <b>Get</b><br>
- <ul>
- N/A
- </ul><br>
- <a name="FRM_SERVOattr"></a>
- <b>Attributes</b><br>
- <ul>
- <li><a href="#IODev">IODev</a><br>
- Specify which <a href="#FRM">FRM</a> to use. (Optional, only required if there is more
- than one FRM-device defined.)
- </li>
- <li>min-pulse<br>
- sets the minimum puls-width to use. Defaults to 544. For most servos this translates into a rotation of 180° counterclockwise.</li>
- <li>max-pulse<br>
- sets the maximum puls-width to use. Defaults to 2400. For most servos this translates into a rotation of 180° clockwise</li>
- <li><a href="#eventMap">eventMap</a><br></li>
- <li><a href="#readingFnAttributes">readingFnAttributes</a><br></li>
- </ul>
- </ul>
- <br>
- =end html
- =cut
|