using HslCommunication.BasicFramework;
using HslCommunication.Core;
using HslCommunication.Core.IMessage;
using HslCommunication.Core.Net;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace HslCommunication.Robot.KUKA
{
///
/// Kuka机器人的数据交互对象,通讯支持的条件为KUKA 的 KRC4 控制器中运行KUKAVARPROXY 这个第三方软件,端口通常为7000
///
///
/// 非常感谢 昆山-LT 网友的测试和意见反馈。
///
public class KukaAvarProxyNet : NetworkDoubleBase, IRobotNet
{
#region Constructor
///
/// 实例化一个默认的对象
///
public KukaAvarProxyNet( )
{
softIncrementCount = new SoftIncrementCount( ushort.MaxValue );
}
///
/// 实例化一个默认的Kuka机器人对象,并指定IP地址和端口号,端口号通常为7000
///
/// Ip地址
/// 端口号
public KukaAvarProxyNet( string ipAddress, int port )
{
IpAddress = ipAddress;
Port = port;
softIncrementCount = new SoftIncrementCount( ushort.MaxValue );
}
#endregion
#region Read Write Support
///
/// 读取埃夫特机器人的原始的字节数据信息,该地址参数是没有任何作用的,随便填什么
///
/// 无效参数
/// 带有成功标识的byte[]数组
public OperateResult Read( string address )
{
OperateResult read = ReadFromCoreServer( PackCommand( BuildReadValueCommand( address ) ) );
if (!read.IsSuccess) return read;
return ExtractActualData( read.Content );
}
///
/// 读取机器人的所有的数据信息,返回JSON格式的数据对象,地址参数无效
///
/// 地址信息
/// 带有成功标识的字符串数据
public OperateResult ReadString( string address )
{
OperateResult read = Read( address );
if (!read.IsSuccess) return OperateResult.CreateFailedResult( read );
return OperateResult.CreateSuccessResult( Encoding.Default.GetString( read.Content ) );
}
///
/// 本机器人不支持该方法操作,将永远返回失败,无效的操作
///
/// 指定的地址信息,有些机器人可能不支持
/// 原始的字节数据信息
/// 是否成功的写入
public OperateResult Write( string address, byte[] value )
{
return Write( address, Encoding.Default.GetString( value ) );
}
///
/// 本机器人支持该方法操作,根据实际的值记性返回
///
/// 指定的地址信息,有些机器人可能不支持
/// 字符串的数据信息
/// 是否成功的写入
public OperateResult Write( string address, string value )
{
OperateResult read = ReadFromCoreServer( PackCommand( BuildWriteValueCommand( address, value ) ) );
if (!read.IsSuccess) return read;
return ExtractActualData( read.Content );
}
#endregion
#region Command Build
///
/// 将核心的指令打包成一个可用于发送的消息对象
///
/// 核心命令
/// 最终实现的可以发送的机器人的字节数据
private byte[] PackCommand( byte[] commandCore )
{
byte[] buffer = new byte[commandCore.Length + 4];
ByteTransform.TransByte( (ushort)softIncrementCount.GetCurrentValue( ) ).CopyTo( buffer, 0 );
ByteTransform.TransByte( (ushort)commandCore.Length ).CopyTo( buffer, 2 );
commandCore.CopyTo( buffer, 4 );
return buffer;
}
private OperateResult ExtractActualData(byte[] response )
{
try
{
if(response[response.Length - 1] != 0x01) return new OperateResult( response[response.Length - 1], "Wrong: " + SoftBasic.ByteToHexString( response, ' ' ) );
int length = response[5] * 256 + response[6];
byte[] buffer = new byte[length];
Array.Copy( response, 7, buffer, 0, length );
return OperateResult.CreateSuccessResult( buffer );
}
catch(Exception ex)
{
return new OperateResult( "Wrong:" + ex.Message + " Code:" + SoftBasic.ByteToHexString( response, ' ' ) );
}
}
private byte[] BuildCommands(byte function, string[] commands )
{
List buffer = new List( );
buffer.Add( function );
for (int i = 0; i < commands.Length; i++)
{
byte[] buffer_command = Encoding.Default.GetBytes( commands[i] );
buffer.AddRange( ByteTransform.TransByte( (ushort)buffer_command.Length ) );
buffer.AddRange( buffer_command );
}
return buffer.ToArray( );
}
private byte[] BuildReadValueCommand( string address )
{
return BuildCommands( 0x00, new string[] { address } );
}
private byte[] BuildWriteValueCommand( string address, string value )
{
return BuildCommands( 0x01, new string[] { address, value } );
}
#endregion
#region Private Member
private SoftIncrementCount softIncrementCount; // 自增消息的对象
#endregion
#region Object Override
///
/// 返回表示当前对象的字符串
///
/// 字符串
public override string ToString( )
{
return $"KukaAvarProxyNet Robot[{IpAddress}:{Port}]";
}
#endregion
}
}