fhem-mirror/FHEM/42_Nextion.pm
2017-05-25 16:12:04 +00:00

965 lines
29 KiB
Perl

##############################################################################
#
# 42_Nextion.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 <http://www.gnu.org/licenses/>.
#
##############################################################################
#
# 42_Nextion (c) Johannes Viegener / https://github.com/viegener/Telegram-fhem
#
# $Id$
#
##############################################################################
# 0.0 2016-03-23 Started
# Inital Version to communicate with Nextion via transparent bridge send raw commands and get returns as readings
# Fix for result without error
# multiCommandSend (allows also set logic)
# 0.2 2016-03-29 Basic capabilities
#
# SendAndWaitforAnswer
# disconnect with state modification
# put error/cmdresult in reading on send command
# text readings for messages starting with $ and specific codes
# initPageX attributes and execution when page is entered with replaceSetMagic
# Init commands - Attribute initCommands
# init commands will also be sent on reconnect
# currentPage reading will only be maintained if attribut hasSendMe is set
# 0.3 2016-04-03 init commands and test with notifys
#
# Convert iso to/from utf-8 on messages from nextion
# ReplaceSetMagic called once per command in initPage due to issue with fhem-pl change
# Initial documentation completed
# added new set commands: page and pageCmd
# 0.4 2016-04-24 documentation / page and pageCmds
#
# expectAnswer can be set to ignore any commands
# reduce log level for normal operation
# fix disconnect
# 0.5 2016-06-30 disconnect-fix / log reduced / expectAnswer
#
# 0.6 2016-10-29 available through SVN
# timeout on sec 1 (not 0.5) - msg554933
# disabled attribute and change in connections - msg554933
#
# 0.8 2016-03-01 revert changes
# fix for page 10 not recognized : #msg592948
# _connect/Disconnect/isConnected subs
# init device after notify on initialized
# fix connection - to work also if nextion is unavailable
#
#
# Extended log for read function
# remove leading ff
# fault tolerant command reader allow empty commands and missing \xff
# print filtered messages - with quoted chars
# changed log levels to 4 for verbose / 5 will print all messages
# fix replacesetmagic to ensure device hash is given
# 2016-05-25 fault tolerance in reader / fixes
#
#
#
##############################################
##############################################
### TODO
#
#
# rectextold1-5 --> #msg611695
# timeout with checkalive check?
#
# react on events with commands allowing values from FHEM
# remove wait for answer by attribute
# number of pages as define (std max 0-9)
#
##############################################
##############################################
### Considerations for the Nextion UI
#
# - sendme is needed on pages otherwise initCmds can not be used
# - to react on button usages $ cmds are introduced
# add in postinitialize event something like
# print "$b0="
# get 1
# or for double state buttons / switches
# print "$bt0="
# print bt0.val#
# will result in $b0=1 / $bt0=0 or $bt0=1 being sent to fhem
# - Recommended use bkcmd=3 in pre initialize event of page 0 (to not to have to wait on timeouts for commands / otherwise fhem is blocked)
#
#
#
##############################################
##############################################
##############################################
package main;
use strict;
use warnings;
use Time::HiRes qw(gettimeofday);
use Encode qw( decode encode );
use Data::Dumper;
#########################
# Forward declaration
sub Nextion_Read($@);
sub Nextion_Write($$$);
sub Nextion_ReadAnswer($$);
sub Nextion_Ready($);
#########################
# Globals
my %Nextion_errCodes = (
"\x00" => "Invalid instruction",
"\x03" => "Page ID invalid",
"\x04" => "Picture ID invalid",
"\x05" => "Font ID invalid",
"\x11" => "Baud rate setting invalid",
"\x12" => "Curve control ID or channel number invalid",
"\x1a" => "Variable name invalid",
"\x1b" => "Variable operation invalid",
"\x01" => "Success"
);
##############################################################################
##############################################################################
##
## Module operation
##
##############################################################################
##############################################################################
sub
Nextion_Initialize($)
{
my ($hash) = @_;
require "$attr{global}{modpath}/FHEM/DevIo.pm";
# Provider
$hash->{ReadFn} = "Nextion_Read";
$hash->{WriteFn} = "Nextion_Write";
$hash->{ReadyFn} = "Nextion_Ready";
$hash->{UndefFn} = "Nextion_Undef";
$hash->{ShutdownFn} = "Nextion_Undef";
$hash->{ReadAnswerFn} = "Nextion_ReadAnswer";
$hash->{NotifyFn} = "Nextion_Notify";
$hash->{AttrFn} = "Nextion_Attr";
$hash->{AttrList} = "initPage0:textField-long initPage1:textField-long initPage2:textField-long initPage3:textField-long initPage4:textField-long ".
"initPage5:textField-long initPage6:textField-long initPage7:textField-long initPage8:textField-long initPage9:textField-long ".
"initCommands:textField-long hasSendMe:0,1 expectAnswer:1,0 disable:0,1 ".$readingFnAttributes;
# timeout for connections - msg554933
$hash->{TIMEOUT} = 1; # might be better? 0.5;
# Normal devices
$hash->{DefFn} = "Nextion_Define";
$hash->{SetFn} = "Nextion_Set";
}
#####################################
sub
Nextion_Define($$)
{
my ($hash, $def) = @_;
my @a = split("[ \t][ \t]*", $def);
if(@a != 3) {
return "wrong syntax: define <name> Nextion hostname:23";
}
my $name = $a[0];
my $dev = $a[2];
$hash->{Clients} = ":NEXTION:";
my %matchList = ( "1:NEXTION" => ".*" );
$hash->{MatchList} = \%matchList;
Nextion_Disconnect($hash);
$hash->{DeviceName} = $dev;
return undef if($dev eq "none"); # DEBUGGING
my $ret;
if( $init_done ) {
Nextion_Disconnect($hash);
$ret = Nextion_Connect($hash);
} elsif( $hash->{STATE} ne "???" ) {
$hash->{STATE} = "Initialized";
}
return $ret;
}
#####################################
sub
Nextion_Set($@)
{
my ($hash, @a) = @_;
my $name = shift @a;
my %sets = ("cmd"=>"textField", "raw"=>"textField", "reopen"=>undef, "disconnect"=>undef
, "pageCmd"=>"textField-long", "page"=>"0,1,2,3,4,5,6,7,8,9" );
my $numberOfArgs = int(@a);
return "set $name needs at least one parameter" if($numberOfArgs < 1);
my $type = shift @a;
$numberOfArgs--;
my $ret = undef;
return "Unknown argument $type, choose one of " . join(" ", sort keys %sets) if (!exists($sets{$type}));
if( ($type eq "raw") || ($type eq "cmd") ) {
my $cmd = join(" ", @a );
$ret = Nextion_SendCommand($hash,$cmd, 1);
} elsif($type eq "page") {
if ( $numberOfArgs < 1 ) {
$ret = "No page number given";
} elsif ( $numberOfArgs > 1 ) {
$ret = "Too many parameters (only page number shoudl be provided)";
} elsif ( $a[0] !~ /^[0-9]$/ ) {
$ret = "Page number needs to be a single digit";
} else {
$ret = Nextion_SendCommand($hash,"page ".$a[0], 1);
}
} elsif($type eq "pageCmd") {
if ( $numberOfArgs < 2 ) {
$ret = "No page number(s) or no commands given";
} elsif ( $a[0] !~ /^[0-9](,[0-9])*$/ ) {
$ret = "Page numbers needs to be single digits separated with ,";
} elsif ( ! AttrVal($name,"hasSendMe",0) ) {
$ret = "Attribute hasSendMe not set (no actual page)";
} else {
my @pages = split( /,/, shift @a);
my $cpage = ReadingsVal($name,"currentPage",-1);
my $cmd = join(" ", @a );
foreach my $aPage ( @pages ) {
if ( $aPage == $cpage ) {
$ret = Nextion_SendCommand($hash,$cmd, 1);
last;
}
}
}
} elsif($type eq "reopen") {
Nextion_Disconnect($hash);
delete $hash->{DevIoJustClosed} if($hash->{DevIoJustClosed});
delete($hash->{NEXT_OPEN}); # needed ? - can this ever occur
return Nextion_Connect( $hash, 1 );
} elsif($type eq "disconnect") {
Nextion_Disconnect($hash);
DevIo_setStates($hash, "disconnected");
# DevIo_Disconnected($hash);
# delete $hash->{DevIoJustClosed} if($hash->{DevIoJustClosed});
}
if ( ! defined( $ret ) ) {
Log3 $name, 4, "Nextion_Set $name: $type done succesful: ";
} else {
Log3 $name, 1, "Nextion_Set $name: $type failed with :$ret: ";
}
return $ret;
}
##############################
# attr function for setting fhem attributes for the device
sub Nextion_Attr(@) {
my ($cmd,$name,$aName,$aVal) = @_;
my $hash = $defs{$name};
Log3 $name, 4, "Nextion_Attr $name: called ";
return "\"Nextion_Attr: \" $name does not exist" if (!defined($hash));
if (defined($aVal)) {
Log3 $name, 4, "Nextion_Attr $name: $cmd on $aName to $aVal";
} else {
Log3 $name, 4, "Nextion_Attr $name: $cmd on $aName to <undef>";
}
# $cmd can be "del" or "set"
# $name is device name
# aName and aVal are Attribute name and value
if ($cmd eq "set") {
if ($aName eq 'hasSendMe') {
$aVal = ($aVal eq "1")? "1": "0";
readingsSingleUpdate($hash, "currentPage", -1, 1);
} elsif ($aName eq 'unsupported') {
if ( $aVal !~ /^[[:digit:]]+$/ ) {
return "\"Nextion_Attr: \" unsupported";
}
} elsif ($aName eq 'disable') {
if($aVal eq "1") {
Nextion_Disconnect($hash);
DevIo_setStates($hash, "disabled");
} else {
if($hash->{READINGS}{state}{VAL} eq "disabled") {
DevIo_setStates($hash, "disconnected");
InternalTimer(gettimeofday()+1, "Nextion_Connect", $hash, 0);
}
}
}
$_[3] = $aVal;
}
return undef;
}
######################################
sub Nextion_IsConnected($)
{
my $hash = shift;
# stacktrace();
# Debug "Name : ".$hash->{NAME};
# Debug "FD: ".((exists($hash->{FD}))?"def":"undef");
# Debug "TCPDev: ".((defined($hash->{TCPDev}))?"def":"undef");
return 0 if(!exists($hash->{FD}));
if(!defined($hash->{TCPDev})) {
Nextion_Disconnect($_[0]);
return 0;
}
return 1;
}
######################################
sub Nextion_Disconnect($)
{
my $hash = shift;
my $name = $hash->{NAME};
Log3 $name, 4, "Nextion_Disconnect: $name";
DevIo_CloseDev($hash);
}
######################################
sub Nextion_Connect($;$) {
my ($hash, $mode) = @_;
my $name = $hash->{NAME};
my $ret;
$mode = 0 if!($mode);
return undef if(Nextion_IsConnected($hash));
# Debug "NEXT_OPEN: $name".((defined($hash->{NEXT_OPEN}))?time()-$hash->{NEXT_OPEN}:"undef");
if(!IsDisabled($name)) {
# undefined means timeout / 0 means failed / 1 means ok
if ( DevIo_OpenDev($hash, $mode, "Nextion_DoInit") ) {
if(!Nextion_IsConnected($hash)) {
$ret = "Nextion_Connect: Could not connect :".$name;
Log3 $hash, 2, $ret;
}
}
}
return $ret;
}
#####################################
sub
Nextion_Notify($$)
{
my ($hash,$dev) = @_;
my $name = $hash->{NAME};
my $type = $hash->{TYPE};
return if($dev->{NAME} ne "global");
return if(!grep(m/^INITIALIZED|REREADCFG$/, @{$dev->{CHANGED}}));
if( IsDisabled($name) > 0 ) {
readingsSingleUpdate($hash, 'state', 'disabled', 1 ) if( ReadingsVal($name,'state','' ) ne 'disabled' );
return undef;
}
Nextion_Connect($hash);
return undef;
}
#####################################
sub
Nextion_DoInit($)
{
my $hash = shift;
my $name = $hash->{NAME};
my $ret = undef;
### send init commands
my $initCmds = AttrVal( $name, "initCommands", undef );
Log3 $name, 3, "Nextion_DoInit $name: Execute initCommands :".(defined($initCmds)?$initCmds:"<undef>").":";
## ??? quick hack send on init always page 0 twice to ensure proper start
# Send command handles replaceSetMagic and splitting
$ret = Nextion_SendCommand( $hash, "page 0;page 0", 0 );
# Send command handles replaceSetMagic and splitting
$ret = Nextion_SendCommand( $hash, $initCmds, 0 ) if ( defined( $initCmds ) );
return $ret;
}
#####################################
sub
Nextion_Undef($@)
{
my ($hash, $arg) = @_;
### ??? send finish commands
Nextion_Disconnect($hash);
return undef;
}
#####################################
sub
Nextion_Write($$$)
{
my ($hash,$fn,$msg) = @_;
$msg = sprintf("%s03%04x%s%s", $fn, length($msg)/2+8,
$hash->{HANDLE} ? $hash->{HANDLE} : "00000000", $msg);
DevIo_SimpleWrite($hash, $msg, 1);
}
#####################################
sub
Nextion_SendCommand($$$)
{
my ($hash,$msg,$answer) = @_;
my $name = $hash->{NAME};
my @ret;
Log3 $name, 4, "Nextion_SendCommand $name: send commands :".$msg.": ";
# First replace any magics
# my %dummy;
# my ($err, @a) = ReplaceSetMagic(\%dummy, 0, ( $msg ) );
# if ( $err ) {
# Log3 $name, 1, "$name: Nextion_SendCommand failed on ReplaceSetmagic with :$err: on commands :$msg:";
# } else {
# $msg = join(" ", @a);
# Log3 $name, 4, "$name: Nextion_SendCommand ReplaceSetmagic commnds after :".$msg.":";
# }
# Split commands into separate elements at single semicolons (escape double ;; before)
$msg =~ s/;;/SeMiCoLoN/g;
my @msgList = split(";", $msg);
my $singleMsg;
while(defined($singleMsg = shift @msgList)) {
$singleMsg =~ s/SeMiCoLoN/;/g;
my ($err, @a) = ReplaceSetMagic($hash, 0, ( $singleMsg ) );
if ( $err ) {
Log3 $name, 1, "$name: Nextion_SendCommand failed on ReplaceSetmagic with :$err: on commands :$singleMsg:";
} else {
$singleMsg = join(" ", @a);
Log3 $name, 4, "$name: Nextion_SendCommand ReplaceSetmagic commnds after :".$singleMsg.":";
}
my $lret = Nextion_SendSingleCommand($hash, $singleMsg, $answer);
push(@ret, $lret) if(defined($lret));
}
return join("\n", @ret) if(@ret);
return undef;
}
#####################################
sub
Nextion_SendSingleCommand($$$)
{
my ($hash,$msg,$answer) = @_;
my $name = $hash->{NAME};
$answer = 0 if ( ! AttrVal($name,"expectAnswer",0) );
# ??? handle answer
my $err;
# trim the msg
$msg =~ s/^\s+|\s+$//g;
Log3 $name, 4, "Nextion_SendCommand $name: send command :".$msg.": ";
my $isoMsg = Nextion_EncodeToIso($msg);
DevIo_SimpleWrite($hash, $isoMsg."\xff\xff\xff", 0);
$err = Nextion_ReadAnswer($hash, $isoMsg) if ( $answer );
Log3 $name, 1, "Nextion_SendCommand Error :".$err.": " if ( defined($err) );
Log3 $name, 4, "Nextion_SendCommand Success " if ( ! defined($err) );
# Also set sentMsg Id and result in Readings
readingsBeginUpdate($hash);
readingsBulkUpdate($hash, "cmdSent", $msg);
readingsBulkUpdate($hash, "cmdResult", (( defined($err))?$err:"empty") );
readingsEndUpdate($hash, 1);
return $err;
}
#####################################
# called from the global loop, when the select for hash->{FD} reports data
sub
Nextion_Read($@)
{
my ($hash, $local, $isCmd) = @_;
my $buf = ($local ? $local : DevIo_SimpleRead($hash));
return "" if(!defined($buf));
my $name = $hash->{NAME};
### $buf = unpack('H*', $buf);
my $data = ($hash->{PARTIAL} ? $hash->{PARTIAL} : "");
# drop old data
if($data) {
$data = "" if(gettimeofday() - $hash->{READ_TS} > 5);
delete($hash->{READ_TS});
}
Log3 $name, 5, "Nextion/RAW: $data/$buf";
$data .= $buf;
my $ret;
my $newPageId;
while(length($data) > 0) {
# if ( $data =~ /^([^\xff]*)\xff\xff\xff(.*)$/ ) {
if ( $data =~ /^([^\xff]*)(\xff+)([^\xff].*)?$/ ) {
my $rcvd = $1;
my $ffpart = $2;
$data = $3;
$data = "" if ( ! defined($data) );
if ( length($ffpart) != 3 ) {
Log3 $name, 4, "Nextion/RAW: shortened ffh end sequence (".length($ffpart).") ".Data::Dumper::qquote($rcvd) ;
} else {
Log3 $name, 5, "Nextion/RAW: message found ".Data::Dumper::qquote($rcvd) ;
}
if ( length($rcvd) > 0 ) {
my ( $msg, $text, $val, $id ) = Nextion_convertMsg($rcvd);
if ( defined( $id ) ) {
if ( $id =~ /^[0-9]+$/ ) {
$newPageId = $id;
}
}
Log3 $name, 4, "Nextion: Received message :$msg:";
if ( defined( ReadingsVal($name,"received",undef) ) ) {
if ( defined( ReadingsVal($name,"old1",undef) ) ) {
if ( defined( ReadingsVal($name,"old2",undef) ) ) {
if ( defined( ReadingsVal($name,"old3",undef) ) ) {
if ( defined( ReadingsVal($name,"old4",undef) ) ) {
$hash->{READINGS}{old5}{VAL} = $hash->{READINGS}{old4}{VAL};
$hash->{READINGS}{old5}{TIME} = $hash->{READINGS}{old4}{TIME};
}
$hash->{READINGS}{old4}{VAL} = $hash->{READINGS}{old3}{VAL};
$hash->{READINGS}{old4}{TIME} = $hash->{READINGS}{old3}{TIME};
}
$hash->{READINGS}{old3}{VAL} = $hash->{READINGS}{old2}{VAL};
$hash->{READINGS}{old3}{TIME} = $hash->{READINGS}{old2}{TIME};
}
$hash->{READINGS}{old2}{VAL} = $hash->{READINGS}{old1}{VAL};
$hash->{READINGS}{old2}{TIME} = $hash->{READINGS}{old1}{TIME};
}
$hash->{READINGS}{old1}{VAL} = $hash->{READINGS}{received}{VAL};
$hash->{READINGS}{old1}{TIME} = $hash->{READINGS}{received}{TIME};
}
readingsBeginUpdate($hash);
readingsBulkUpdate($hash,"received",$msg);
readingsBulkUpdate($hash,"rectext",( (defined($text)) ? $text : "" ));
readingsBulkUpdate($hash,"currentPage",$id) if ( ( defined( $id ) ) && ( AttrVal($name,"hasSendMe",0) ) );
readingsEndUpdate($hash, 1);
} else {
Log3 $name, 5, "Nextion/RAW: match with zero length - command missing - ffh #".length($ffpart);
}
} else {
# not matching
# if ( $data =~ /^\xff+([^\xff].*)/ ) {
# Log3 $name, 5, "Nextion/RAW: remove leading ff ";
# $data = $1;
# } elsif ( $data =~ /^[^\xff]*(\xff+)/ ) {
# Log3 $name, 5, "Nextion/RAW: not matching commands but contains ff :".length($1).":";
# } else {
# Log3 $name, 5, "Nextion/RAW: not matching commands no ff";
# }
last;
}
}
$hash->{PARTIAL} = $data;
$hash->{READ_TS} = gettimeofday() if($data);
# initialize last page id found:
if ( defined( $newPageId ) ) {
$newPageId = $newPageId + 0;
my $initCmds = AttrVal( $name, "initPage".sprintf("%d",$newPageId), undef );
Log3 $name, 4, "Nextion_InitPage $name: page :".$newPageId.": with commands :".(defined($initCmds)?$initCmds:"<undef>").":";
return if ( ! defined( $initCmds ) );
# Send command handles replaceSetMagic and splitting
Nextion_SendCommand( $hash, $initCmds, 0 );
}
return $ret if(defined($local));
return undef;
}
#####################################
# This is a direct read for command results
sub
Nextion_ReadAnswer($$)
{
my ($hash, $arg) = @_;
my $name = $hash->{NAME};
Log3 $name, 4, "Nextion_ReadAnswer $name: for send commands :".$arg.": ";
return "No FD (dummy device?)" if(!$hash || ($^O !~ /Win/ && !defined($hash->{FD})));
my $data = "";
for(;;) {
return "Device lost when reading answer for get $arg" if(!$hash->{FD});
my $rin = '';
vec($rin, $hash->{FD}, 1) = 1;
my $nfound = select($rin, undef, undef, 1);
if($nfound < 0) {
next if ($! == EAGAIN() || $! == EINTR() || $! == 0);
my $err = $!;
DevIo_Disconnected($hash);
return"Nextion_ReadAnswer $arg: $err";
}
return "Timeout reading answer for get $arg" if($nfound == 0);
my $buf = DevIo_SimpleRead($hash);
return "No data" if(!defined($buf));
my $ret;
my $data .= $buf;
# not yet long enough select again
# next if ( length($data) < 4 );
# TODO: might have to check for remaining data in buffer?
if ( $data =~ /^\xff*([^\xff])\xff\xff\xff(.*)$/ ) {
my $rcvd = $1;
$data = $2;
$ret = $Nextion_errCodes{$rcvd};
$ret = "Nextion: Unknown error with code ".sprintf( "H%2.2x", $rcvd ) if ( ! defined( $ret ) );
} elsif ( length($data) == 0 ) {
$ret = "No answer";
} else {
$ret = "Message received";
}
# read rest of buffer direct in read function
if ( length($data) > 0 ) {
Nextion_Read($hash, $data);
}
return (($ret eq $Nextion_errCodes{"\x01"}) ? undef : $ret);
}
}
#####################################
sub
Nextion_Ready($)
{
my ($hash) = @_;
# Debug "Name : ".$hash->{NAME};
# stacktrace();
return Nextion_Connect( $hash, 1 ) if($hash->{STATE} eq "disconnected");
return 0;
}
##############################################################################
##############################################################################
##
## Helper
##
##############################################################################
##############################################################################
#####################################
# returns
# msg in Hex converted format
# text equivalent of message if applicable
# val in message either numeric or text
# id of a control <pageid>:<controlid>:... or just a page <pageid> or undef
sub
Nextion_convertMsg($)
{
my ($raw) = @_;
my $msg = "";
my $text;
my $val;
my $id;
my $rcvd = $raw;
while(length($rcvd) > 0) {
my $char = ord($rcvd);
$rcvd = substr($rcvd,1);
$msg .= " " if ( length($msg) > 0 );
$msg .= sprintf( "H%2.2x", $char );
$msg .= "(".chr($char).")" if ( ( $char >= 32 ) && ( $char <= 127 ) ) ;
}
if ( $raw =~ /^(\$.*=)(\x71?)(.*)$/s ) {
# raw msg with $ start is manually defined message standard
# sent on release event
# print "$bt0="
# get bt0.val OR print bt0.val
#
$text = $1;
my $rest = $3;
if ( length($rest) == 4 ) {
$val = ord($rest);
$rest = substr($rest,1);
$val += ord($rest)*256;
$rest = substr($rest,1);
$val += ord($rest)*256*256;
$rest = substr($rest,1);
$val += ord($rest)*256*256*256;
$text .= sprintf("%d",$val);
} else {
$text .= $rest;
$val = $rest;
}
} elsif ( $raw =~ /^\x70(.*)$/s ) {
# string return
$val = $1;
# Log3 undef, 1, "Nextion_convertMsg String message val :".$val.": ";
$text = "string \"" . $val . "\"";
} elsif ( $raw =~ /^\x71(.*)$/ ) {
# numeric return
$text = "num ";
my $rest = $1;
if ( length($rest) == 4 ) {
$val = ord($rest);
$rest = substr($rest,1);
$val += ord($rest)*256;
$rest = substr($rest,1);
$val += ord($rest)*256*256;
$rest = substr($rest,1);
$val += ord($rest)*256*256*256;
$text .= sprintf("%d",$val);
} else {
$text .= $rest;
$val = $rest;
}
} elsif ( $raw =~ /^\x66(.)$/s ) {
# need to parse multiline due to issue with page 10 --> x0A
# page started
$text = "page ";
my $rest = $1;
$id = ord($rest);
$text .= sprintf("%d",$id);
}
$text = Nextion_DecodeFromIso( $text );
$msg = Nextion_DecodeFromIso( $msg );
return ( $msg, $text, $val, $id );
}
#####################################
sub
Nextion_EncodeToIso($)
{
my ($orgmsg) = @_;
# encode in ISO8859-1 from UTF8
# decode to Perl's internal format
my $msg = decode( 'utf-8', $orgmsg );
# encode to iso-8859-1
$msg = encode( 'iso-8859-1', $msg );
return $msg;
}
#####################################
sub
Nextion_DecodeFromIso($)
{
my ($orgmsg) = @_;
# encode in ISO8859-1 from UTF8
# decode to Perl's internal format
my $msg = decode( 'iso-8859-1', $orgmsg );
# encode to iso-8859-1
$msg = encode( 'utf-8', $msg );
return $msg;
}
##################################################################################################################
##################################################################################################################
##################################################################################################################
1;
=pod
=item summary interact with Nextion touch displays
=item summary_DE interagiert mit Nextion Touchscreens
=begin html
<a name="Nextion"></a>
<h3>Nextion</h3>
<ul>
This module connects remotely to a Nextion display that is connected through a ESP8266 or similar serial to network connection
<a href="http://wiki.iteadstudio.com/Nextion_HMI_Solution">Nextion</a> devices are relatively inexpensive tft touch displays, that include also a controller that can hold a user interface and communicates via serial protocol to the outside world.
<br>
A description of the Hardwarelayout for connecting the ESP8266 module and the Nextion Dispaly is in the correspdong forum thread <a href="https://forum.fhem.de/index.php/topic,51267.0.html">https://forum.fhem.de/index.php/topic,51267.0.html</a>.
<br><br>
<a name="Nextiondefine"></a>
<b>Define</b>
<ul>
<code>define &lt;name&gt; Nextion &lt;hostname/ip&gt;:23 </code>
<br><br>
Defines a Nextion device on the given hostname / ip and port (should be port 23/telnetport normally)
<br><br>
Example: <code>define nxt Nextion 10.0.0.1:23</code><br>
<br>
</ul>
<br><br>
<a name="Nextionset"></a>
<b>Set</b>
<ul>
<code>set &lt;name&gt; &lt;what&gt; [&lt;value&gt;]</code>
<br><br>
where &lt;what&gt; / &lt;value&gt; is one of
<br><br>
<li><code>raw &lt;nextion command&gt;</code><br>Sends the given raw message to the nextion display. The supported commands are described with the Nextion displays: <a href="http://wiki.iteadstudio.com/Nextion_Instruction_Set">http://wiki.iteadstudio.com/Nextion_Instruction_Set</a>
<br>
Examples:<br>
<dl>
<dt><code>set nxt raw page 0</code></dt>
<dd> switch the display to page 0 <br> </dd>
<dt><code>set nxt raw b0.txt</code></dt>
<dd> get the text for button 0 <br> </dd>
<dl>
</li>
<li><code>cmd &lt;nextion command&gt;</code><br>same as raw
</li>
<li><code>page &lt;0 - 9&gt;</code><br>set the page number given as new page on the nextion display.
</li>
<li><code>pageCmd &lt;one or multiple page numbers separated by ,&gt; &lt;cmds&gt;</code><br>Execute the given commands if the current page on the screen is in the list given as page number.
</li>
</ul>
<br><br>
<a name="Nextionattr"></a>
<b>Attributes</b>
<br><br>
<ul>
<li><code>hasSendMe &lt;0 or 1&gt;</code><br>Specify if the display definition on the Nextion display is using the "send me" checkbox to send current page on page changes. This will then change the reading currentPage accordingly
</li>
<li><code>initCommands &lt;series of commands&gt;</code><br>Display will be initialized with these commands when the connection to the device is established (or reconnected). Set logic for executing perl or getting readings can be used. Multiple commands will be separated by ;<br>
Example<br>
&nbsp;&nbsp;<code>t1.txt="Hallo";p1.val=1;</code>
</li>
<li><code>initPage1 &lt;series of commands&gt;</code> to <code>initPage9 &lt;series of commands&gt;</code><br>When the corresponding page number will be displayed the given commands will be sent to the display. See also initCommands.<br>
Example<br>
&nbsp;&nbsp;<code>t1.txt="Hallo";p1.val=1;</code>
</li>
<li><code>expectAnswer &lt;1 or 0&gt;</code><br>Specify if an answer from display is expected. If set to zero no answer is expected at any time on a command.
</li>
</ul>
<br><br>
<a name="Nextionreadings"></a>
<b>Readings</b>
<ul>
<li><code>received &lt;Hex values of the last received message from the display&gt;</code><br> The message is converted in hex values (old messages are stored in the readings old1 ... old5). Example for a message is <code>H65(e) H00 H04 H00</code> </li>
<li><code>rectext &lt;text or empty&gt;</code><br> Translating the received message into text form if possible. Beside predefined data that is sent from the display on specific changes, custom values can be sent in the form <code>$name=value</code>. This can be sent by statements in the Nextion display event code <br>
<code>print "$bt0="<br>
get bt0.val</code>
</li>
<li><code>currentPage &lt;page number on the display&gt;</code><br> Shows the number of the UI screen as configured on the Nextion display that is currently shown.<br>This is only valid if the attribute <code>hasSendMe</code> is set to 1 and used also in the display definition of the Nextion.</li>
<li><code>cmdSent &lt;cmd&gt;</code><br> Last cmd sent to the Nextion Display </li>
<li><code>cmdResult &lt;result text&gt;</code><br> Result of the last cmd sent to the display (or empty)</li>
</ul>
<br><br>
</ul>
=end html
=cut