############################################################################## # # 70_ONKYO_AVR.pm # # This file is part of Fhem. # # Fhem is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 2 of the License, or # (at your option) any later version. # # Fhem is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Fhem. If not, see . # ############################################################################## # # ONKYO_AVR (c) Martin Gutenbrunner / https://github.com/delmar43/FHEM # original credits to Loredo # # This module enables FHEM to interact with Onkyo and newer Pioneer audio devices. # # Discussed in FHEM Forum: https://forum.fhem.de/index.php/topic,15024.0.html # ############################################################################### # $Id$ package main; use strict; use warnings; use Data::Dumper; use Symbol qw; use File::Path; use File::stat; use File::Temp; use File::Copy; # initialize ################################################################## sub ONKYO_AVR_Initialize($) { my ($hash) = @_; Log3 $hash, 5, "ONKYO_AVR_Initialize: Entering"; require "$attr{global}{modpath}/FHEM/DevIo.pm"; require "$attr{global}{modpath}/FHEM/ONKYOdb.pm"; $hash->{DefFn} = "ONKYO_AVR_Define"; $hash->{UndefFn} = "ONKYO_AVR_Undefine"; $hash->{SetFn} = "ONKYO_AVR_Set"; $hash->{GetFn} = "ONKYO_AVR_Get"; $hash->{ReadFn} = "ONKYO_AVR_Read"; $hash->{WriteFn} = "ONKYO_AVR_Write"; $hash->{ReadyFn} = "ONKYO_AVR_Ready"; $hash->{NotifyFn} = "ONKYO_AVR_Notify"; $hash->{ShutdownFn} = "ONKYO_AVR_Shutdown"; $hash->{parseParams} = 1; no warnings 'qw'; my @attrList = qw( do_not_notify:1,0 disabledForIntervals volumeSteps:1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20 volumeMax:slider,0,1,100 inputs disable:0,1 wakeupCmd:textField connectionCheck:off,30,45,60,75,90,105,120 timeout:1,2,3,4,5 ); use warnings 'qw'; $hash->{AttrList} = join( " ", @attrList ) . " " . $readingFnAttributes; $data{RC_layout}{ONKYO_AVR_SVG} = "ONKYO_AVR_RClayout_SVG"; $data{RC_layout}{ONKYO_AVR} = "ONKYO_AVR_RClayout"; $data{RC_makenotify}{ONKYO_AVR} = "ONKYO_AVR_RCmakenotify"; # 98_powerMap.pm support $hash->{powerMap} = { model => { 'TX-NR626' => { rname_E => 'energy', rname_P => 'consumption', map => { stateAV => { absent => 0, off => 0, muted => 85, '*' => 140, }, }, }, }, }; } # regular Fn ################################################################## sub ONKYO_AVR_Define($$$) { my ( $hash, $a, $h ) = @_; my $name = $hash->{NAME}; my $infix = "ONKYO_AVR"; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_Define()"; delete $attr{$name}{model}; eval { require XML::Simple; }; return "Please install Perl XML::Simple to use module ONKYO_AVR" if ($@); if ( int(@$a) < 3 ) { my $msg = "Wrong syntax: define ONKYO_AVR { | } []"; Log3 $name, 4, $msg; return $msg; } RemoveInternalTimer($hash); DevIo_CloseDev($hash); delete $hash->{NEXT_OPEN} if ( defined( $hash->{NEXT_OPEN} ) ); $hash->{Clients} = ":ONKYO_AVR_ZONE:"; $hash->{TIMEOUT} = AttrVal( $name, "timeout", "3" ); # used zone to control $hash->{ZONE} = "1"; $hash->{INPUT} = ""; $hash->{SCREENLAYER} = "0"; # protocol version $hash->{PROTOCOLVERSION} = @$a[3] || 2013; if ( !( $hash->{PROTOCOLVERSION} =~ /^(2013|pre2013)$/ ) ) { return "Invalid protocol, choose one of 2013 pre2013"; } if ( $hash->{PROTOCOLVERSION} eq "pre2013" ) { $hash->{model} = 'pre2013'; } # set default settings on first define if ( $init_done && !defined( $hash->{OLDDEF} ) ) { fhem 'attr ' . $name . ' stateFormat stateAV'; fhem 'attr ' . $name . ' cmdIcon muteT:rc_MUTE previous:rc_PREVIOUS next:rc_NEXT play:rc_PLAY pause:rc_PAUSE stop:rc_STOP shuffleT:rc_SHUFFLE repeatT:rc_REPEAT'; fhem 'attr ' . $name . ' webCmd volume:muteT:input:previous:next'; fhem 'attr ' . $name . ' devStateIcon on:rc_GREEN@green:off off:rc_STOP:on absent:rc_RED playing:rc_PLAY@green:pause paused:rc_PAUSE@green:play muted:rc_MUTE@green:muteT fast-rewind:rc_REW@green:play fast-forward:rc_FF@green:play interrupted:rc_PAUSE@yellow:play'; } $hash->{helper}{receiver}{device}{zonelist}{zone}{1}{name} = "Main"; $hash->{helper}{receiver}{device}{zonelist}{zone}{1}{value} = "1"; $modules{ONKYO_AVR_ZONE}{defptr}{$name}{1} = $hash; $hash->{DeviceName} = @$a[2]; if ( ONKYO_AVR_addExtension( $name, "ONKYO_AVR_CGI", $infix ) ) { $hash->{fhem}{infix} = $infix; } # connect using serial connection (old blocking style) if ( $hash->{DeviceName} =~ m/^UNIX:(SEQPACKET|STREAM):(.*)$/ || $hash->{DeviceName} =~ m/^FHEM:DEVIO:(.*)(:(.*))/ ) { my $ret = DevIo_OpenDev( $hash, 0, "ONKYO_AVR_DevInit" ); return $ret; } # connect using TCP connection (non-blocking style) else { # add missing port if required $hash->{DeviceName} = $hash->{DeviceName} . ":60128" if ( $hash->{DeviceName} !~ m/^(.+):([0-9]+)$/ ); DevIo_OpenDev( $hash, 0, "ONKYO_AVR_DevInit", sub() { my ( $hash, $err ) = @_; Log3 $name, 4, "ONKYO_AVR $name: $err" if ($err); } ); } return undef; } sub ONKYO_AVR_Undefine($$) { my ( $hash, $name ) = @_; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_Undefine()"; if ( defined( $hash->{fhem}{infix} ) ) { ONKYO_AVR_removeExtension( $hash->{fhem}{infix} ); } RemoveInternalTimer($hash); foreach my $d ( sort keys %defs ) { if ( defined( $defs{$d} ) && defined( $defs{$d}{IODev} ) && $defs{$d}{IODev} == $hash ) { my $lev = ( $reread_active ? 4 : 2 ); Log3 $name, $lev, "deleting port for $d"; delete $defs{$d}{IODev}; } } DevIo_CloseDev($hash); return undef; } sub ONKYO_AVR_Set($$$) { my ( $hash, $a, $h ) = @_; my $name = $hash->{NAME}; my $zone = $hash->{ZONE}; my $state = ReadingsVal( $name, "power", "off" ); my $presence = ReadingsVal( $name, "presence", "absent" ); my $return; my $reading; my $inputs_txt = ""; my $channels_txt = ""; my @implicit_cmds; my $implicit_txt = ""; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_Set()"; return "Argument is missing" if ( int(@$a) < 1 ); # Input alias handling if ( defined( $attr{$name}{inputs} ) && $attr{$name}{inputs} ne "" ) { my @inputs = split( ':', $attr{$name}{inputs} ); if (@inputs) { foreach (@inputs) { if (m/[^,\s]+(,[^,\s]+)+/) { my @input_names = split( ',', $_ ); $inputs_txt .= $input_names[1] . ","; $input_names[1] =~ s/\s/_/g; $hash->{helper}{receiver}{input_aliases}{ $input_names[0] } = $input_names[1]; $hash->{helper}{receiver}{input_names}{ $input_names[1] } = $input_names[0]; } else { $inputs_txt .= $_ . ","; } } } $inputs_txt =~ s/\s/_/g; $inputs_txt = substr( $inputs_txt, 0, -1 ); } # if we could read the actual available inputs from the receiver, use them elsif (defined( $hash->{helper}{receiver} ) && ref( $hash->{helper}{receiver} ) eq "HASH" && defined( $hash->{helper}{receiver}{device}{selectorlist}{count} ) && $hash->{helper}{receiver}{device}{selectorlist}{count} > 0 ) { foreach my $input ( @{ $hash->{helper}{receiver}{device}{selectorlist}{selector} } ) { if ( $input->{value} eq "1" && $input->{zone} ne "00" && $input->{id} ne "80" ) { my $id = $input->{id}; my $name = trim( $input->{name} ); $inputs_txt .= $name . ","; } } $inputs_txt =~ s/\s/_/g; $inputs_txt = substr( $inputs_txt, 0, -1 ); } # use general list of possible inputs else { # Find out valid inputs my $inputs = ONKYOdb::ONKYO_GetRemotecontrolValue( $zone, ONKYOdb::ONKYO_GetRemotecontrolCommand( $zone, "input" ) ); foreach my $input ( sort keys %{$inputs} ) { $inputs_txt .= $input . "," if ( !( $input =~ /^(07|08|09|up|down|query)$/ ) ); } $inputs_txt = substr( $inputs_txt, 0, -1 ); } # list of network channels/services my $channels_src = "internal"; if ( defined( $hash->{helper}{receiver} ) && ref( $hash->{helper}{receiver} ) eq "HASH" && defined( $hash->{helper}{receiver}{device}{netservicelist}{count} ) && $hash->{helper}{receiver}{device}{netservicelist}{count} > 0 ) { foreach my $id ( sort keys %{ $hash->{helper}{receiver}{device}{netservicelist}{netservice} } ) { if ( defined( $hash->{helper}{receiver}{device}{netservicelist} {netservice}{$id}{value} ) && $hash->{helper}{receiver}{device}{netservicelist} {netservice}{$id}{value} eq "1" ) { $channels_txt .= trim( $hash->{helper}{receiver}{device}{netservicelist} {netservice}{$id}{name} ) . ","; } } $channels_txt =~ s/\s/_/g; $channels_txt = substr( $channels_txt, 0, -1 ); $channels_src = "receiver"; } # use general list of possible channels else { # Find out valid channels my $channels = ONKYOdb::ONKYO_GetRemotecontrolValue( "1", ONKYOdb::ONKYO_GetRemotecontrolCommand( "1", "net-service" ) ); foreach my $channel ( sort keys %{$channels} ) { $channels_txt .= $channel . "," if ( !( $channel =~ /^(up|down|query)$/ ) ); } $channels_txt = substr( $channels_txt, 0, -1 ); } # for each reading, check if there is a known command for it # and allow to set values if there are any available if ( defined( $hash->{READINGS} ) ) { foreach my $reading ( keys %{ $hash->{READINGS} } ) { my $cmd_raw = ONKYOdb::ONKYO_GetRemotecontrolCommand( $zone, $reading ); my @readingExceptions = ( "volume", "input", "mute", "sleep", "center-temporary-level", "subwoofer-temporary-level", "balance", "preset", ); if ( $cmd_raw && !( grep $_ eq $reading, @readingExceptions ) ) { my $cmd_details = ONKYOdb::ONKYO_GetRemotecontrolCommandDetails( $zone, $cmd_raw ); my $value_list = ""; my $debuglist; foreach my $value ( keys %{ $cmd_details->{values} } ) { next if ( $value eq "QSTN" ); if ( defined( $cmd_details->{values}{$value}{name} ) ) { $value_list .= "," if ( $value_list ne "" ); $value_list .= $cmd_details->{values}{$value}{name} if ( ref( $cmd_details->{values}{$value}{name} ) eq "" ); $value_list .= $cmd_details->{values}{$value}{name}[0] if ( ref( $cmd_details->{values}{$value}{name} ) eq "ARRAY" ); } } if ( $value_list ne "" ) { push @implicit_cmds, $reading; $implicit_txt .= " $reading:$value_list"; } } # tone-* elsif ( $reading =~ /^tone.*-([a-zA-Z]+)$/ ) { $implicit_txt .= " $reading:slider,-10,1,10"; } # center-temporary-level elsif ( $reading eq "center-temporary-level" ) { $implicit_txt .= " $reading:slider,-12,1,12"; } # subwoofer*-temporary-level elsif ( $reading =~ /^subwoofer.*-temporary-level$/ ) { $implicit_txt .= " $reading:slider,-15,1,12"; } } } my $preset_txt = ""; if ( defined( $hash->{helper}{receiver}{preset} ) ) { foreach my $id ( sort keys %{ $hash->{helper}{receiver}{preset} } ) { my $presetName = $hash->{helper}{receiver}{preset}{$id}; next if ( !$presetName || $presetName eq "" ); $preset_txt = "preset:" if ( $preset_txt eq "" ); $preset_txt .= "," if ( $preset_txt eq "preset:" && ReadingsVal( $name, "preset", "-" ) eq "" ); $presetName =~ s/\s/_/g; $preset_txt .= $presetName . ","; } } $preset_txt = substr( $preset_txt, 0, -1 ) if ( $preset_txt ne "" ); if ( $preset_txt eq "" ) { $preset_txt = "preset:"; $preset_txt .= "," if ( ReadingsVal( $name, "preset", "-" ) eq "" ); $preset_txt .= "1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40"; } my $shuffle_txt = "shuffle:"; $shuffle_txt .= "," if ( ReadingsVal( $name, "shuffle", "-" ) eq "-" ); $shuffle_txt .= "off,on,on-album,on-folder"; my $repeat_txt = "repeat:"; $repeat_txt .= "," if ( ReadingsVal( $name, "repeat", "-" ) eq "-" ); $repeat_txt .= "off,all,all-folder,one"; my $usage = "Unknown argument '" . @$a[1] . "', choose one of rawCommand toggle:noArg on:noArg off:noArg volume:slider,0,1,100 volumeDown:noArg volumeUp:noArg mute:off,on muteT:noArg play:noArg pause:noArg stop:noArg previous:noArg next:noArg shuffleT:noArg repeatT:noArg remoteControl:play,pause,repeat,stop,top,down,up,right,delete,display,ff,left,mode,return,rew,select,setup,0,1,2,3,4,5,6,7,8,9,prev,next,shuffle,menu channelDown:noArg channelUp:noArg inputDown:noArg inputUp:noArg internet-radio-preset:1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40 input:" . $inputs_txt; $usage .= " channel:$channels_txt"; $usage .= " presetDown:noArg presetUp:noArg $preset_txt"; $usage .= " $shuffle_txt"; $usage .= " $repeat_txt"; $usage .= $implicit_txt if ( $implicit_txt ne "" ); $usage .= " sleep:off,5,10,15,30,60,90"; if ( ReadingsVal( $name, "currentTrackPosition", "--:--" ) ne "--:--" ) { $usage .= " currentTrackPosition"; } my $cmd = ''; return "Device is offline and cannot be controlled at that stage." if ( $presence eq "absent" && lc( @$a[1] ) ne "on" && lc( @$a[1] ) ne "?" && lc( @$a[1] ) ne "help" ); readingsBeginUpdate($hash); # create inputList reading for frontends readingsBulkUpdate( $hash, "inputList", $inputs_txt ) if ( ReadingsVal( $name, "inputList", "-" ) ne $inputs_txt ); # create channelList reading for frontends readingsBulkUpdate( $hash, "channelList", $channels_txt ) if ( ( $channels_src eq "internal" && ReadingsVal( $name, "channelList", "-" ) eq "-" ) || ( $channels_src eq "receiver" && ReadingsVal( $name, "channelList", "-" ) ne $channels_txt ) ); if ( lc( @$a[1] ) eq "rawcommand" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2] . " " . @$a[3] if ( !@$a[4] || @$a[4] ne "quiet" ); $return = ONKYO_AVR_SendRawCommand( $hash, @$a[2] . @$a[3] ); } # channel elsif ( lc( @$a[1] ) eq "channel" ) { if ( !defined( @$a[2] ) ) { $return = "Syntax: CHANNELNAME [USERNAME PASSWORD]"; } else { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name channel " . @$a[2]; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "2B" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "2B" ); my $ret = fhem "sleep 1;set $name channel " . @$a[2]; $return .= $ret if ($ret); } elsif ( ReadingsVal( $name, "channel", "" ) ne @$a[2] || ( defined( @$a[3] ) && defined( @$a[4] ) ) ) { my $servicename = ""; my $channelname = @$a[2]; if ( defined( $hash->{helper}{receiver} ) && ref( $hash->{helper}{receiver} ) eq "HASH" && defined( $hash->{helper}{receiver}{device}{netservicelist}{count} ) && $hash->{helper}{receiver}{device}{netservicelist}{count} > 0 ) { $channelname =~ s/_/ /g; foreach my $id ( sort keys %{ $hash->{helper}{receiver}{device}{netservicelist} {netservice} } ) { if ( defined( $hash->{helper}{receiver}{device} {netservicelist}{netservice}{$id}{value} ) && $hash->{helper}{receiver}{device} {netservicelist}{netservice}{$id}{value} eq "1" && $hash->{helper}{receiver}{device} {netservicelist}{netservice}{$id}{name} eq $channelname ) { $servicename .= uc($id); last; } } } else { my $channels = ONKYOdb::ONKYO_GetRemotecontrolValue( "1", ONKYOdb::ONKYO_GetRemotecontrolCommand( "1", "net-service" ) ); $servicename = $channels->{$channelname} if ( defined( $channels->{$channelname} ) ); } Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $servicename = uc($channelname) if ( $servicename eq "" ); $servicename .= "0" if ( !defined( @$a[3] ) ); $servicename .= "1" . @$a[3] if ( defined( @$a[3] ) ); $servicename .= @$a[4] if ( defined( @$a[4] ) ); $return = ONKYO_AVR_SendCommand( $hash, "net-service", $servicename ); } } } # channelDown elsif ( lc( @$a[1] ) eq "channeldown" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name channelDown"; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "2B" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "2B" ); my $ret = fhem "sleep 1;set $name channelDown"; $return .= $ret if ($ret); } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "chdn" ); } } # channelUp elsif ( lc( @$a[1] ) eq "channelup" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name channelUp"; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "2B" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "2B" ); my $ret = fhem "sleep 1;set $name channelUp"; $return .= $ret if ($ret); } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "chup" ); } } # currentTrackPosition elsif ( lc( @$a[1] ) eq "currenttrackposition" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { if ( @$a[2] !~ /^[0-9][0-9]:[0-5][0-9]$/ ) { $return = "Time needs to have format mm:ss and between 00:00 and 99:59"; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb-time-seek", @$a[2] ); } } } # internet-radio-preset elsif ( lc( @$a[1] ) eq "internet-radio-preset" ) { if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name " . @$a[1] . " " . @$a[2]; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "2B" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "2B" ); my $ret = fhem "sleep 5;set $name " . @$a[1] . " " . @$a[2]; $return .= $ret if ($ret); } elsif ( @$a[2] =~ /^\d*$/ ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), ONKYO_AVR_dec2hex( @$a[2] ) ); } else { $return = "Invalid argument format"; } } } # preset elsif ( lc( @$a[1] ) eq "preset" ) { if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name preset " . @$a[2]; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "24" && $hash->{INPUT} ne "25" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "24" ); my $ret = fhem "sleep 1;set $name preset " . @$a[2]; $return .= $ret if ($ret); } elsif ( lc( @$a[2] ) eq "up" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), "UP" ); } elsif ( lc( @$a[2] ) eq "down" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), "DOWN" ); } elsif ( @$a[2] =~ /^\d*$/ ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), ONKYO_AVR_dec2hex( @$a[2] ) ); } elsif ( defined( $hash->{helper}{receiver}{preset} ) ) { foreach my $id ( sort keys %{ $hash->{helper}{receiver}{preset} } ) { my $presetName = $hash->{helper}{receiver}{preset}{$id}; next if ( !$presetName || $presetName eq "" ); $presetName =~ s/\s/_/g; if ( $presetName eq @$a[2] ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), uc($id) ); last; } } } } } # presetDown elsif ( lc( @$a[1] ) eq "presetdown" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name presetDown"; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "24" && $hash->{INPUT} ne "25" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "24" ); my $ret = fhem "sleep 1;set $name presetDown"; $return .= $ret if ($ret); } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "preset", "down" ); } } # presetUp elsif ( lc( @$a[1] ) eq "presetup" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); my $ret = fhem "sleep 5;set $name presetUp"; $return .= $ret if ($ret); } elsif ( $hash->{INPUT} ne "24" && $hash->{INPUT} ne "25" ) { $return = ONKYO_AVR_SendCommand( $hash, "input", "24" ); my $ret = fhem "sleep 1;set $name presetUp"; $return .= $ret if ($ret); } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "preset", "up" ); } } # tone-* elsif ( lc( @$a[1] ) =~ /^(tone.*)-(bass|treble)$/ ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { if ( $state eq "off" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } elsif ( lc( @$a[2] ) eq "up" ) { my $setVal = ""; $setVal = "B" if ( $2 eq "bass" ); $setVal = "T" if ( $2 eq "treble" ); $return = ONKYO_AVR_SendCommand( $hash, lc($1), $setVal . "UP" ); } elsif ( lc( @$a[2] ) eq "down" ) { my $setVal = ""; $setVal = "B" if ( $2 eq "bass" ); $setVal = "T" if ( $2 eq "treble" ); $return = ONKYO_AVR_SendCommand( $hash, lc($1), $setVal . "DOWN" ); } elsif ( @$a[2] =~ /^-*\d+$/ ) { my $setVal = ""; $setVal = "B" if ( $2 eq "bass" ); $setVal = "T" if ( $2 eq "treble" ); $setVal .= "+" if ( @$a[2] > 0 ); $setVal .= "-" if ( @$a[2] < 0 ); my $setVal2 = @$a[2]; $setVal2 = substr( $setVal2, 1 ) if ( $setVal2 < 0 ); $setVal2 = ONKYO_AVR_dec2hex($setVal2); $setVal2 = substr( $setVal2, 1 ) if ( $setVal2 ne "00" ); $return = ONKYO_AVR_SendCommand( $hash, lc($1), $setVal . $setVal2 ); } } } # center-temporary-level # subwoofer-temporary-level elsif (lc( @$a[1] ) eq "center-temporary-level" || lc( @$a[1] ) eq "subwoofer-temporary-level" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { if ( $state eq "off" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } elsif ( lc( @$a[2] ) eq "up" ) { $return = ONKYO_AVR_SendCommand( $hash, lc($1), "UP" ); } elsif ( lc( @$a[2] ) eq "down" ) { $return = ONKYO_AVR_SendCommand( $hash, lc($1), "DOWN" ); } elsif ( @$a[2] =~ /^-*\d+$/ ) { my $setVal = ""; $setVal = "+" if ( @$a[2] > 0 ); $setVal = "-" if ( @$a[2] < 0 ); my $setVal2 = @$a[2]; $setVal2 = substr( $setVal2, 1 ) if ( $setVal2 < 0 ); $setVal2 = ONKYO_AVR_dec2hex($setVal2); $setVal2 = substr( $setVal2, 1 ) if ( $setVal2 ne "00" ); $return = ONKYO_AVR_SendCommand( $hash, lc( @$a[1] ), $setVal . $setVal2 ); } } } # toggle elsif ( lc( @$a[1] ) eq "toggle" ) { if ( $state eq "off" ) { $return = fhem "set $name on"; } else { $return = fhem "set $name off"; } } # on elsif ( lc( @$a[1] ) eq "on" ) { if ( $presence eq "absent" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " (wakeup)"; my $wakeupCmd = AttrVal( $name, "wakeupCmd", "" ); if ( $wakeupCmd ne "" ) { $wakeupCmd =~ s/\$DEVICE/$name/g; if ( $wakeupCmd =~ s/^[ \t]*\{|\}[ \t]*$//g ) { Log3 $name, 4, "ONKYO_AVR executing wake-up command (Perl): $wakeupCmd"; $return = eval $wakeupCmd; } else { Log3 $name, 4, "ONKYO_AVR executing wake-up command (fhem): $wakeupCmd"; $return = fhem $wakeupCmd; } } else { $return = "Device is offline and cannot be controlled at that stage."; $return .= "\nYou may enable network-standby to allow a permanent connection to the device by the following command:\nget $name remoteControl network-standby on" if ( ReadingsVal( $name, "network-standby", "off" ) ne "on" ); } } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); # don't wait for receiver to confirm power on # readingsBeginUpdate($hash); # power readingsBulkUpdate( $hash, "power", "on" ) if ( ReadingsVal( $name, "power", "-" ) ne "on" ); # stateAV my $stateAV = ONKYO_AVR_GetStateAV($hash); readingsBulkUpdate( $hash, "stateAV", $stateAV ) if ( ReadingsVal( $name, "stateAV", "-" ) ne $stateAV ); readingsEndUpdate( $hash, 1 ); } } # off elsif ( lc( @$a[1] ) eq "off" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "power", "off" ); } # remoteControl elsif ( lc( @$a[1] ) eq "remotecontrol" ) { if ( !defined( @$a[2] ) ) { $return = "No argument given, choose one of minutes off"; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( lc( @$a[2] ) eq "play" || lc( @$a[2] ) eq "pause" || lc( @$a[2] ) eq "repeat" || lc( @$a[2] ) eq "stop" || lc( @$a[2] ) eq "top" || lc( @$a[2] ) eq "down" || lc( @$a[2] ) eq "up" || lc( @$a[2] ) eq "right" || lc( @$a[2] ) eq "delete" || lc( @$a[2] ) eq "display" || lc( @$a[2] ) eq "ff" || lc( @$a[2] ) eq "left" || lc( @$a[2] ) eq "mode" || lc( @$a[2] ) eq "return" || lc( @$a[2] ) eq "rew" || lc( @$a[2] ) eq "select" || lc( @$a[2] ) eq "setup" || lc( @$a[2] ) eq "0" || lc( @$a[2] ) eq "1" || lc( @$a[2] ) eq "2" || lc( @$a[2] ) eq "3" || lc( @$a[2] ) eq "4" || lc( @$a[2] ) eq "5" || lc( @$a[2] ) eq "6" || lc( @$a[2] ) eq "7" || lc( @$a[2] ) eq "8" || lc( @$a[2] ) eq "9" ) { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", lc( @$a[2] ) ); } elsif ( lc( @$a[2] ) eq "prev" ) { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "trdn" ); } elsif ( lc( @$a[2] ) eq "next" ) { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "trup" ); } elsif ( lc( @$a[2] ) eq "shuffle" ) { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "random" ); } elsif ( lc( @$a[2] ) eq "menu" ) { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "men" ); } else { $return = "Unsupported remoteControl command: " . @$a[2]; } } } # play elsif ( lc( @$a[1] ) eq "play" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "play" ); } } # pause elsif ( lc( @$a[1] ) eq "pause" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "pause" ); } } # stop elsif ( lc( @$a[1] ) eq "stop" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "stop" ); } } # shuffle elsif ( lc( @$a[1] ) eq "shuffle" || lc( @$a[1] ) eq "shufflet" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "random" ); } } # repeat elsif ( lc( @$a[1] ) eq "repeat" || lc( @$a[1] ) eq "repeatt" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "repeat" ); } } # previous elsif ( lc( @$a[1] ) eq "previous" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "trdn" ); } } # next elsif ( lc( @$a[1] ) eq "next" ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; if ( $state ne "on" ) { $return = "Device power is turned off, this function is unavailable at that stage."; } else { $return = ONKYO_AVR_SendCommand( $hash, "net-usb", "trup" ); } } # sleep elsif ( lc( @$a[1] ) eq "sleep" ) { if ( !defined( @$a[2] ) ) { $return = "No argument given, choose one of minutes off"; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( @$a[2] eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "sleep", "off" ); } elsif ( @$a[2] =~ m/^\d+$/ && @$a[2] > 0 && @$a[2] <= 90 ) { $return = ONKYO_AVR_SendCommand( $hash, "sleep", ONKYO_AVR_dec2hex( @$a[2] ) ); } else { $return = "Argument does not seem to be a valid integer between 0 and 90"; } } } # mute elsif ( lc( @$a[1] ) eq "mute" || lc( @$a[1] ) eq "mutet" ) { if ( defined( @$a[2] ) ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; } if ( $state eq "on" ) { if ( !defined( @$a[2] ) || @$a[2] eq "toggle" ) { $return = ONKYO_AVR_SendCommand( $hash, "mute", "toggle" ); } elsif ( lc( @$a[2] ) eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "mute", "off" ); } elsif ( lc( @$a[2] ) eq "on" ) { $return = ONKYO_AVR_SendCommand( $hash, "mute", "on" ); } else { $return = "Argument does not seem to be one of on off toogle"; } } else { $return = "Device needs to be ON to mute/unmute audio."; } } # volume elsif ( lc( @$a[1] ) eq "volume" ) { if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { my $volm = AttrVal( $name, "volumeMax", 0 ); @$a[2] = $volm if ( $volm && @$a[2] > $volm ); Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( $state eq "on" ) { if ( @$a[2] =~ m/^\d+$/ && @$a[2] >= 0 && @$a[2] <= 100 ) { $return = ONKYO_AVR_SendCommand( $hash, "volume", ONKYO_AVR_dec2hex( @$a[2] ) ); } else { $return = "Argument does not seem to be a valid integer between 0 and 100"; } } else { $return = "Device needs to be ON to adjust volume."; } } } # volumeUp/volumeDown elsif ( lc( @$a[1] ) =~ /^(volumeup|volumedown)$/ ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; my $volumeSteps = AttrVal( $name, "volumeSteps", "1" ); my $volume = ReadingsVal( $name, "volume", "0" ); if ( $state eq "on" ) { if ( lc( @$a[1] ) eq "volumeup" ) { if ( $volumeSteps > 1 ) { $return = ONKYO_AVR_SendCommand( $hash, "volume", ONKYO_AVR_dec2hex( $volume + $volumeSteps ) ); } else { $return = ONKYO_AVR_SendCommand( $hash, "volume", "level-up" ); } } else { if ( $volumeSteps > 1 ) { $return = ONKYO_AVR_SendCommand( $hash, "volume", ONKYO_AVR_dec2hex( $volume - $volumeSteps ) ); } else { $return = ONKYO_AVR_SendCommand( $hash, "volume", "level-down" ); } } } else { $return = "Device needs to be ON to adjust volume."; } } # input elsif ( lc( @$a[1] ) eq "input" ) { if ( !defined( @$a[2] ) ) { $return = "No input given"; } else { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); $return .= fhem "sleep 2;set $name input " . @$a[2]; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; $return = ONKYO_AVR_SendCommand( $hash, "input", @$a[2] ); } } } # inputUp elsif ( lc( @$a[1] ) eq "inputup" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); $return .= fhem "sleep 2;set $name inputUp"; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "input", "up" ); } } # inputDown elsif ( lc( @$a[1] ) eq "inputdown" ) { if ( $state eq "off" ) { $return = ONKYO_AVR_SendCommand( $hash, "power", "on" ); $return .= fhem "sleep 2;set $name inputDown"; } else { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1]; $return = ONKYO_AVR_SendCommand( $hash, "input", "down" ); } } # implicit commands through available readings elsif ( grep $_ eq @$a[1], @implicit_cmds ) { Log3 $name, 3, "ONKYO_AVR set $name " . @$a[1] . " " . @$a[2]; if ( !defined( @$a[2] ) ) { $return = "No argument given"; } else { $return = ONKYO_AVR_SendCommand( $hash, @$a[1], @$a[2] ); } } # return usage hint else { $return = $usage; } readingsEndUpdate( $hash, 1 ); # return result return $return; } sub ONKYO_AVR_Get($$$) { my ( $hash, $a, $h ) = @_; my $name = $hash->{NAME}; my $zone = $hash->{ZONE}; my $state = ReadingsVal( $name, "power", "off" ); my $presence = ReadingsVal( $name, "presence", "absent" ); my $commands = ONKYOdb::ONKYO_GetRemotecontrolCommand($zone); my $commands_details = ONKYOdb::ONKYO_GetRemotecontrolCommandDetails($zone); my $return; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_Get()"; return "Argument is missing" if ( int(@$a) < 1 ); # readings return $hash->{READINGS}{ @$a[1] }{VAL} if ( defined( $hash->{READINGS}{ @$a[1] } ) ); return "Device is offline and cannot be controlled at that stage." if ( $presence eq "absent" ); # createZone if ( lc( @$a[1] ) eq "createzone" ) { if ( !defined( @$a[2] ) ) { $return = "Syntax: ZONE ID or NAME"; } else { $return = fhem "define " . $name . "_" . @$a[2] . " ONKYO_AVR_ZONE " . @$a[2]; $return = $name . "_" . @$a[2] . " created" if ( !$return || $return eq "" ); } } # statusRequest elsif ( lc( @$a[1] ) eq "statusrequest" ) { Log3 $name, 3, "ONKYO_AVR get $name " . @$a[1]; ONKYO_AVR_SendCommand( $hash, "power", "query" ); ONKYO_AVR_SendCommand( $hash, "input", "query" ); ONKYO_AVR_SendCommand( $hash, "mute", "query" ); ONKYO_AVR_SendCommand( $hash, "volume", "query" ); ONKYO_AVR_SendCommand( $hash, "sleep", "query" ); ONKYO_AVR_SendCommand( $hash, "audio-information", "query" ); ONKYO_AVR_SendCommand( $hash, "video-information", "query" ); ONKYO_AVR_SendCommand( $hash, "listening-mode", "query" ); ONKYO_AVR_SendCommand( $hash, "video-picture-mode", "query" ); ONKYO_AVR_SendCommand( $hash, "phase-matching-bass", "query" ); ONKYO_AVR_SendCommand( $hash, "center-temporary-level", "query" ); ONKYO_AVR_SendCommand( $hash, "subwoofer-temporary-level", "query" ); fhem "sleep 1 quiet;get $name remoteControl net-receiver-information query quiet"; } # remoteControl elsif ( lc( @$a[1] ) eq "remotecontrol" ) { # Output help for commands if ( !defined( @$a[2] ) || @$a[2] eq "help" || @$a[2] eq "?" ) { my $valid_commands = "Usage: \n\nValid commands in zone$zone:\n\n\n" . "COMMAND\t\t\tDESCRIPTION\n\n"; # For each valid command foreach my $command ( sort keys %{$commands} ) { my $command_raw = $commands->{$command}; # add command including description if found if ( defined( $commands_details->{$command_raw}{description} ) ) { $valid_commands .= $command . "\t\t\t" . $commands_details->{$command_raw}{description} . "\n"; } # add command only else { $valid_commands .= $command . "\n"; } } $valid_commands .= "\nTry '<command> help' to find out well known values.\n\n\n"; $return = $valid_commands; } else { # Reading values for command from HASH table my $values = ONKYOdb::ONKYO_GetRemotecontrolValue( $zone, $commands->{ @$a[2] } ); @$a[3] = "query" if ( !defined( @$a[3] ) && defined( $values->{query} ) ); # Output help for values if ( !defined( @$a[3] ) || @$a[3] eq "help" || @$a[3] eq "?" ) { # Get all details for command my $command_details = ONKYOdb::ONKYO_GetRemotecontrolCommandDetails( $zone, $commands->{ @$a[2] } ); my $valid_values = "Usage: " . @$a[2] . " \n\nWell known values:\n\n\n" . "VALUE\t\t\tDESCRIPTION\n\n"; # For each valid value foreach my $value ( sort keys %{$values} ) { # add value including description if found if ( defined( $command_details->{description} ) ) { $valid_values .= $value . "\t\t\t" . $command_details->{description} . "\n"; } # add value only else { $valid_values .= $value . "\n"; } } $valid_values .= "\n\n\n"; $return = $valid_values; } # normal processing else { Log3 $name, 3, "ONKYO_AVR get $name " . @$a[1] . " " . @$a[2] . " " . @$a[3] if ( !@$a[4] || @$a[4] ne "quiet" ); ONKYO_AVR_SendCommand( $hash, @$a[2], @$a[3] ); $return = "Sent command: " . @$a[2] . " " . @$a[3] if ( !@$a[4] || @$a[4] ne "quiet" ); } } } else { $return = "Unknown argument " . @$a[1] . ", choose one of statusRequest:noArg"; # createZone my $zones = ""; if ( defined( $hash->{helper}{receiver}{device}{zonelist}{zone} ) ) { foreach my $zoneID ( keys %{ $hash->{helper}{receiver}{device}{zonelist}{zone} } ) { next if ( !defined( $hash->{helper}{receiver}{device}{zonelist}{zone} {$zoneID}{value} ) || $hash->{helper}{receiver}{device}{zonelist}{zone} {$zoneID}{value} ne "1" || $zoneID eq "1" ); $zones .= "," if ( $zones ne "" ); $zones .= $zoneID; } } $return .= " createZone:$zones" if ( $zones ne "" ); $return .= " createZone:2,3,4" if ( $zones eq "" ); # remoteControl $return .= " remoteControl:"; foreach my $command ( sort keys %{$commands} ) { $return .= "," . $command; } } return $return; } sub ONKYO_AVR_Read($) { my ($hash) = @_; my $name = $hash->{NAME}; # read from serial device my $buf = DevIo_SimpleRead($hash); return "" if ( !defined($buf) ); $buf = $hash->{PARTIAL} . $buf; # reset connectionCheck timer my $checkInterval = AttrVal( $name, "connectionCheck", "60" ); RemoveInternalTimer($hash); if ( $checkInterval ne "off" ) { my $next = gettimeofday() + $checkInterval; $hash->{helper}{nextConnectionCheck} = $next; InternalTimer( $next, "ONKYO_AVR_connectionCheck", $hash, 0 ); } Log3 $name, 5, "ONKYO_AVR $name: raw " . ONKYO_AVR_hexdump($buf); my $lastchr = substr( $buf, -1, 1 ); if ( $lastchr ne "\n" ) { $hash->{PARTIAL} = $buf; Log3( $hash, 5, "ONKYO_AVR_Read: partial command received" ); return; } else { $hash->{PARTIAL} = ""; } my @cmds = split( 'ISCP', $buf ); for my $el (@cmds) { if ( $el ne '' ) { ONKYO_AVR_Read2( $hash, $name, "ISCP$el" ); } } readingsDelete( $hash, 'model' ); return; } sub ONKYO_AVR_Read2($$$) { my ( $hash, $name, $buf ) = @_; my $state = ReadingsVal( $name, "power", "off" ); my $zone = 0; my $definedZones = scalar keys %{ $modules{ONKYO_AVR_ZONE}{defptr}{$name} }; my $length = length $buf; return unless ( $length >= 16 ); my ( $magic, $header_size, $data_size, $version, $res1, $res2, $res3 ) = unpack 'a4 N N C4', $buf; Log3 $name, 5, "ONKYO_AVR $name: Unexpected magic: expected 'ISCP', got '$magic'" and return unless ( $magic eq 'ISCP' ); Log3 $name, 5, "ONKYO_AVR $name: unusual packet length: $length < $header_size + $data_size" and return unless ( $length >= $header_size + $data_size ); Log3 $name, 5, "ONKYO_AVR $name: Unexpected version: expected '0x01', got '0x%02x' " . $version and return unless ( $version == 0x01 ); Log3 $name, 5, "ONKYO_AVR $name: Unexpected header size: expected '0x10', got '0x%02x' " . $header_size and return unless ( $header_size == 0x10 ); substr $buf, 0, $header_size, ''; my $value_raw = substr $buf, 0, $data_size, ''; my $sd = substr $value_raw, 0, 2, ''; $value_raw =~ s/([\032\r\n]|[\032\r]|[\r\n]|[\r])+$//; Log3 $name, 5, "ONKYO_AVR $name: Unexpected start/destination: expected '!1', got '$sd'" and return unless ( $sd eq '!1' ); my $cmd_raw; my $cmd; my $value = ""; # conversion based on zone foreach my $zoneID ( keys %{ $modules{ONKYO_AVR_ZONE}{defptr}{$name} } ) { next if ( defined( $hash->{helper}{receiver}{device}{zonelist}{zone}{$zoneID} {value} ) && $hash->{helper}{receiver}{device}{zonelist}{zone}{$zoneID}{value} ne "1" ); my $commandDB = ONKYOdb::ONKYO_GetRemotecontrolCommandDetails($zoneID); foreach my $key ( keys %{$commandDB} ) { if ( $value_raw =~ s/^$key(.*)// ) { $cmd_raw = $key; $cmd = $commandDB->{$cmd_raw}{name}; $value_raw = $1; # Decode input through device information if ( $cmd eq "input" && defined( $hash->{helper}{receiver} ) && ref( $hash->{helper}{receiver} ) eq "HASH" && defined( $hash->{helper}{receiver}{input}{$value_raw} ) ) { $value = $hash->{helper}{receiver}{input}{$value_raw}; Log3 $name, 5, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): return zone$zoneID value '$value_raw' converted through device information to '" . $value . "'"; } # Decode through HASH table elsif ( defined( $commandDB->{$cmd_raw}{values}{"$value_raw"}{name} ) ) { if ( ref( $commandDB->{$cmd_raw}{values}{"$value_raw"}{name} ) eq "ARRAY" ) { $value = $commandDB->{$cmd_raw}{values}{"$value_raw"}{name}[0]; Log3 $name, 5, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): return zone$zoneID value '$value_raw' converted through ARRAY from HASH table to '" . $value . "'"; } else { $value = $commandDB->{$cmd_raw}{values}{"$value_raw"}{name}; Log3 $name, 5, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): return zone$zoneID value '$value_raw' converted through VALUE from HASH table to '" . $value . "'"; } } # return as decimal elsif ($value_raw =~ m/^[0-9A-Fa-f][0-9A-Fa-f]$/ && $cmd_raw =~ /^(MVL|ZVL|VL3|VL4|SLP|PRS|PRZ|PR3|PR4|PRM|PTS|NPR|NPZ|NP3|NP4)$/ ) { $value = ONKYO_AVR_hex2dec($value_raw); Log3 $name, 5, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): return zone$zoneID value '$value_raw' converted from HEX to DEC '$value'"; } # just return the original return value if there is # no decoding function elsif ( lc($value_raw) ne "n/a" ) { $value = $value_raw; Log3 $name, 5, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): unconverted return of zone$zoneID value '$value'"; } # end here if we got N/A result (few exceptions) elsif ($cmd ne "audio-information" && $cmd ne "video-information" ) { $value = $value_raw; Log3 $name, 4, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): device sent: zone$zoneID command unavailable"; return; } last; } } if ($cmd_raw) { $zone = $zoneID; last; } } if ( !$cmd_raw ) { $cmd_raw = substr( $value_raw, 0, 3 ); $value_raw =~ s/^...//; $cmd = "_" . $cmd_raw; $value = $value_raw; Log3 $name, 4, "ONKYO_AVR $name: con $cmd($cmd_raw$value_raw): FAIL: Don't know how to convert, not in ONKYOdb or zone may not be defined: $cmd_raw$value_raw"; return if ( !$cmd_raw || $cmd_raw eq "" ); } if ( $zone > 1 ) { Log3 $hash, 5, "ONKYO_AVR $name dispatch: this is for zone$zone"; my $zoneDispatch; $zoneDispatch->{INPUT_RAW} = $value_raw if ( $cmd eq "input" ); $zoneDispatch->{zone} = $zone; $zoneDispatch->{$cmd} = $value; Dispatch( $hash, $zoneDispatch, undef ); return; } # Parsing for zone1 (main) # Log3 $name, 4, "ONKYO_AVR $name: rcv $cmd = $value" if ( $cmd ne "net-usb-jacket-art" && $cmd ne "net-usb-time-info" ); $hash->{INPUT} = $value_raw if ( $cmd eq "input" ); my $zoneDispatch; # Update readings readingsBeginUpdate($hash); if ( $cmd eq "audio-information" ) { my @audio_split = split( /,/, $value ); if ( scalar(@audio_split) >= 6 ) { readingsBulkUpdate( $hash, "audin_src", $audio_split[0] ) if ( ReadingsVal( $name, "audin_src", "-" ) ne $audio_split[0] ); readingsBulkUpdate( $hash, "audin_enc", $audio_split[1] ) if ( ReadingsVal( $name, "audin_enc", "-" ) ne $audio_split[1] ); my ($audin_srate) = split( /[:\s]+/, $audio_split[2], 2 ) || ""; readingsBulkUpdate( $hash, "audin_srate", $audin_srate ) if ( ReadingsVal( $name, "audin_srate", "-" ) ne $audin_srate ); my ($audin_ch) = split( /[:\s]+/, $audio_split[3], 2 ) || ""; readingsBulkUpdate( $hash, "audin_ch", $audin_ch ) if ( ReadingsVal( $name, "audin_ch", "-" ) ne $audin_ch ); readingsBulkUpdate( $hash, "audout_mode", $audio_split[4] ) if ( ReadingsVal( $name, "audout_mode", "-" ) ne $audio_split[4] ); my ($audout_ch) = split( /[:\s]+/, $audio_split[5], 2 ) || ""; readingsBulkUpdate( $hash, "audout_ch", $audout_ch ) if ( ReadingsVal( $name, "audout_ch", "-" ) ne $audout_ch ); } else { foreach ( "audin_src", "audin_enc", "audin_srate", "audin_ch", "audout_ch", "audout_mode", ) { readingsBulkUpdate( $hash, $_, "" ) if ( ReadingsVal( $name, $_, "-" ) ne "" ); } } } elsif ( $cmd eq "video-information" ) { my @video_split = split( /,/, $value ); if ( scalar(@video_split) >= 8 ) { # Video-in resolution my @vidin_res_string = split( / +/, $video_split[1] ); my $vidin_res; if ( defined( $vidin_res_string[0] ) && defined( $vidin_res_string[2] ) && defined( $vidin_res_string[3] ) && uc( $vidin_res_string[0] ) ne "UNKNOWN" && uc( $vidin_res_string[2] ) ne "UNKNOWN" && uc( $vidin_res_string[3] ) ne "UNKNOWN" ) { $vidin_res = $vidin_res_string[0] . "x" . $vidin_res_string[2] . $vidin_res_string[3]; } else { $vidin_res = ""; } # Video-out resolution my @vidout_res_string = split( / +/, $video_split[5] ); my $vidout_res; if ( defined( $vidout_res_string[0] ) && defined( $vidout_res_string[2] ) && defined( $vidout_res_string[3] ) && uc( $vidout_res_string[0] ) ne "UNKNOWN" && uc( $vidout_res_string[2] ) ne "UNKNOWN" && uc( $vidout_res_string[3] ) ne "UNKNOWN" ) { $vidout_res = $vidout_res_string[0] . "x" . $vidout_res_string[2] . $vidout_res_string[3]; } else { $vidout_res = ""; } # Video-in color depth my ($vidin_cdepth) = split( /[:\s]+/, $video_split[3], 2 ) || ""; # Video-out color depth my ($vidout_cdepth) = split( /[:\s]+/, $video_split[7], 2 ) || ""; readingsBulkUpdate( $hash, "vidin_src", $video_split[0] ) if ( ReadingsVal( $name, "vidin_src", "-" ) ne $video_split[0] ); readingsBulkUpdate( $hash, "vidin_res", $vidin_res ) if ( ReadingsVal( $name, "vidin_res", "-" ) ne $vidin_res ); readingsBulkUpdate( $hash, "vidin_cspace", $video_split[2] ) if ( ReadingsVal( $name, "vidin_cspace", "-" ) ne $video_split[2] ); readingsBulkUpdate( $hash, "vidin_cdepth", $vidin_cdepth ) if ( ReadingsVal( $name, "vidin_cdepth", "-" ) ne $vidin_cdepth ); readingsBulkUpdate( $hash, "vidout_dst", $video_split[4] ) if ( ReadingsVal( $name, "vidout_dst", "-" ) ne $video_split[4] ); readingsBulkUpdate( $hash, "vidout_res", $vidout_res ) if ( ReadingsVal( $name, "vidout_res", "-" ) ne $vidout_res ); readingsBulkUpdate( $hash, "vidout_cspace", $video_split[6] ) if ( ReadingsVal( $name, "vidout_cspace", "-" ) ne $video_split[6] ); readingsBulkUpdate( $hash, "vidout_cdepth", $vidout_cdepth ) if ( ReadingsVal( $name, "vidout_cdepth", "-" ) ne $vidout_cdepth ); readingsBulkUpdate( $hash, "vidout_mode", $video_split[8] ) if ( defined( $video_split[8] ) && ReadingsVal( $name, "vidout_mode", "-" ) ne $video_split[8] ); } else { foreach ( "vidin_src", "vidin_res", "vidin_cspace", "vidin_cdepth", "vidout_dst", "vidout_res", "vidout_cspace", "vidout_cdepth", "vidout_mode", ) { readingsBulkUpdate( $hash, $_, "" ) if ( ReadingsVal( $name, $_, "-" ) ne "" ); } } } elsif ( $cmd eq "net-receiver-information" ) { if ( $value =~ /^<\?xml/ ) { no strict; my $xml_parser = XML::Simple->new( NormaliseSpace => 0, KeepRoot => 0, ForceArray => [ "zone", "netservice", "preset", "control" ], SuppressEmpty => 0, KeyAttr => { zone => "id", netservice => "id", preset => "id", control => "id", }, ); delete $hash->{helper}{receiver}; eval { $hash->{helper}{receiver} = $xml_parser->XMLin($value); }; use strict; # Safe input names my $inputs; foreach my $input ( @{ $hash->{helper}{receiver}{device}{selectorlist}{selector} } ) { if ( $input->{value} eq "1" && $input->{zone} ne "00" && $input->{id} ne "80" ) { my $id = uc( $input->{id} ); my $name = trim( $input->{name} ); $name =~ s/\s/_/g; $hash->{helper}{receiver}{input}{$id} = $name; $inputs .= $name . ":"; } } if ( !defined( $attr{$name}{inputs} ) ) { $inputs = substr( $inputs, 0, -1 ); $attr{$name}{inputs} = $inputs; } # Safe preset names my $presets; foreach my $id ( keys %{ $hash->{helper}{receiver}{device}{presetlist}{preset} } ) { my $name = trim( $hash->{helper}{receiver}{device}{presetlist}{preset}{$id} {name} ); next if ( !$name || $name eq "" ); $name =~ s/\s/_/g; $hash->{helper}{receiver}{preset}{$id} = $name; } # Zones my $reading = "zones"; if ( defined( $hash->{helper}{receiver}{device}{zonelist}{zone} ) ) { my $zones = "0"; foreach my $zoneID ( keys %{ $hash->{helper}{receiver}{device}{zonelist}{zone} } ) { next if ( $hash->{helper}{receiver}{device}{zonelist}{zone} {$zoneID}{value} ne "1" ); $zones++; } readingsBulkUpdate( $hash, $reading, $zones ) if ( ReadingsVal( $name, $reading, "" ) ne $zones ); } # Brand $reading = "brand"; if ( defined( $hash->{helper}{receiver}{device}{$reading} ) && ( !defined( $hash->{READINGS}{$reading}{VAL} ) || $hash->{READINGS}{$reading}{VAL} ne $hash->{helper}{receiver}{device}{$reading} ) ) { readingsBulkUpdate( $hash, $reading, $hash->{helper}{receiver}{device}{$reading} ); } # Model $reading = "model"; if ( defined( $hash->{helper}{receiver}{device}{$reading} )) { $hash->{model} = $hash->{helper}{receiver}{device}{$reading}; } # Firmware version $reading = "firmwareversion"; if ( defined( $hash->{helper}{receiver}{device}{$reading} ) && ( !defined( $hash->{READINGS}{$reading}{VAL} ) || $hash->{READINGS}{$reading}{VAL} ne $hash->{helper}{receiver}{device}{$reading} ) ) { readingsBulkUpdate( $hash, $reading, $hash->{helper}{receiver}{device}{$reading} ); } # device_id $reading = "deviceid"; if ( defined( $hash->{helper}{receiver}{device}{id} ) && ( !defined( $hash->{READINGS}{$reading}{VAL} ) || $hash->{READINGS}{$reading}{VAL} ne $hash->{helper}{receiver}{device}{id} ) ) { readingsBulkUpdate( $hash, $reading, $hash->{helper}{receiver}{device}{id} ); } # device_year $reading = "deviceyear"; if ( defined( $hash->{helper}{receiver}{device}{year} ) && ( !defined( $hash->{READINGS}{$reading}{VAL} ) || $hash->{READINGS}{$reading}{VAL} ne $hash->{helper}{receiver}{device}{year} ) ) { readingsBulkUpdate( $hash, $reading, $hash->{helper}{receiver}{device}{year} ); } } # Input alias handling # if ( defined( $attr{$name}{inputs} ) ) { my @inputs = split( ':', $attr{$name}{inputs} ); if (@inputs) { foreach (@inputs) { if (m/[^,\s]+(,[^,\s]+)+/) { my @input_names = split( ',', $_ ); $input_names[1] =~ s/\s/_/g; $hash->{helper}{receiver}{input_aliases} { $input_names[0] } = $input_names[1]; $hash->{helper}{receiver}{input_names} { $input_names[1] } = $input_names[0]; } } } } ONKYO_AVR_SendCommand( $hash, "input", "query" ); } elsif ( $cmd eq "net-usb-device-status" ) { if ( $value =~ /^(.)(.)(.)$/ ) { # network-connection my $netConnStatus = "none"; $netConnStatus = "ethernet" if ( $1 eq "E" ); $netConnStatus = "wireless" if ( $1 eq "W" ); readingsBulkUpdate( $hash, "networkConnection", $netConnStatus ) if ( ReadingsVal( $name, "networkConnection", "-" ) ne $netConnStatus ); # usbFront my $usbFront = "none"; $usbFront = "iOS" if ( $2 eq "i" ); $usbFront = "Memory_NAS" if ( $2 eq "M" ); $usbFront = "wireless" if ( $2 eq "W" ); $usbFront = "bluetooth" if ( $2 eq "B" ); $usbFront = "GoogleUSB" if ( $2 eq "G" ); $usbFront = "disabled" if ( $2 eq "x" ); readingsBulkUpdate( $hash, "USB_Front", $usbFront ) if ( ReadingsVal( $name, "USB_Front", "-" ) ne $usbFront ); # usbRear my $usbRear = "none"; $usbRear = "iOS" if ( $3 eq "i" ); $usbRear = "Memory_NAS" if ( $3 eq "M" ); $usbRear = "wireless" if ( $3 eq "W" ); $usbRear = "bluetooth" if ( $3 eq "B" ); $usbRear = "GoogleUSB" if ( $3 eq "G" ); $usbRear = "disabled" if ( $3 eq "x" ); readingsBulkUpdate( $hash, "USB_Rear", $usbRear ) if ( ReadingsVal( $name, "USB_Rear", "-" ) ne $usbRear ); } } elsif ($cmd eq "net-usb-jacket-art" && $value ne "on" && $value ne "off" ) { if ( $value =~ /^([012])([012\-])(.*)$/ ) { my $type = "bmp"; $type = "jpg" if ( $1 eq "1" ); $type = "link" if ( $1 eq "2" ); if ( $2 eq "-" ) { $hash->{helper}{cover}{$type}{data} = "$3"; } else { $hash->{helper}{cover}{$type}{parts} = "1" if ( "$2" eq "0" ); $hash->{helper}{cover}{$type}{parts}++ if ( "$2" ne "0" ); $hash->{helper}{cover}{$type}{data} = "" if ( "$2" eq "0" ); $hash->{helper}{cover}{$type}{data} .= "$3" if ( "$2" eq "0" || $hash->{helper}{cover}{$type}{data} ne "" ); } Log3 $name, 4, "ONKYO_AVR $name: rcv $cmd($type) in progress, part " . $hash->{helper}{cover}{$type}{parts}; # complete album art received if ( ( $2 eq "2" || $2 eq "-" ) && $type eq "link" && $hash->{helper}{cover}{$type}{data} ne "" ) { $hash->{helper}{currentCover} = $hash->{helper}{cover}{$type}{data}; readingsBulkUpdate( $hash, "currentAlbumArtURI", "" ); readingsBulkUpdate( $hash, "currentAlbumArtURL", $hash->{helper}{currentCover} ); $zoneDispatch->{currentAlbumArtURI} = ""; $zoneDispatch->{currentAlbumArtURL} = $hash->{helper}{currentCover}; } elsif ($2 eq "2" && $type ne "link" && $hash->{helper}{cover}{$type}{data} ne "" ) { my $AlbumArtName = $name . "_CurrentAlbumArt." . $type; my $AlbumArtURI = AttrVal( "global", "modpath", "." ) . "/www/images/default/ONKYO_AVR/$AlbumArtName"; my $AlbumArtURL = "?/ONKYO_AVR/cover/$AlbumArtName"; mkpath( AttrVal( "global", "modpath", "." ) . '/www/images/default/ONKYO_AVR/' ); ONKYO_AVR_WriteFile( $AlbumArtURI, ONKYO_AVR_hex2image( $hash->{helper}{cover}{$type}{data} ) ); Log3 $name, 4, "ONKYO_AVR $name: rcv $cmd($type) completed in " . $hash->{helper}{cover}{$type}{parts} . " parts. Saved to $AlbumArtURI"; delete $hash->{helper}{cover}{$type}{data}; $hash->{helper}{currentCover} = $AlbumArtURI; readingsBulkUpdate( $hash, "currentAlbumArtURI", $AlbumArtURI ); readingsBulkUpdate( $hash, "currentAlbumArtURL", $AlbumArtURL ); $zoneDispatch->{currentAlbumArtURI} = $AlbumArtURI; $zoneDispatch->{currentAlbumArtURL} = $AlbumArtURL; } } else { Log3 $name, 4, "ONKYO_AVR $name: received cover art tile could not be decoded: " . $value; } } # currentTrackPosition # currentTrackDuration elsif ( $cmd eq "net-usb-time-info" ) { my @times = split( /\//, $value ); if ( gettimeofday() - time_str2num( ReadingsTimestamp( $name, "currentTrackPosition", "1970-01-01 01:00:00" ) ) >= 5 ) { readingsBulkUpdate( $hash, "currentTrackPosition", $times[0] ) if ( ReadingsVal( $name, "currentTrackPosition", "-" ) ne $times[0] ); $zoneDispatch->{currentTrackPosition} = $times[0]; } if ( ReadingsVal( $name, "currentTrackDuration", "-" ) ne $times[1] ) { readingsBulkUpdate( $hash, "currentTrackDuration", $times[1] ); $zoneDispatch->{currentTrackDuration} = $times[1]; } } # currentArtist elsif ( $cmd eq "net-usb-artist-name-info" ) { readingsBulkUpdate( $hash, "currentArtist", $value ) if ( ReadingsVal( $name, "currentArtist", "-" ) ne $value ); $zoneDispatch->{currentArtist} = $value; } # currentAlbum elsif ( $cmd eq "net-usb-album-name-info" ) { readingsBulkUpdate( $hash, "currentAlbum", $value ) if ( ReadingsVal( $name, "currentAlbum", "-" ) ne $value ); $zoneDispatch->{currentAlbum} = $value; } # currentTitle elsif ( $cmd eq "net-usb-title-name" ) { readingsBulkUpdate( $hash, "currentTitle", $value ) if ( ReadingsVal( $name, "currentTitle", "-" ) ne $value ); $zoneDispatch->{currentTitle} = $value; } elsif ( $cmd eq "net-usb-list-title-info" ) { if ( $value =~ /^(..)(.)(.)(....)(....)(..)(..)(..)(..)(..)(.*)$/ ) { # channel my $channel = $1 || "00"; my $channelUc = uc($channel); $hash->{CHANNEL} = $channel; $channel = lc($channel); my $channelname = ""; # Get all details for command my $command_details = ONKYOdb::ONKYO_GetRemotecontrolCommandDetails( "1", ONKYOdb::ONKYO_GetRemotecontrolCommand( "1", "net-service" ) ); # we know the channel name from receiver info if ( defined( $hash->{helper}{receiver} ) && ref( $hash->{helper}{receiver} ) eq "HASH" && defined( $hash->{helper}{receiver}{device}{netservicelist} {netservice}{$channel}{name} ) ) { $channelname = $hash->{helper}{receiver}{device}{netservicelist} {netservice}{$channel}{name}; $channelname =~ s/\s/_/g; } # we know the channel name from ONKYOdb elsif ( defined( $command_details->{values}{$channelUc} ) ) { if ( ref( $command_details->{values}{$channelUc}{name} ) eq "ARRAY" ) { $channelname = $command_details->{values}{$channelUc}{name}[0]; } else { $channelname = $command_details->{values}{$channelUc}{name}; } } # some specials for net-usb-list-title-info elsif ( $channel =~ /^f./ ) { $channelname = "USB_Front" if $channel eq "f0"; $channelname = "USB_Rear" if $channel eq "f1"; $channelname = "Internet_Radio" if $channel eq "f2"; $channelname = "" if $channel eq "f3"; } # we don't know the channel name, sorry else { Log3 $name, 4, "ONKYO_AVR $name: net-usb-list-title-info: received unknown channel ID $channel"; $channelname = $channel; } if ( ReadingsVal( $name, "channel", "-" ) ne $channelname ) { my $currentAlbumArtURI = AttrVal( "global", "modpath", "." ) . "/FHEM/lib/UPnP/sonos_empty.jpg"; my $currentAlbumArtURL = "?/ONKYO_AVR/cover/empty.jpg"; readingsBulkUpdate( $hash, "channel", $channelname ); readingsBulkUpdate( $hash, "currentAlbum", "" ) if ( ReadingsVal( $name, "currentAlbum", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentAlbumArtURI", $currentAlbumArtURI ) if ( ReadingsVal( $name, "currentAlbumArtURI", "-" ) ne $currentAlbumArtURI ); readingsBulkUpdate( $hash, "currentAlbumArtURL", $currentAlbumArtURL ) if ( ReadingsVal( $name, "currentAlbumArtURL", "-" ) ne $currentAlbumArtURL ); readingsBulkUpdate( $hash, "currentArtist", "" ) if ( ReadingsVal( $name, "currentArtist", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentTitle", "" ) if ( ReadingsVal( $name, "currentTitle", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentTrackPosition", "--:--" ) if ( ReadingsVal( $name, "currentTrackPosition", "-" ) ne "--:--" ); readingsBulkUpdate( $hash, "currentTrackDuration", "--:--" ) if ( ReadingsVal( $name, "currentTrackDuration", "-" ) ne "--:--" ); $zoneDispatch->{CHANNEL_RAW} = $hash->{CHANNEL}; $zoneDispatch->{channel} = $channelname; $zoneDispatch->{currentAlbum} = ""; $zoneDispatch->{currentAlbumArtURI} = $currentAlbumArtURI; $zoneDispatch->{currentAlbumArtURL} = $currentAlbumArtURL; $zoneDispatch->{currentArtist} = ""; $zoneDispatch->{currentTitle} = ""; $zoneDispatch->{currentTrackPosition} = "--:--"; $zoneDispatch->{currentTrackDuration} = "--:--"; } # screenType my $screenType = $2 || "0"; my $uiTypes = { '0' => 'List', '1' => 'Menu', '2' => 'Playback', '3' => 'Popup', '4' => 'Keyboard', '5' => 'Menu List', }; my $uiType = $uiTypes->{$screenType}; readingsBulkUpdate( $hash, "screenType", $screenType ) if ( ReadingsVal( $name, "screenType", "-" ) ne $screenType ); # screenLayerInfo my $screenLayerInfo = $3 || "0"; my $layerInfos = { '0' => 'NET TOP', '1' => 'Service Top,DLNA/USB/iPod', '2' => 'under 2nd Layer', }; my $layerInfo = $layerInfos->{$screenLayerInfo}; $hash->{SCREENLAYER} = $screenLayerInfo; readingsBulkUpdate( $hash, "screenLayerInfo", $screenLayerInfo ) if ( readingsBulkUpdate( $hash, "screenLayerInfo", "" ) ne $screenLayerInfo ); # screenListPos my $screenListPos = $4 || "0000"; foreach my $line ( keys %{ $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list} } ) { $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$line}{listpos} = 0; } $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$screenListPos} {listpos} = 1 if ( $screenListPos ne "-" ); readingsBulkUpdate( $hash, "screenListPos", $screenListPos ) if ( readingsBulkUpdate( $hash, "screenListPos", "" ) ne $screenListPos ); # screenItemCnt my $screenItemCnt = $5 || "0000"; readingsBulkUpdate( $hash, "screenItemCnt", $screenItemCnt ) if ( ReadingsVal( $name, "screenItemCnt", "-" ) ne $screenItemCnt ); # screenLayer my $screenLayer = $6 || "00"; readingsBulkUpdate( $hash, "screenLayer", $screenLayer ) if ( ReadingsVal( $name, "screenLayer", "-" ) ne $screenLayer ); # my $reserved = $7; my $screenIconLeft = $8 || "00"; my $iconsLeft = { '00' => 'Internet Radio', '01' => 'Server', '02' => 'USB', '03' => 'iPod', '04' => 'DLNA', '05' => 'WiFi', '06' => 'Favorite', '10' => 'Account(Spotify)', '11' => 'Album(Spotify)', '12' => 'Playlist(Spotify)', '13' => 'Playlist-C(Spotify)', '14' => 'Starred(Spotify)', '15' => 'What\'s New(Spotify)', '16' => 'Track(Spotify)', '17' => 'Artist(Spotify)', '18' => 'Play(Spotify)', '19' => 'Search(Spotify)', '1A' => 'Folder(Spotify)', 'FF' => 'None' }; my $iconLeft = $iconsLeft->{$screenIconLeft}; readingsBulkUpdate( $hash, "screenIconLeft", $screenIconLeft ) if ( ReadingsVal( $name, "screenIconLeft", "-" ) ne $screenIconLeft ); my $screenIconRight = $9 || "00"; my $iconsRight = { '00' => 'DLNA', '01' => 'Favorite', '02' => 'vTuner', '03' => 'SiriusXM', '04' => 'Pandora', '05' => 'Rhapsody', '06' => 'Last.fm', '07' => 'Napster', '08' => 'Slacker', '09' => 'Mediafly', '0A' => 'Spotify', '0B' => 'AUPEO!', '0C' => 'radiko', '0D' => 'e-onkyo', '0E' => 'TuneIn Radio', '0F' => 'MP3tunes', '10' => 'Simfy', '11' => 'Home Media', 'FF' => 'None' }; my $iconRight = $iconsRight->{$screenIconRight}; readingsBulkUpdate( $hash, "screenIconRight", $screenIconRight ) if ( ReadingsVal( $name, "screenIconRight", "-" ) ne $screenIconRight ); # screenStatus my $screenStatus = $10 || "00"; my $statusInfos = { '00' => '', '01' => 'Connecting', '02' => 'Acquiring License', '03' => 'Buffering', '04' => 'Cannot Play', '05' => 'Searching', '06' => 'Profile update', '07' => 'Operation disabled', '08' => 'Server Start-up', '09' => 'Song rated as Favorite', '0A' => 'Song banned from station', '0B' => 'Authentication Failed', '0C' => 'Spotify Paused(max 1 device)', '0D' => 'Track Not Available', '0E' => 'Cannot Skip' }; my $statusInfo = $statusInfos->{$screenStatus}; if ( defined( $statusInfos->{$screenStatus} ) ) { readingsBulkUpdate( $hash, "screenStatus", $statusInfos->{$screenStatus} ) if ( ReadingsVal( $name, "screenStatus", "-" ) ne $statusInfos->{$screenStatus} ); } else { readingsBulkUpdate( $hash, "screenStatus", $screenStatus ) if ( ReadingsVal( $name, "screenStatus", "-" ) ne $screenStatus ); } # screenTitle my $screenTitle = $11 || ""; $screenTitle = "" if ( $screenTitle eq "NE" ); readingsBulkUpdate( $hash, "screenTitle", $screenTitle ) if ( ReadingsVal( $name, "screenTitle", "-" ) ne $screenTitle ); } } elsif ( $cmd eq "net-usb-menu-status" ) { if ( $value =~ /^(.)(..)(..)(.)(.)(..)$/ ) { my $menuState = $1; } } # screen/list elsif ( $cmd eq "net-usb-list-info" ) { if ( $value =~ /^(.)(.)(.)(.*)/ ) { my $item; if ( $2 eq "-" ) { $item = $2; } elsif ( $2 < 10 ) { $item = "000" . $2; } elsif ( $2 < 100 ) { $item = "00" . $2; } elsif ( $2 < 1000 ) { $item = "0" . $2; } my $properties; if ( $1 ne "C" ) { $properties = { '-' => 'no', '0' => 'Playing', 'A' => 'Artist', 'B' => 'Album', 'F' => 'Folder', 'M' => 'Music', 'P' => 'Playlist', 'S' => 'Search', 'a' => 'Account', 'b' => 'Playlist-C', 'c' => 'Starred', 'd' => 'Unstarred', 'e' => 'What\'s New' }; } # line item details if ( $1 eq "A" || $1 eq "U" ) { $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$item}{property} = $3; $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$item}{data} = $4; $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$item}{curser} = 0 if ( !defined( $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list} {$item}{curser} ) ); # screenItemType readingsBulkUpdate( $hash, "screenItemT" . $item, $3 ) if ( ReadingsVal( $name, "screenItemT" . $item, "-" ) ne $3 ); # screenItemContent readingsBulkUpdate( $hash, "screenItemC" . $item, $4 ) if ( ReadingsVal( $name, "screenItemC" . $item, "-" ) ne $4 ); } #Cursor Position (Update) elsif ( $1 eq "C" ) { foreach my $item ( keys %{ $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list} } ) { $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$item}{curser} = 0; } $hash->{SCREEN}{ $hash->{SCREENLAYER} }{list}{$item}{curser} = 1 if ( $item ne "-" ); readingsBulkUpdate( $hash, "screenCurser", $2 ) if ( ReadingsVal( $name, "screenCurser", "" ) ne $2 ); if ( $3 eq "P" ) { #Page Information Update (Page Clear or Disable List Info) Log3 $name, 4, "ONKYO_AVR $name: page clear"; for ( my $idx=0; $idx < 10; $idx++ ) { readingsBulkUpdate( $hash, "screenItemC000" . $idx, '' ); } } } } else { Log3 $name, 4, "ONKYO_AVR $name: screen/list: ERROR - unable to parse: " . $value; } } # screen/list XML elsif ( $cmd eq "net-usb-list-info-xml" ) { if ( $value =~ /^(.)(....)(.)(.)(..)(.*)/ ) { Log3 $name, 4, "ONKYO_AVR $name: rcv $cmd($1) unknown type" and return if ( $1 ne "X" ); Log3 $name, 4, "ONKYO_AVR $name: rcv $cmd($1) in progress"; my $uiTypes = { '0' => 'List', '1' => 'Menu', '2' => 'Playback', '3' => 'Popup', '4' => 'Keyboard', '5' => 'Menu List', }; my $uiType = $uiTypes->{$4}; $hash->{helper}{listinfo}{$3}{$2} = $6; } else { Log3 $name, 4, "ONKYO_AVR $name: net-usb-list-info-xml could not be parsed: " . $value; } } elsif ( $cmd eq "net-usb-play-status" ) { if ( $value =~ /^(.)(.)(.)$/ ) { my $status; # playStatus $status = "stopped"; $status = "playing" if ( $1 eq "P" ); $status = "paused" if ( $1 eq "p" ); $status = "fast-forward" if ( $1 eq "F" ); $status = "fast-rewind" if ( $1 eq "R" ); $status = "interrupted" if ( $1 eq "E" ); readingsBulkUpdate( $hash, "playStatus", $status ) if ( ReadingsVal( $name, "playStatus", "-" ) ne $status ); $zoneDispatch->{playStatus} = $status; # stateAV my $stateAV = ONKYO_AVR_GetStateAV($hash); readingsBulkUpdate( $hash, "stateAV", $stateAV ) if ( ReadingsVal( $name, "stateAV", "-" ) ne $stateAV ); if ( $status eq "stopped" ) { my $currentAlbumArtURI = AttrVal( "global", "modpath", "." ) . "/FHEM/lib/UPnP/sonos_empty.jpg"; my $currentAlbumArtURL = "?/ONKYO_AVR/cover/empty.jpg"; readingsBulkUpdate( $hash, "currentAlbum", "" ) if ( ReadingsVal( $name, "currentAlbum", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentAlbumArtURI", $currentAlbumArtURI ) if ( ReadingsVal( $name, "currentAlbumArtURI", "-" ) ne $currentAlbumArtURI ); readingsBulkUpdate( $hash, "currentAlbumArtURL", $currentAlbumArtURL ) if ( ReadingsVal( $name, "currentAlbumArtURL", "-" ) ne $currentAlbumArtURL ); readingsBulkUpdate( $hash, "currentArtist", "" ) if ( ReadingsVal( $name, "currentArtist", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentTitle", "" ) if ( ReadingsVal( $name, "currentTitle", "-" ) ne "" ); readingsBulkUpdate( $hash, "currentTrackPosition", "--:--" ) if ( ReadingsVal( $name, "currentTrackPosition", "-" ) ne "--:--" ); readingsBulkUpdate( $hash, "currentTrackDuration", "--:--" ) if ( ReadingsVal( $name, "currentTrackDuration", "-" ) ne "--:--" ); $zoneDispatch->{currentAlbum} = ""; $zoneDispatch->{currentAlbumArtURI} = $currentAlbumArtURI; $zoneDispatch->{currentAlbumArtURL} = $currentAlbumArtURL; $zoneDispatch->{currentArtist} = ""; $zoneDispatch->{currentTitle} = ""; $zoneDispatch->{currentTrackPosition} = "--:--"; $zoneDispatch->{currentTrackDuration} = "--:--"; } # repeat $status = "-"; $status = "off" if ( $2 eq "-" ); $status = "all" if ( $2 eq "R" ); $status = "all-folder" if ( $2 eq "F" ); $status = "one" if ( $2 eq "1" ); readingsBulkUpdate( $hash, "repeat", $status ) if ( ReadingsVal( $name, "repeat", "-" ) ne $status ); $zoneDispatch->{repeat} = $status; # shuffle $status = "-"; $status = "off" if ( $2 eq "-" ); $status = "on" if ( $3 eq "S" ); $status = "on-album" if ( $3 eq "A" ); $status = "on-folder" if ( $3 eq "F" ); readingsBulkUpdate( $hash, "shuffle", $status ) if ( ReadingsVal( $name, "shuffle", "-" ) ne $status ); $zoneDispatch->{shuffle} = $status; } } elsif ( $cmd =~ /^net-usb/ && $value ne "on" && $value ne "off" ) { } elsif ( $cmd =~ /^net-keyboard/ ) { } # net-popup-* elsif ( $cmd eq "net-popup-message" ) { if ( $value =~ /^(B|T|L)(.*[a-z])([A-Z].*[a-z.!?])(0|1|2)([A-Z].*[a-z])$/ ) { readingsBulkUpdate( $hash, "net-popup-type", "top" ) if ( $1 eq "T" ); readingsBulkUpdate( $hash, "net-popup-type", "bottom" ) if ( $1 eq "B" ); readingsBulkUpdate( $hash, "net-popup-type", "list" ) if ( $1 eq "L" ); readingsBulkUpdate( $hash, "net-popup-title", $2 ); readingsBulkUpdate( $hash, "net-popup-text", $3 ); readingsBulkUpdate( $hash, "net-popup-button-position", "hidden" ) if ( $4 eq "0" || $4 eq "" ); readingsBulkUpdate( $hash, "net-popup-button-position", $4 ) if ( $4 ne "0" && $4 ne "" ); readingsBulkUpdate( $hash, "net-popup-button1-text", $5 ); $zoneDispatch->{"net-popup-type"} = ReadingsVal( $name, "net-popup-type", "" ); $zoneDispatch->{"net-popup-title"} = $2; $zoneDispatch->{"net-popup-text"} = $3; $zoneDispatch->{"net-popup-button-position"} = ReadingsVal( $name, "net-popup-button-position", "" ); $zoneDispatch->{"net-popup-button1-text"} = $5; } else { Log3 $name, 4, "ONKYO_AVR $name: Could not decompile net-popup-message: $value"; } } # tone-* elsif ( $cmd =~ /^tone-/ ) { if ( $value =~ /^B(..)T(..)$/ ) { my $bass = $1; my $treble = $2; my $bassName = $cmd . "-bass"; my $trebleName = $cmd . "-treble"; my $prefixBass = ""; my $prefixTreble = ""; # tone-*-bass $prefixBass = "-" if ( $bass =~ /^\-.*/ ); $bass = substr( $bass, 1 ) if ( $bass =~ /^[\+|\-].*/ ); $bass = $prefixBass . ONKYO_AVR_hex2dec($bass); readingsBulkUpdate( $hash, $bassName, $bass ) if ( ReadingsVal( $name, $bassName, "-" ) ne $bass ); # tone-*-treble $prefixTreble = "-" if ( $treble =~ /^\-.*/ ); $treble = substr( $treble, 1 ) if ( $treble =~ /^[\+|\-].*/ ); $treble = $prefixTreble . ONKYO_AVR_hex2dec($treble); readingsBulkUpdate( $hash, $trebleName, $treble ) if ( ReadingsVal( $name, $trebleName, "-" ) ne $treble ); } # tone-subwoofer elsif ( $value =~ /^B(..)$/ ) { my $bass = $1; my $prefix = ""; $prefix = "-" if ( $bass =~ /^\-.*/ ); $bass = substr( $bass, 1 ) if ( $bass =~ /^[\+|\-].*/ ); $bass = $prefix . ONKYO_AVR_hex2dec($bass); readingsBulkUpdate( $hash, $cmd, $bass ) if ( ReadingsVal( $name, $cmd, "-" ) ne $bass ); } } else { if ( $cmd eq "input" ) { # Input alias handling if ( defined( $hash->{helper}{receiver}{input_aliases}{$value} ) ) { Log3 $name, 4, "ONKYO_AVR $name: Input aliasing '$value' to '" . $hash->{helper}{receiver}{input_aliases}{$value} . "'"; $value = $hash->{helper}{receiver}{input_aliases}{$value}; } } # subwoofer-temporary-level # center-temporary-level elsif ($cmd eq "subwoofer-temporary-level" || $cmd eq "center-temporary-level" ) { my $prefix = ""; $prefix = "-" if ( $value =~ /^\-.*/ ); $value = substr( $value, 1 ) if ( $value =~ /^[\+|\-].*/ ); $value = $prefix . ONKYO_AVR_hex2dec($value); } # preset elsif ( $cmd eq "preset" ) { if ( defined( $hash->{helper}{receiver}{preset} ) ) { foreach my $id ( sort keys %{ $hash->{helper}{receiver}{preset} } ) { my $presetName = $hash->{helper}{receiver}{preset}{$id}; next if ( !$presetName || $presetName eq "" ); $presetName =~ s/\s/_/g; if ( $id eq $value ) { $value = $presetName; last; } } } $value = "" if ( $value eq "0" ); $zoneDispatch->{preset} = $value; } readingsBulkUpdate( $hash, $cmd, $value ) if ( ReadingsVal( $name, $cmd, "-" ) ne $value ); # stateAV my $stateAV = ONKYO_AVR_GetStateAV($hash); readingsBulkUpdate( $hash, "stateAV", $stateAV ) if ( ReadingsVal( $name, "stateAV", "-" ) ne $stateAV ); } readingsEndUpdate( $hash, 1 ); if ( $zoneDispatch && $definedZones > 1 ) { Log3 $name, 5, "ONKYO_AVR $name: Forwarding information from main zone1 to slave zones"; Dispatch( $hash, $zoneDispatch, undef ); } return; } sub ONKYO_AVR_Write($$) { my ( $hash, $cmd ) = @_; my $name = $hash->{NAME}; my $str = ONKYO_AVR_Pack( $cmd, $hash->{PROTOCOLVERSION} ); Log3 $name, 1, "ONKYO_AVR $name: $hash->{DeviceName} snd ERROR - could not transcode $cmd to HEX command" and return if ( !$str ); # Log3 $name, 5, # "ONKYO_AVR $name: $hash->{DeviceName} snd " . ONKYO_AVR_hexdump($str); Log3 $name, 5, "ONKYO_AVR $name: $hash->{DeviceName} snd $str"; DevIo_SimpleWrite( $hash, "$str", 0 ); # do connection check latest after TIMEOUT my $next = gettimeofday() + $hash->{TIMEOUT}; if ( !defined( $hash->{helper}{nextConnectionCheck} ) || $hash->{helper}{nextConnectionCheck} > $next ) { $hash->{helper}{nextConnectionCheck} = $next; RemoveInternalTimer($hash); InternalTimer( $next, "ONKYO_AVR_connectionCheck", $hash, 0 ); } } sub ONKYO_AVR_Ready($) { my ($hash) = @_; my $name = $hash->{NAME}; if ( ReadingsVal( $name, "state", "disconnected" ) eq "disconnected" ) { DevIo_OpenDev( $hash, 1, undef, sub() { my ( $hash, $err ) = @_; Log3 $name, 4, "ONKYO_AVR $name: $err" if ($err); } ); return; } # This is relevant for windows/USB only my $po = $hash->{USBDev}; my ( $BlockingFlags, $InBytes, $OutBytes, $ErrorFlags ); if ($po) { ( $BlockingFlags, $InBytes, $OutBytes, $ErrorFlags ) = $po->status; } return ( $InBytes && $InBytes > 0 ); } sub ONKYO_AVR_Notify($$) { my ( $hash, $dev ) = @_; my $name = $hash->{NAME}; my $devName = $dev->{NAME}; my $definedZones = scalar keys %{ $modules{ONKYO_AVR_ZONE}{defptr}{$name} }; my $presence = ReadingsVal( $name, "presence", "-" ); return if ( !$dev->{CHANGED} ); # Some previous notify deleted the array. # work on global events related to us if ( $devName eq "global" ) { foreach my $change ( @{ $dev->{CHANGED} } ) { if ( $change !~ /^(\w+)\s(\w+)\s?(\w*)\s?(.*)$/ || $2 ne $name ) { return; } # DEFINED # MODIFIED elsif ( $1 eq "DEFINED" || $1 eq "MODIFIED" ) { Log3 $hash, 5, "ONKYO_AVR " . $name . ": processing my global event $1: $3 -> $4"; if ( lc( ReadingsVal( $name, "state", "?" ) ) eq "opened" ) { DoTrigger( $name, "CONNECTED" ); } else { DoTrigger( $name, "DISCONNECTED" ); } } # unknown event else { Log3 $hash, 5, "ONKYO_AVR " . $name . ": WONT BE processing my global event $1: $3 -> $4"; } } return; } # do nothing for any other device elsif ( $devName ne $name ) { return; } readingsBeginUpdate($hash); foreach my $change ( @{ $dev->{CHANGED} } ) { # DISCONNECTED if ( $change eq "DISCONNECTED" ) { Log3 $hash, 5, "ONKYO_AVR " . $name . ": processing change $change"; # disable connectionCheck and wait # until DevIo reopened the connection RemoveInternalTimer($hash); readingsBulkUpdate( $hash, "presence", "absent" ) if ( $presence ne "absent" ); readingsBulkUpdate( $hash, "power", "off" ) if ( ReadingsVal( $name, "power", "on" ) ne "off" ); # stateAV my $stateAV = ONKYO_AVR_GetStateAV($hash); readingsBulkUpdate( $hash, "stateAV", $stateAV ) if ( ReadingsVal( $name, "stateAV", "-" ) ne $stateAV ); # send to slaves if ( $definedZones > 1 ) { Log3 $name, 5, "ONKYO_AVR $name: Dispatching state change to slaves"; Dispatch( $hash, { "presence" => "absent", "power" => "off", }, undef ); } } # CONNECTED elsif ( $change eq "CONNECTED" ) { Log3 $hash, 5, "ONKYO_AVR " . $name . ": processing change $change"; readingsBulkUpdate( $hash, "presence", "present" ) if ( $presence ne "present" ); # stateAV my $stateAV = ONKYO_AVR_GetStateAV($hash); readingsBulkUpdate( $hash, "stateAV", $stateAV ) if ( ReadingsVal( $name, "stateAV", "-" ) ne $stateAV ); ONKYO_AVR_SendCommand( $hash, "power", "query" ); ONKYO_AVR_SendCommand( $hash, "network-standby", "query" ); ONKYO_AVR_SendCommand( $hash, "input", "query" ); ONKYO_AVR_SendCommand( $hash, "mute", "query" ); ONKYO_AVR_SendCommand( $hash, "volume", "query" ); ONKYO_AVR_SendCommand( $hash, "sleep", "query" ); ONKYO_AVR_SendCommand( $hash, "audio-information", "query" ); ONKYO_AVR_SendCommand( $hash, "video-information", "query" ); ONKYO_AVR_SendCommand( $hash, "listening-mode", "query" ); ONKYO_AVR_SendCommand( $hash, "video-picture-mode", "query" ); ONKYO_AVR_SendCommand( $hash, "phase-matching-bass", "query" ); ONKYO_AVR_SendCommand( $hash, "center-temporary-level", "query" ); ONKYO_AVR_SendCommand( $hash, "subwoofer-temporary-level", "query" ); fhem "sleep 1 quiet;get $name remoteControl net-receiver-information query quiet"; # send to slaves if ( $definedZones > 1 ) { Log3 $name, 5, "ONKYO_AVR $name: Dispatching state change to slaves"; Dispatch( $hash, { "presence" => "present", }, undef ); } } } readingsEndUpdate( $hash, 1 ); } sub ONKYO_AVR_Shutdown($) { my ($hash) = @_; my $name = $hash->{NAME}; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_Shutdown()"; DevIo_CloseDev($hash); return undef; } # module Fn #################################################################### sub ONKYO_AVR_DevInit($) { my ($hash) = @_; my $name = $hash->{NAME}; if ( lc( ReadingsVal( $name, "state", "?" ) ) eq "opened" ) { DoTrigger( $name, "CONNECTED" ); } else { DoTrigger( $name, "DISCONNECTED" ); } } sub ONKYO_AVR_addExtension($$$) { my ( $name, $func, $link ) = @_; my $url = "?/$link"; return 0 if ( defined( $data{FWEXT}{$url} ) && $data{FWEXT}{$url}{deviceName} ne $name ); Log3 $name, 2, "ONKYO_AVR $name: Registering ONKYO_AVR for webhook URI $url ..."; $data{FWEXT}{$url}{deviceName} = $name; $data{FWEXT}{$url}{FUNC} = $func; $data{FWEXT}{$url}{LINK} = $link; return 1; } sub ONKYO_AVR_removeExtension($) { my ($link) = @_; my $url = "?/$link"; my $name = $data{FWEXT}{$url}{deviceName}; Log3 $name, 2, "ONKYO_AVR $name: Unregistering ONKYO_AVR for webhook URI $url..."; delete $data{FWEXT}{$url}; } sub ONKYO_AVR_CGI() { my ($request) = @_; # data received if ( $request =~ m,^\?\/ONKYO_AVR\/cover\/(.+)\.(.+)$, ) { Log3 undef, 5, "ONKYO_AVR: sending cover $1.$2"; if ( $1 eq "empty" && $2 eq "jpg" ) { FW_serveSpecial( 'sonos_empty', 'jpg', AttrVal( "global", "modpath", "." ) . '/FHEM/lib/UPnP', 1 ); } else { FW_serveSpecial( $1, $2, AttrVal( "global", "modpath", "." ) . '/www/images/default/ONKYO_AVR', 1 ); } return ( undef, undef ); } # no data received else { Log3 undef, 5, "ONKYO_AVR: received malformed request\n$request"; } return ( "text/plain; charset=utf-8", "Call failure: " . $request ); } sub ONKYO_AVR_SendCommand($$$) { my ( $hash, $cmd, $value ) = @_; my $name = $hash->{NAME}; my $zone = $hash->{ZONE}; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_SendCommand()"; # Input alias handling if ( $cmd eq "input" ) { # Resolve input alias to correct name if ( defined( $hash->{helper}{receiver}{input_names}{$value} ) ) { $value = $hash->{helper}{receiver}{input_names}{$value}; } # Resolve device specific input alias $value =~ s/_/ /g; if ( defined( $hash->{helper}{receiver}{device}{selectorlist}{selector} ) && ref( $hash->{helper}{receiver}{device}{selectorlist}{selector} ) eq "ARRAY" ) { foreach my $input ( @{ $hash->{helper}{receiver}{device}{selectorlist}{selector} } ) { if ( $input->{value} eq "1" && $input->{zone} ne "00" && $input->{id} ne "80" && $value eq trim( $input->{name} ) ) { $value = uc( $input->{id} ); last; } } } } # Resolve command and value to ISCP raw command my $cmd_raw = ONKYOdb::ONKYO_GetRemotecontrolCommand( $zone, $cmd ); my $value_raw = ONKYOdb::ONKYO_GetRemotecontrolValue( $zone, $cmd_raw, $value ); if ( !defined($cmd_raw) ) { Log3 $name, 4, "ONKYO_AVR $name: command '$cmd$value' is an unregistered command within zone$zone, be careful! Will be handled as raw command"; $cmd_raw = $cmd; $value_raw = $value; } elsif ( !defined($value_raw) ) { Log3 $name, 4, "ONKYO_AVR $name: $cmd - Warning, value '$value' not found in HASH table, will be sent to receiver 'as is'"; $value_raw = $value; } Log3 $name, 4, "ONKYO_AVR $name: snd $cmd -> $value ($cmd_raw$value_raw)"; if ( $cmd_raw ne "" && $value_raw ne "" ) { ONKYO_AVR_Write( $hash, $cmd_raw . $value_raw ); } return; } sub ONKYO_AVR_SendRawCommand ($$) { my ( $hash, $rawCmd ) = @_; my $name = $hash->{NAME}; my $zone = $hash->{ZONE}; Log3 $name, 5, "ONKYO_AVR $name: called function ONKYO_AVR_SendRawCommand()"; ONKYO_AVR_Write( $hash, $rawCmd ); return; } sub ONKYO_AVR_connectionCheck ($) { my ($hash) = @_; my $name = $hash->{NAME}; my $verbose = AttrVal( $name, "verbose", "" ); RemoveInternalTimer($hash); $hash->{STATE} = "opened"; # assume we have an open connection $attr{$name}{verbose} = 0 if ( $verbose eq "" || $verbose < 4 ); my $connState = DevIo_Expect( $hash, ONKYO_AVR_Pack( "PWRQSTN", $hash->{PROTOCOLVERSION} ), $hash->{TIMEOUT} ); # successful connection if ( defined($connState) ) { # reset connectionCheck timer my $checkInterval = AttrVal( $name, "connectionCheck", "60" ); if ( $checkInterval ne "off" ) { my $next = gettimeofday() + $checkInterval; $hash->{helper}{nextConnectionCheck} = $next; InternalTimer( $next, "ONKYO_AVR_connectionCheck", $hash, 0 ); } } $attr{$name}{verbose} = $verbose if ( $verbose ne "" ); delete $attr{$name}{verbose} if ( $verbose eq "" ); } sub ONKYO_AVR_WriteFile($$) { my ( $fileName, $data ) = @_; open IMGFILE, '>' . $fileName; binmode IMGFILE; print IMGFILE $data; close IMGFILE; } sub ONKYO_AVR_Pack($;$) { my ( $d, $protocol ) = @_; # ------------------ # < 2013 (produced by TX-NR515) # ------------------ # # EXAMPLE REQUEST FOR PWRQSTN # 4953 4350 0000 0010 0000 000a 0100 0000 ISCP............ # 2131 5057 5251 5354 4e0d !1PWRQSTN. # # EXAMPLE REPLY FOR PWRQSTN # 4953 4350 0000 0010 0000 000a 0100 0000 ISCP............ # 2131 5057 5230 311a 0d0a !1PWR01... # # ------------------ # 2013+ (produced by TX-NR626) # ------------------ # # EXAMPLE REQUEST FOR PWRQSTN # 4953 4350 0000 0010 0000 000b 0100 0000 ISCP............ # 2131 5057 5251 5354 4e0d 0a !1PWRQSTN.. # # EXAMPLE REPLY FOR PWRQSTN # 4953 4350 0000 0010 0000 000a 0100 0000 ISCP............ # 2131 5057 5230 311a 0d0a !1PWR01... # # add start character and destination unit type 1=receiver $d = '!1' . $d; # If protocol is defined as pre-2013 use EOF code for older models if ( defined($protocol) && $protocol eq "pre2013" ) { # = 0x0d $d .= "\r"; } # otherwise use EOF code for newer models else { # = 0x0d0a $d .= "\r\n"; } pack( "a* N N N a*", 'ISCP', 0x10, ( length $d ), 0x01000000, $d ); } sub ONKYO_AVR_hexdump { my $s = shift; my $r = unpack 'H*', $s; $s =~ s/[^ -~]/./g; $r . ' ' . $s; } sub ONKYO_AVR_hex2dec($) { my ($hex) = @_; return unpack( 's', pack 's', hex($hex) ); } sub ONKYO_AVR_hex2image($) { my ($hex) = @_; return pack( "H*", $hex ); } sub ONKYO_AVR_dec2hex($) { my ($dec) = @_; my $hex = uc( sprintf( "%x", $dec ) ); return "0" . $hex if ( length($hex) eq 1 ); return $hex; } sub ONKYO_AVR_GetStateAV($) { my ($hash) = @_; my $name = $hash->{NAME}; if ( ReadingsVal( $name, "presence", "absent" ) eq "absent" ) { return "absent"; } elsif ( ReadingsVal( $name, "power", "off" ) eq "off" ) { return "off"; } elsif ( ReadingsVal( $name, "mute", "off" ) eq "on" ) { return "muted"; } elsif ( $hash->{INPUT} eq "2B" && ReadingsVal( $name, "playStatus", "stopped" ) ne "stopped" ) { return ReadingsVal( $name, "playStatus", "stopped" ); } else { return ReadingsVal( $name, "power", "off" ); } } sub ONKYO_AVR_RCmakenotify($$) { my ( $name, $ndev ) = @_; my $nname = "notify_$name"; fhem( "define $nname notify $name set $ndev remoteControl " . '$EVENT', 1 ); Log3 undef, 2, "[remotecontrol:ONKYO_AVR] Notify created: $nname"; return "Notify created by ONKYO_AVR: $nname"; } sub ONKYO_AVR_RClayout_SVG() { my @row; $row[0] = ":rc_BLANK.svg,:rc_BLANK.svg,power toggle:rc_POWER.svg"; $row[1] = "volume level-up:rc_VOLUP.svg,mute toggle:rc_MUTE.svg,preset up:rc_UP.svg"; $row[2] = "volume level-down:rc_VOLDOWN.svg,sleep:time_timer.svg,preset down:rc_DOWN.svg"; $row[3] = ":rc_BLANK.svg,tuning up:rc_UP.svg,:rc_BLANK.svg"; $row[4] = "left:rc_LEFT.svg,enter:rc_OK.svg,right:rc_RIGHT.svg"; $row[5] = "input usb:rc_USB.svg,tuning down:rc_DOWN.svg,input dlna:rc_MEDIAMENU.svg"; $row[6] = "input tv-cd:rc_TV.svg,input fm:rc_RADIO.svg,input pc:it_pc.svg"; $row[7] = "attr rc_iconpath icons/remotecontrol"; $row[8] = "attr rc_iconprefix black_btn_"; return @row; } sub ONKYO_AVR_RClayout() { my @row; $row[0] = "hdmi-output 01:HDMI_main,hdmi-output 02:HDMI_sub,power toggle:POWEROFF"; $row[1] = "volume level-up:VOLUP,mute toggle:MUTE,preset up:UP"; $row[2] = "volume level-down:VOLDOWN,sleep:SLEEP,preset down:DOWN"; $row[3] = ":blank,tuning up:UP,:blank"; $row[4] = "left:LEFT,enter:OK,right:RIGHT"; $row[5] = "input usb:SOURCE,tuning down:DOWN,input dlna:DLNA"; $row[6] = "input tv-cd:TV,input fm:FMRADIO,input pc:PC"; $row[7] = "attr rc_iconpath icons/remotecontrol"; $row[8] = "attr rc_iconprefix black_btn_"; return @row; } 1; =pod =item device =item summary control for ONKYO AV receivers via network or serial connection =item summary_DE Steuerung von ONKYO AV Receiver per Netzwerk oder seriell =begin html

ONKYO_AVR

    Define
      define <name> ONKYO_AVR <ip-address-or-hostname[:PORT]> [<protocol-version>]
      define <name> ONKYO_AVR <devicename[@baudrate]> [<protocol-version>]

      This module controls ONKYO A/V receivers in real-time via network connection.
      Some newer Pioneer A/V models seem to run ONKYO's ISCP protocol as well and therefore should be fully supported by this module.
      Use ONKYO_AVR_ZONE to control slave zones.

      Instead of IP address or hostname you may set a serial connection format for direct connectivity.


      Example:
        define avr ONKYO_AVR 192.168.0.10

        # With explicit port
        define avr ONKYO_AVR 192.168.0.10:60128

        # With explicit protocol version 2013 and later
        define avr ONKYO_AVR 192.168.0.10 2013

        # With protocol version prior 2013
        define avr ONKYO_AVR 192.168.0.10 pre2013
        # With protocol version prior 2013 and serial connection
        define avr ONKYO_AVR /dev/ttyUSB1@9600 pre2013


    Set
      set <name> <command> [<parameter>]

      Currently, the following commands are defined:
      • channel   -   set active network service (e.g. Spotify)
      • currentTrackPosition   -   seek to specific time for current track
      • input   -   switches between inputs
      • inputDown   -   switches one input down
      • inputUp   -   switches one input up
      • mute on,off   -   controls volume mute
      • muteT   -   toggle mute state
      • next   -   skip track
      • off   -   turns the device in standby mode
      • on   -   powers on the device
      • pause   -   pause current playback
      • play   -   start playback
      • power on,off   -   set power mode
      • preset   -   switches between presets
      • presetDown   -   switches one preset down
      • presetUp   -   switches one preset up
      • previous   -   back to previous track
      • rawCommand Send raw command to device. No space between command and value (eg. TFRB+9 to set bass level to +9)
      • remoteControl Send specific remoteControl command to device
      • repeat off,all,all-folder,one   -   set repeat setting
      • repeatT   -   toggle repeat state
      • shuffle off,on,on-album,on-folder   -   set shuffle setting
      • shuffleT   -   toggle shuffle state
      • sleep 1..90,off   -   sets auto-turnoff after X minutes
      • stop   -   stop current playback
      • toggle   -   switch between on and off
      • volume 0...100   -   set the volume level in percentage
      • volumeUp   -   increases the volume level
      • volumeDown   -   decreases the volume level

        Other set commands may appear dynamically based on previously used "get avr remoteControl"-commands and resulting readings.
        See "get avr remoteControl <Set-name> help" to get more information about possible readings and set values.


    Get
      get <name> <what>

      Currently, the following commands are defined:

      • createZone   -   creates a separate ONKYO_AVR_ZONE device for available zones of the device
      • remoteControl   -   sends advanced remote control commands based on current zone; you may use "get avr remoteControl <Get-command> help" to see details about possible values and resulting readings. In Case the device does not support the command, just nothing happens as normally the device does not send any response. In case the command is temporarily not available you may see according feedback from the log file using attribute verbose=4.
      • statusRequest   -   clears cached settings and re-reads device XML configurations


    Attributes
      • connectionCheck   1..120,off   Pings the device every X seconds to verify connection status. Defaults to 60 seconds.
      • inputs   -   List of inputs, auto-generated after first connection to the device. Inputs may be deleted or re-ordered as required. To rename an input, one needs to put a comma behind the current name and enter the new name.
      • model   -   Contains the model name of the device. Cannot not be changed manually as it is going to be overwritten be the module.
      • volumeSteps   -   When using set commands volumeUp or volumeDown, the volume will be increased or decreased by these steps. Defaults to 1.
      • volumeMax   1...100   When set, any volume higher than this is going to be replaced by this value.
      • wakeupCmd   -   In case the device is unreachable and one is sending set command "on", this FHEM command will be executed before the actual "on" command is sent. E.g. may be used to turn on a switch before the device becomes available via network.


    Generated Readings/Events:
    • audin_* - Shows technical details about current audio input
    • brand - Shows brand name of the device manufacturer
    • channel - Shows current network service name when (e.g. streaming services like Spotify); part of FHEM-4-AV-Devices compatibility
    • currentAlbum - Shows current Album information; part of FHEM-4-AV-Devices compatibility
    • currentArtist - Shows current Artist information; part of FHEM-4-AV-Devices compatibility
    • currentMedia - currently no in use
    • currentTitle - Shows current Title information; part of FHEM-4-AV-Devices compatibility
    • currentTrack* - Shows current track timer information; part of FHEM-4-AV-Devices compatibility
    • deviceid - Shows device name as set in device settings
    • deviceyear - Shows model device year
    • firmwareversion - Shows current firmware version
    • input - Shows currently used input; part of FHEM-4-AV-Devices compatibility
    • mute - Reports the mute status of the device (can be "on" or "off")
    • playStatus - Shows current network service playback status; part of FHEM-4-AV-Devices compatibility
    • power - Reports the power status of the device (can be "on" or "off")
    • presence - Reports the presence status of the receiver (can be "absent" or "present"). In case of an absent device, control is not possible.
    • repeat - Shows current network service repeat status; part of FHEM-4-AV-Devices compatibility
    • screen* - Experimental: Gives some information about text that is being shown via on-screen menu
    • shuffle - Shows current network service shuffle status; part of FHEM-4-AV-Devices compatibility
    • sleep - Reports current sleep state (can be "off" or shows timer in minutes)
    • state - Reports current network connection status to the device
    • stateAV - Zone status from user perspective combining readings presence, power, mute and playStatus to a useful overall status.
    • volume - Reports current volume level of the receiver in percentage values (between 0 and 100 %)
    • vidin_* - Shows technical details about current video input before image processing
    • vidout_* - Shows technical details about current video output after image processing
    • zones - Shows total available zones of device

    Using remoteControl get-command might result in creating new readings in case the device sends any data.
=end html =begin html_DE

ONKYO_AVR

    Eine deutsche Version der Dokumentation ist derzeit nicht vorhanden. Die englische Version ist hier zu finden:
=end html_DE =cut