QT中的OpcDa 客戶端 實現

前段時間開發Windows下的設備端軟件接觸到了OPC DA,雖然如今有了更強大的OPC UA,可是爲了兼容一些老的設備,不得不硬着頭皮去啃這個老掉牙的已通過時了的技術。原本只是想粗略瞭解一下,簡單寫一個應付了事,誰知道過程一波三折。ios

因爲OPCDA採用COM技術,而我又不想再去學COM編程以及VC、AFX、MFC,我採起調用 WTClient 的懶人方法,而且只花了幾個小時就整了一個出來,簡單測試經過,結果經同事驗證,Float、long等類型統統沒法讀取沒法寫入,而且每隔半小時左右便與Kepsrever斷開鏈接。而後處處找解決方案,最後在官方網站獲得答案:免費版本有30分鐘時間限制,數據類型估計也順手給限了,只有BOOL和WORD少數幾個類型可使用,這期間又是浪費好多時間。git

通過這番教訓,決定用QT從頭開始寫一個本身的庫,因而有了如下內容,提供給須要的人,稍候會傳到github(早已傳上去了,可是忘記修改本文,須要的去拿吧:https://github.com/cncap/qt-opcda)。github

參考:OPCDA服務器與客戶程序開發指南修訂版編程

#ifndef OPCCLIENT_H
#define OPCCLIENT_H
#include <iostream>
#include "opcerror.h"
#include "opccomn.h"
#include "OpcEnum.h"
#include "Opcda.h"
#include "copctransaction.h"
#include "wldef.h"
#include <QObject>
#include <QDebug>
#include "windows.h"
/**
 * Copyright 2016 Chn.Captain (chn.captain@gmail.com)
 */
class COPCDataCallback;
class ItemDef
{
public:
  QString    name;
  DWORD         accessRights;
  VARIANT    value;
  WORD        quality;
  OPCHANDLE    hServerHandle;
  OPCHANDLE    hClientHandle;
  FILETIME    time;
  VARTYPE    type;
  ItemDef()
  {
    type = VT_EMPTY;
    quality = 0;
    hServerHandle = 0;
    hClientHandle = 0;

    //ZeroMemory(&time, sizeof(time));
  }
  ItemDef( const ItemDef& it )
  {
    name =        it.name;
    value =        it.value;
    quality =        it.quality;
    hServerHandle =     it.hServerHandle;
    hClientHandle =     it.hClientHandle;
    time =        it.time;
    type =        it.type;
  }
};

class OPCClient:public QObject
{
  Q_OBJECT
public:
  explicit OPCClient();
  OPCClient(QString s);
  ~OPCClient();
  COPCTransaction *m_COPCTransaction;

  bool isWriteAble;

  bool isConnected();
  HRESULT Connect(QString s);
  void DisConnect();
  HRESULT AddGroup(QString n, DWORD update_rate , int async=1);
  QStringList AddItems(QStringList names);
  bool ReadItem(ItemDef *item);

  bool WriteValue(QString &name, QString &value );
  bool WriteValue( DWORD cHandle, FILETIME &time, VARIANT &value, WORD Quality );
  HRESULT WriteValue_Async(ItemDef * item);
  HRESULT WriteValue_Sync (ItemDef * item);

  HRESULT RemoveItems(QStringList inames);
  HRESULT RemoveAllItems();

  void dataCallback();
  QList<ItemDef*> *items;

  QString VariantString(VARIANT &pValue);
  QString QualityString(UINT qnr);
  QString ValueTypeString(const VARTYPE& v);
  void StringVariant(ItemDef* item, QString v);
  QString AccessTypeString(qint16 accessIdent);
  QString TimeString(FILETIME t);

  HRESULT hResult;
  HRESULT *pErrors;

  WString  m_ServerName;
  WString  m_GroupName;
  DWORD    m_UpdateRate;
  CLSID    m_clsid;
  DWORD       m_Cookie;
  bool     m_Async;

  IConnectionPointContainer* _pIConnectionPointContainer;
  IConnectionPoint* _pIConnectionPoint;
  IOPCServer *_pIOPCServer;
  IOPCBrowseServerAddressSpace *_pIOpcNamespace;
  IOPCItemProperties *_pIOpcProperties;
  IOPCGroupStateMgt *_pIOPCGroupStateMgt;
  IOPCItemMgt *_pIOPCItemMgt;
  IOPCSyncIO * _pIOPCSyncIO;
  IOPCAsyncIO2 * _pIOPCAsyncIO2;
  IOPCDataCallback * _pIOPCDataCallback;
  COPCDataCallback *m_OPCDataCallback;

  OPCHANDLE  m_GroupHandle;
  OPCHANDLE  m_ServerHandle;
  void       ClearOPCITEMDEF( OPCITEMDEF *idef, int count = 1 );
  ItemDef*   getItemByName(QString s);
  ItemDef*   getItemByHandle(DWORD h);
  WString    qstr2Wstr(QString s);

};

#endif //OPCCLIENT_H
#include "OPCClient.h"
#include "copcdatacallback.h"
#include <QDateTime>
#include <QMessageBox>
/**
 * Copyright 2016 Chn.Captain (chn.captain@gmail.com)
 */
OPCClient::OPCClient(QString s):
  m_ServerName(qstr2Wstr(s))
{
}
OPCClient::OPCClient()
{
  _pIConnectionPointContainer = NULL;
  _pIConnectionPoint          = NULL;
  _pIOPCServer                = NULL;
  _pIOPCDataCallback          = NULL;
  _pIOpcNamespace             = NULL;
  _pIOpcProperties            = NULL;
  _pIOPCGroupStateMgt         = NULL;
  _pIOPCItemMgt               = NULL;
  _pIOPCSyncIO                = NULL;
  _pIOPCAsyncIO2              = NULL;
  m_GroupHandle               = 0;
  m_ServerHandle              = 0;
  m_OPCDataCallback           = NULL;
  m_COPCTransaction           = NULL;
  m_Cookie                    = 0;
  pErrors                     = NULL;
  //state
  isWriteAble                 = NULL;

  items = new QList<ItemDef*>();
}
/**
 * 創建OPC鏈接
 * @brief OPCClient::Connect
 * @param s
 * @return
 */
HRESULT OPCClient::Connect(QString s)
{
  this->m_ServerName = qstr2Wstr(s);
  hResult = CoInitialize(0);
  if (FAILED(hResult))
    {
      if (hResult == S_FALSE)
        {
          qDebug()<<"Error CoInitialize():COM Library already initialized";
        }
      else
        {
          qDebug()<< "Error CoInitialize():Initialisation of COM Library failed. Error Code= " << hResult;
          _pIOPCServer = 0;
          CoUninitialize();
          return hResult;
        }
    }

  hResult = CLSIDFromProgID(this->m_ServerName, &this->m_clsid);
  if (FAILED(hResult))
    {
      qDebug()<< "Error CLSIDFromProgID():Retrival of CLSID failed";
      CoUninitialize();
      return hResult;
    }

  hResult = CoCreateInstance (this->m_clsid, NULL, CLSCTX_LOCAL_SERVER ,IID_IOPCServer, (void**)&_pIOPCServer);
  if (FAILED(hResult))
    {
      qDebug()<< "Error CoCreateInstance():Creation of IOPCServer-Object failed";
      _pIOPCServer = 0;
      CoTaskMemFree(&this->m_clsid);
      CoUninitialize();
      return hResult;
    }
  hResult = _pIOPCServer->QueryInterface(IID_IOPCBrowseServerAddressSpace, (void**)&_pIOpcNamespace);
  if (FAILED(hResult))
    {

      qDebug()<< "Error CoCreateInstance():Creation of IID_IOPCBrowseServerAddressSpace-Object failed";
      CoUninitialize();
      return hResult;
    }
  hResult = _pIOPCServer->QueryInterface(IID_IOPCItemProperties, (void**)&_pIOpcProperties);
  if (FAILED(hResult))
    {
      qDebug()<< "Error CoCreateInstance():Creation of IID_IOPCItemProperties-Object failed";
      CoUninitialize();
      return hResult;
    }
  return hResult;
}

/**
 * 添加分組
 * @brief OPCClient::AddGroup
 * @param n
 * @param update_rate
 * @return
 */
HRESULT OPCClient::AddGroup(QString n, DWORD update_rate, int async)
{
  m_Async = async;
  m_GroupName = qstr2Wstr(n);
  m_ServerHandle = 1;
  m_UpdateRate = 100;
  long TimeBias;
  DWORD LanguageCode = 0x416, RevisedUpdateRate;
  float DeadBand; //刷新間隔
  hResult=_pIOPCServer->AddGroup(m_GroupName,  // [in] group name
                                 TRUE,                  // [in] active
                                 update_rate,           // [in] request this Update Rate from Server
                                 m_ServerHandle,        // [in] Client handle
                                 &TimeBias,             // [in] no time interval to system UTC time
                                 &DeadBand,             // [in] no deadband, so all data changes are reported
                                 LanguageCode,          // [in] Server uses English language for text values
                                 &m_GroupHandle,        // [out] Server handle to identify this group in later calls
                                 &RevisedUpdateRate,    // [out] the answer form the Server to the requested update rate
                                 IID_IOPCGroupStateMgt, // [in] requested interface type of the group object
                                 (LPUNKNOWN*)&this->_pIOPCGroupStateMgt); // [out] pointer to the requested interface

  if( hResult ==OPC_E_DUPLICATENAME ) {
      qDebug()<< "1001:分組名稱已存在.";
    }
  if( hResult ==E_NOINTERFACE ) {
      qDebug()<< "1002:IOPCServer::AddGroup returned E_NOINTERFACE (IID_IOPCGroupStateMgt)";
    }
  if( update_rate != m_UpdateRate ){
      qDebug()<< "1003: OPC server rejected data refresh interval. Setting it to %1 ms." << update_rate;
    }
  if (hResult == OPC_S_UNSUPPORTEDRATE)
    {
      qDebug()<< "1004:請求的刷新速率與實際的刷新速率不一致";
    }
  if( FAILED(hResult) || _pIOPCGroupStateMgt == NULL) {
      qDebug()<< "1005: 建立分組失敗";
      DisConnect();
    }

  hResult = _pIOPCGroupStateMgt->QueryInterface(IID_IOPCItemMgt,(void**)&this->_pIOPCItemMgt);
  if (FAILED(hResult))  {
      qDebug()<<"1006: 項目操做指針(_pIOPCItemMgt)建立失敗";
      DisConnect();
    }
  //查詢 group 對象的同步接口



  if(m_Async){
      hResult = _pIOPCItemMgt->QueryInterface(IID_IOPCAsyncIO2, (void**)&this->_pIOPCAsyncIO2);
      if (FAILED(hResult))  {
          qDebug()<<"1008: IOPCAsyncIO2 沒有發現,錯誤的查詢!";
          DisConnect();
        }

      m_COPCTransaction = new COPCTransaction;
      hResult = _pIOPCItemMgt->QueryInterface(IID_IConnectionPointContainer, (void**)&this->_pIConnectionPointContainer);

      //hResult =  _pIOPCItemMgt->QueryInterface(&_pIConnectionPointContainer);
      if (FAILED(hResult))  {
          qDebug()<<"1009: IConnectionPointContainer 建立失敗!";
          DisConnect();
        }
      hResult = _pIConnectionPointContainer->FindConnectionPoint( IID_IOPCDataCallback, &_pIConnectionPoint );
      if( FAILED(hResult) || _pIConnectionPoint == NULL) {
          qDebug()<< "1010: A group of OPC with no IOPCDataCallback. OPC server appears to not conform to the OPC DA 2.0 standard (%s).";
          DisConnect();
        }
      if( m_OPCDataCallback == NULL ) {

          if(!m_COPCTransaction){
              m_COPCTransaction = new COPCTransaction;
            }
          m_OPCDataCallback = new COPCDataCallback(m_COPCTransaction);
          m_OPCDataCallback->AddRef();
          // m_OPCDataCallback->receiver = m_DataReceiver;
        }
      hResult = _pIConnectionPoint->Advise(m_OPCDataCallback, &m_Cookie);
      qDebug()<< hResult;
      if(FAILED(hResult)) {
          qDebug()<< "1011: OPCDataCallback set faild." << m_Cookie;
          _pIConnectionPointContainer->Release();
          DisConnect();
        }
      isWriteAble = true;
    }else{
      hResult = _pIOPCItemMgt->QueryInterface(IID_IOPCSyncIO, (void**)&this->_pIOPCSyncIO);
      if (FAILED(hResult))  {
          qDebug()<<"1007: IOPCSyncIO 沒有發現,錯誤的查詢!";
          DisConnect();
        }
      isWriteAble = true;
    }

  return hResult;
}

/**
 * 添加監視項
 * @brief OPCClient::AddItems
 * @param inames
 * @return
 */
QStringList OPCClient::AddItems(QStringList inames)
{
  QStringList r_inames;

  if(!_pIOPCItemMgt) return r_inames;

  pErrors = NULL;
  DWORD iCount=items->count();
  OPCITEMRESULT * pResults;
  qDebug()<<"1--- begin add item -----------";
  for(int i=0;i<inames.count();i++) {
      QString iname = inames[i];
      if(getItemByName(iname)){
          qDebug() << "    already exsits."<< iname;
          continue;
        }
      r_inames.append(iname);
    }
  int nCount=r_inames.count();
  if(nCount>0){
      OPCITEMDEF idef[nCount];
      for(int i=0;i<nCount;i++) {
          QString iname = r_inames[i];
          qDebug()<< "    build item def"<< iname;
          idef[i].szAccessPath= SysAllocString(L"");           // path SysAllocString(qstr2Wstr(name))
          idef[i].szItemID    = SysAllocString(qstr2Wstr(iname));       // name
          idef[i].bActive     = TRUE;
          idef[i].hClient     = iCount+i;
          idef[i].dwBlobSize  = 0;
          idef[i].pBlob       = NULL;
          idef[i].vtRequestedDataType = 0;
        }
      qDebug()<<"2--- end of add item-----------";

      hResult = _pIOPCItemMgt->AddItems(nCount,  // [in] add items
                                        idef,        // [in] see above
                                        &pResults,   // [out] array with additional information about the item
                                        &pErrors);   // [out] tells which of the items was successfully added.

      if(hResult==S_OK && hResult!=S_FALSE){
          for(int i=0;i < nCount; i++) {
              QString iname = r_inames[i];
              DWORD cItemHandle = iCount+i;
              ItemDef *item = new ItemDef;
              item->name = iname;
              item->accessRights = pResults[i].dwAccessRights;
              item->hClientHandle = cItemHandle;
              item->type = pResults[i].vtCanonicalDataType;
              item->hServerHandle = pResults[i].hServer;
              items->append(item);
              //    檢測 Item 的可讀寫性
              //            if (pResults[i].dwAccessRights != (OPC_READABLE +  OPC_WRITEABLE))   {
              //                qDebug()<<"Item 不可讀,也不可寫,請檢查服務器配置";
              //              }
            }
        }

     if(pResults) CoTaskMemFree( pResults );
      if(pErrors)  CoTaskMemFree( pErrors );
      ClearOPCITEMDEF(idef, nCount);
    }
   return r_inames;
}
/**
 * 同步讀取指定項
 * @brief OPCClient::ReadItem
 * @param item
 * @return
 */
bool OPCClient::ReadItem(ItemDef *item)
{
  bool result = false;
  OPCITEMSTATE *pItemValue;
  hResult = _pIOPCSyncIO->Read(OPC_DS_DEVICE, 1, &item->hServerHandle,  &pItemValue, &pErrors);

  if(SUCCEEDED(hResult)){
      item->value = pItemValue[0].vDataValue;
      item->quality = pItemValue[0].wQuality;
      item->time = pItemValue[0].ftTimeStamp;
      item->hClientHandle = pItemValue[0].hClient;
      result = true;
      VariantClear(&pItemValue->vDataValue);
    }else{
      qDebug()<< "同步讀取項目失敗" << hResult;
    }

  CoTaskMemFree(pErrors);
  CoTaskMemFree(pItemValue);

  return result;
}
/**
 * 寫入值
 * @brief OPCClient::WriteValue
 * @param cHandle
 * @param time
 * @param value
 * @param Quality
 * @return
 */
bool OPCClient::WriteValue( DWORD cHandle, FILETIME &time, VARIANT &value, WORD Quality )
{
    ItemDef * item = getItemByHandle( cHandle );
    if( !item )
        return false;

    item->quality = Quality;
    item->value = value;
    item->time = time;

    if( m_Async )
        return SUCCEEDED( WriteValue_Async(item) );
    else
        return SUCCEEDED( WriteValue_Sync( item ) );
}
bool OPCClient::WriteValue(QString &n, QString &v )
{
    ItemDef * item = getItemByName(n);
    if(!item )
        return false;
    StringVariant(item, v);
    if( m_Async )
        return SUCCEEDED( WriteValue_Async(item) );
    else
        return SUCCEEDED( WriteValue_Sync( item ) );
}
HRESULT OPCClient::WriteValue_Async(ItemDef * item)
{
          QString str;
          pErrors    = NULL;
          if( m_COPCTransaction == NULL ) return E_FAIL;
          if( !isConnected() ) return E_FAIL;

         // m_Group->QueryInterface( &AsyncIO );
          if( _pIOPCAsyncIO2 == NULL ) {
                  qDebug()<< "Failed to get interface IOPCAsyncIO2";
                  return E_FAIL;
          }
          DWORD cancelID = 0;

          hResult = _pIOPCAsyncIO2->Write( 1, &item->hServerHandle, &item->value, rand(), &cancelID, &pErrors);
          if( FAILED(hResult) /*|| FAILED(pErrors[0])*/) {
                  qDebug()<< "Parameter [%s] is not passed" << item->name;
                  hResult = E_FAIL;

          }
          if( pErrors )
          {
                  switch( pErrors[0] ) {
                          case OPC_S_CLAMP:
                                  qDebug()<< "AsyncIO->Write(%s) -> [OPC_S_CLAMP] The value was accepted but was clamped."<<item->name;
                                  break;
                          case OPC_E_RANGE: qDebug()<< "AsyncIO->Write(%s) -> [OPC_E_RANGE] The value was out of range."<<item->name;
                                  break;
                          case OPC_E_BADTYPE:
                                  str=QString("AsyncIO->Write(%1) -> [OPC_E_BADTYPE] The passed data type (%2) cannot be accepted for this item.").arg(item->name, item->value.vt);
                                  qDebug()<< str;
                                  break;
                          case OPC_E_BADRIGHTS:
                                  qDebug()<< "AsyncIO->Write(%s) -> [OPC_E_BADRIGHTS] The item is not writeable."<<item->name;
                                  break;
                          case OPC_E_INVALIDHANDLE:
                                  qDebug()<< "AsyncIO->Write(%s) -> [OPC_E_INVALIDHANDLE] The passed item handle was invalid."<<item->name;
                                  break;
                          case OPC_E_UNKNOWNITEMID:
                                  qDebug()<< "AsyncIO->Write(%s) -> [OPC_E_UNKNOWNITEMID] The item is no longer available in the server address space."<<item->name;
                                  break;
                  }

          }
          if( pErrors )CoTaskMemFree( pErrors );
          return hResult;

}

HRESULT OPCClient::WriteValue_Sync( ItemDef * item )
{
  pErrors    = NULL;
  if( m_COPCTransaction == NULL ) return E_FAIL;
  if( !isConnected() ) return E_FAIL;

        qDebug() << "Sync Write start (hdl="<< item->hServerHandle << ") value=" << VariantString(item->value);

        hResult = _pIOPCSyncIO->Write( 1, &item->hServerHandle, &item->value, &pErrors);

        qDebug() << "Sync Write finished. hr = " << hResult;

    if( FAILED(hResult) /*|| FAILED(pErrors[0])*/) {
        qDebug() << "Parameter [%s] is not passed"<<item->name;
        hResult = E_FAIL;

        if( pErrors != NULL)
        {
            switch( pErrors[0] ) {
            case OPC_S_CLAMP:
                qDebug() << "SyncIO->Write(%s) -> [OPC_S_CLAMP] The value was accepted but was clamped." << item->name ;
                break;
            case OPC_E_RANGE:
                qDebug() << "SyncIO->Write(%s) -> [OPC_E_RANGE] The value was out of range."<<item->name  ;
                break;
            case OPC_E_BADTYPE:
                qDebug() << "SyncIO->Write(%s) -> [OPC_E_BADTYPE] The passed data type cannot be accepted for this item."<<item->name ;
                break;
            case OPC_E_BADRIGHTS:
                qDebug() << "SyncIO->Write(%s) -> [OPC_E_BADRIGHTS] The item is not writeable."<<item->name ;
                break;
            case OPC_E_INVALIDHANDLE:
                qDebug() << "SyncIO->Write(%s) -> [OPC_E_INVALIDHANDLE] The passed item handle was invalid."<<item->name ;
                break;
            case OPC_E_UNKNOWNITEMID:
                qDebug() << "SyncIO->Write(%s) -> [OPC_E_UNKNOWNITEMID] The item is no longer available in the server address space."<<item->name ;
                break;
            }
        }
    }

    if( pErrors )
        CoTaskMemFree( pErrors );

    return hResult;
}

/**
 * 移除全部項目
 * @brief OPCClient::RemoveItems
 * @param _items
 * @return
 */
HRESULT OPCClient::RemoveItems(QStringList inames)
{
  int _iCount = inames.count();
  if( _iCount==0) return -1;
  OPCHANDLE _hpSvr[_iCount];
  for(int i=0;i<_iCount ;i++){
      if(getItemByName(inames.value(i))){
          _hpSvr[i] = getItemByName(inames.at(i))->hServerHandle;
        }

    }

  hResult = _pIOPCItemMgt->RemoveItems(_iCount, _hpSvr, &pErrors);
  if(SUCCEEDED(hResult)){
      for(int i=0;i<_iCount ;i++){
          items->removeAll(getItemByName(inames.at(i)));;
        }
    }

  return hResult;
}
/**
 * @brief OPCClient::RemoveAllItems
 * @return
 */
HRESULT OPCClient::RemoveAllItems()
{
  int _iCount = items->count();
  OPCHANDLE _hpSvr[_iCount];
  qDebug()<<"3---begin delete-----------";
  for(int i=0;i<_iCount ;i++){
      qDebug() <<"    next one->"<< items->at(i)->hServerHandle << items->at(i)->name;
      _hpSvr[i] = items->at(i)->hServerHandle;
    }

  hResult = _pIOPCItemMgt->RemoveItems(_iCount, _hpSvr, &pErrors);
  qDebug() <<"4---end delete-----------" << hResult;
  if(SUCCEEDED(hResult)){
      while(items->count()!=0){
          items->removeAt(0);
        }
    }
  return hResult;
}
/**
 * 斷開鏈接
 * @brief OPCClient::DisConnect
 */
void OPCClient::DisConnect()
{
  if(_pIOPCSyncIO){
      _pIOPCSyncIO->Release();
      _pIOPCSyncIO=0;
    }
  if (_pIConnectionPoint)
    {
      _pIConnectionPoint->Unadvise(m_Cookie);
      _pIConnectionPoint->Release();
      _pIConnectionPoint = 0;
    }
  if(_pIOPCItemMgt){
      _pIOPCItemMgt->Release();
      _pIOPCItemMgt = 0;
    }
  if(_pIOPCGroupStateMgt){
      _pIOPCGroupStateMgt->Release();
      _pIOPCGroupStateMgt = 0;
    }
  if(_pIOPCServer){
      _pIOPCServer->RemoveGroup(m_GroupHandle, TRUE);
      _pIOPCServer->Release();
      _pIOPCServer=0;
    }
  if(m_OPCDataCallback){
      m_OPCDataCallback->Release();
      m_OPCDataCallback = 0;
    }
  if(m_COPCTransaction){
      m_COPCTransaction->deleteLater();
      m_COPCTransaction = 0;
    }

  m_GroupHandle = 0;
  m_ServerHandle = 0;
  CoUninitialize();
  delete items;
}

void OPCClient::dataCallback()
{


  //  // 創建異步回調
  //  CComObject<COPCDataCallback>* pCOPCDataCallback;
  //  // 回調對象的指針
  //  // 經過 ATL 模板建立回調對象的實例
  //  CComObject<COPCDataCallback>::CreateInstance(&pCOPCD ataCallback);
  //  // 查詢 IUnknown 接口
  //  LPUNKNOWN pCbUnk;
  //  pCbUnk = pCOPCDataCallback->GetUnknown();
  //  // 創建一個服務器的鏈接點與客戶程序接收器之間的鏈接
  //  HRESULT hRes = AtlAdvise( m_IOPCGroupStateMgt, // [in]  //鏈接點的 IUnknown 接口
  //                            pCbUnk, // [in] 回調對象的 IUnknown 接口
  //                            IID_IOPCDataCallback,// [in] 鏈接點 ID
  //                            &m_dwAdvise    // [out] 惟一的標識符
  //                            );
  //    if (hRes != S_OK)    {
  //        AfxMessageBox("Advise 失敗!");
  //      }
}

/**
 * 鏈接狀態
 * @brief OPCClient::isConnected
 * @return
 */
bool OPCClient::isConnected()
{
  return ( _pIOPCServer && _pIOPCGroupStateMgt );
}

void OPCClient::ClearOPCITEMDEF( OPCITEMDEF *idef, int count )
{
  if( idef )
    for(int i=0;i<count;i++)
      {
        if( idef[i].szItemID != NULL )
          SysFreeString( idef[i].szItemID );
        if( idef[i].szAccessPath != NULL )
          SysFreeString( idef[i].szAccessPath );
      }
}
/**
 * 獲取ItemDef by name
 * @brief OPCClient::getItemByName
 * @param s
 * @return
 */
ItemDef* OPCClient::getItemByName(QString s)
{
  int c = items->count();

  for(int i=0; i<c; i++) {
      if(items->at(i)->name == s){
          return items->at(i);
        }
    }
  return 0;
}
/**
 * 獲取ItemDef by Handle
 * @brief OPCClient::getItemByHandle
 * @param s
 * @return
 */
ItemDef* OPCClient::getItemByHandle(DWORD h)
{
  int c = items->count();

  for(int i=0; i<c; i++) {
      if(items->at(i)->hClientHandle == h){
          return items->at(i);
        }
    }
  return 0;
}
/**
 * 析構函數
 * @brief OPCClient::~OPCClient
 */
OPCClient::~OPCClient()
{
  DisConnect();
}
//////////////////////////////////////////////////

/**
 * QString 轉換爲 const wchar*
 * @brief OPCClient::qstr2Wstr
 * @param s
 * @return
 */
WString OPCClient::qstr2Wstr(QString s)
{
  return reinterpret_cast<const wchar_t*>(s.utf16());
  //wchar_t* tempWide = const_cast< wchar_t* >( tempConstWide );
}
/**
 * 輸出項目值字符串
 * @brief OPCClient::VariantString
 * @param pValue
 * @return
 */
QString OPCClient::VariantString(VARIANT &pValue)
{
  QString valueString;
  //qDebug()<< QString(" Format type:%1 ").arg(pValue.vt);
  switch(pValue.vt)
    {
    case VT_I1:
    case VT_UI1:        // BYTE
      {
        int i = pValue.iVal;
        valueString= QString::number(i);
        break;
      }
    case VT_I2:            // SHORT
      valueString= QString::number(pValue.iVal);
      break;
    case VT_UI2:        // UNSIGNED SHORT
      valueString= QString::number(pValue.uiVal);

      break;
    case VT_I4:            // LONG
      valueString= QString::number(pValue.lVal);
      break;
    case VT_UI4:        // UNSIGNED LONG
      valueString= QString::number(pValue.ulVal);
      break;
    case VT_INT:        // INTEGER
      valueString= QString::number(pValue.intVal);
      break;
    case VT_UINT:        // UNSIGNED INTEGER
      valueString= QString::number(pValue.uintVal);

      break;
    case VT_R4:            // FLOAT
      //sprintf    (buf, "%5.2f ", pValue.fltVal );
      valueString= QString::number(pValue.fltVal);
      break;
    case VT_R8:            // DOUBLE
      //sprintf    (buf, "%9.4f ", pValue.dblVal );
      valueString= QString::number(pValue.dblVal);
      break;
    case VT_BSTR:        //BSTR
      {
        //sprintf    (buf, "%ls ", pValue.bstrVal );
        BSTR bstr_str = pValue.bstrVal;
        QString q_str((QChar*)bstr_str, wcslen(bstr_str));
        valueString = q_str;

        break;
      }
    case VT_BOOL:
      {
        if (pValue.boolVal)
          valueString = "TRUE";
        else
          valueString = "FALSE";
        break;
      }
    case VT_DATE:
      {
        QDateTime dt;
        //qDebug()<< pValue.date;
        dt.fromMSecsSinceEpoch(pValue.date);
        valueString =  dt.toString("yyyy-MM-dd hh:mm:ss");
        break;
      }
    default:
      valueString = QString(" unknown type:%1 ").arg(pValue.vt);
      break;
    }
  return valueString;
}
/**
 * 輸出項目值質量字符串
 * @brief OPCClient::VariantString
 * @param pValue
 * @return
 */
QString OPCClient::QualityString(UINT qnr)
{
  QString result;
  switch(qnr)   {
    case OPC_QUALITY_BAD:
      result = "BAD";
      break;
    case OPC_QUALITY_UNCERTAIN:
      result= "UNCERTAIN";
      break;
    case OPC_QUALITY_GOOD:
      result = "GOOD";
      break;
    case OPC_QUALITY_NOT_CONNECTED:
      result = "NOT_CONNECTED";
      break;
    case OPC_QUALITY_DEVICE_FAILURE:
      result = "DEVICE_FAILURE";
      break;
    case OPC_QUALITY_SENSOR_FAILURE:
      result = "SENSOR_FAILURE";
      break;
    case OPC_QUALITY_LAST_KNOWN:
      result = "LAST_KNOWN";
      break;
    case OPC_QUALITY_COMM_FAILURE:
      result = "COMM_FAILURE";
      break;
    case OPC_QUALITY_OUT_OF_SERVICE:
      result = "OUT_OF_SERVICE";
      break;
    case OPC_QUALITY_LAST_USABLE:
      result = "LAST_USABLE";
      break;
    case OPC_QUALITY_SENSOR_CAL:
      result = "SENSOR_CAL";
      break;
    case OPC_QUALITY_EGU_EXCEEDED:
      result = "EGU_EXCEEDED";
      break; case OPC_QUALITY_SUB_NORMAL:
      result = "SUB_NORMAL";
      break;
    case OPC_QUALITY_LOCAL_OVERRIDE:
      result = "LOCAL_OVERRIDE";
      break;
    default:
      result = "UNKNOWN ERROR";
    }
  return result;
}
/**
 * 輸出值類型字符串
 * @brief OPCClient::ValueTypeString
 * @param v
 * @return
 */
QString OPCClient::ValueTypeString(const VARTYPE& v)
{
  QString str;

  switch(v) {
    case VT_BSTR:
      str = QString("VT_BSTR");
      break;
    case VT_I1:
      str = QString("VT_I1");
      break;
    case VT_I2:
      str = QString("VT_I2");
      break;
    case VT_I4:
      str = QString("VT_I4");
      break;
    case VT_I8:
      str = QString("VT_I8");
      break;
    case VT_R4:
      str = QString("VT_R4");
      break;
    case VT_R8:
      str = QString("VT_R8");
      break;
    case VT_UI1:
      str = QString("VT_UI1");
      break;
    case VT_UI2:
      str = QString("VT_UI2");
      break;
    case VT_UI4:
      str = QString("VT_UI4");
      break;
    case VT_UI8:
      str = QString("VT_UI8");
      break;
    case VT_INT:
      str = QString("VT_INT");
      break;
    case VT_UINT:
      str = QString("VT_UINT");
      break;
    case VT_BOOL:
      str = QString("VT_BOOL");
      break;
    case VT_DATE:
      str = QString("VT_DATE");
      break;
    default:
      str = QString("unknown");
      break;
    };
  return str;
}
/**
 * 賦值
 * @brief OPCClient::StringVariant
 * @param item
 * @param v
 * @return
 */
void OPCClient::StringVariant(ItemDef* item, QString v)
{
  switch (item->value.vt)
    {
    case VT_UI1:
    case VT_I1:
        //cbRead = sizeof(BYTE);
        break;
    case VT_I2:
    case VT_UI2:
    case VT_BOOL:
        item->value.iVal = v.toShort();
        break;
    case VT_I4:
    case VT_UI4:
    case VT_INT:
    case VT_UINT:
    case VT_ERROR:
      item->value.lVal = v.toLong();
      break;
    case VT_R4:
      item->value.fltVal = v.toFloat();
      break;
    case VT_I8:
    case VT_UI8:
      item->value.llVal = v.toLongLong();
        break;
    case VT_R8:
    case VT_CY:
    case VT_DATE:
      item->value.dblVal = v.toDouble();
        break;
    case VT_BSTR:
      {
        //Value.bstrVal =  SysAllocString(v.utf16());
        //Value.bstrVal = qstringToBstr(v);
        wchar_t *pW = new wchar_t[v.size()+1];
        v.toWCharArray(pW);
        //dialog.m_valueValue.bstrVal = SysAllocString(dialog.m_value.toStdWString().c_str());
        break;
      }
    default:
      break;
    }

  return;
}
/**
 * 輸出權限字符串
 * @brief OPCClient::AccessTypeString
 * @param accessIdent
 * @return
 */
QString OPCClient::AccessTypeString(qint16 accessIdent)
{
  QString s;
  switch(accessIdent){
    case OPC_READABLE:
      s = "Read";
      break;
    case OPC_WRITEABLE:
      s = "Write";
      break;
    default:
      s = "Read/Write";
    }
  return s;
}
QString OPCClient::TimeString(FILETIME t)
{
  return QString::number(t.dwLowDateTime);
}
相關文章
相關標籤/搜索