⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sendmessagedlg.cpp

📁 短消息代码~! 短消息代码~! PC发送sms
💻 CPP
📖 第 1 页 / 共 2 页
字号:
        {
            // 组内其它字节,将其右边部分与残余数据相加,得到一个目标编码字节
            *pDst = (*pSrc << (8-nChar)) | nLeft;
    
            // 将该字节剩下的左边部分,作为残余数据保存起来
            nLeft = *pSrc >> nChar;
            // 修改目标串的指针和计数值 pDst++;
            nDst++; 
        } 
        
        // 修改源串的指针和计数值
        pSrc++; nSrc++;
    }
    
    // 返回目标串长度
    return nDst; 
}
     
// 7-bit解码
// pSrc: 源编码串指针
// pDst: 目标字符串指针
// nSrcLength: 源编码串长度
// 返回: 目标字符串长度
int gsmDecode7bit(const unsigned char* pSrc, char* pDst, int nSrcLength)
{
    int nSrc;        // 源字符串的计数值
    int nDst;        // 目标解码串的计数值
    int nByte;       // 当前正在处理的组内字节的序号,范围是0-6
    unsigned char nLeft;    // 上一字节残余的数据
    
    // 计数值初始化
    nSrc = 0;
    nDst = 0;
    
    // 组内字节序号和残余数据初始化
    nByte = 0;
    nLeft = 0;
    
    // 将源数据每7个字节分为一组,解压缩成8个字节
    // 循环该处理过程,直至源数据被处理完
    // 如果分组不到7字节,也能正确处理
    while(nSrc<nSrcLength)
    {
        // 将源字节右边部分与残余数据相加,去掉最高位,得到一个目标解码字节
        *pDst = ((*pSrc << nByte) | nLeft) & 0x7f;
        // 将该字节剩下的左边部分,作为残余数据保存起来
        nLeft = *pSrc >> (7-nByte);
    
        // 修改目标串的指针和计数值
        pDst++;
        nDst++;
    
        // 修改字节计数值
        nByte++;
    
        // 到了一组的最后一个字节
        if(nByte == 7)
        {
            // 额外得到一个目标解码字节
            *pDst = nLeft;
    
            // 修改目标串的指针和计数值
            pDst++;
            nDst++;
    
            // 组内字节序号和残余数据初始化
            nByte = 0;
            nLeft = 0;
        }
    
        // 修改源串的指针和计数值
        pSrc++;
        nSrc++;
    }
    
    *pDst = 0;
    
    // 返回目标串长度
    return nDst;
}


// UCS2编码
// pSrc: 源字符串指针
// pDst: 目标编码串指针
// nSrcLength: 源字符串长度
// 返回: 目标编码串长度
int gsmEncodeUcs2(const char* pSrc, unsigned char* pDst, int nSrcLength)
{
    int nDstLength;        // UNICODE宽字符数目
    WCHAR wchar[128];      // UNICODE串缓冲区
    
    // 字符串-->UNICODE串
    nDstLength = ::MultiByteToWideChar(CP_ACP, 0, pSrc, nSrcLength, wchar, 128);
    
    // 高低字节对调,输出
    for(int i=0; i<nDstLength; i++)
    {
        // 先输出高位字节
        *pDst++ = wchar[i] >> 8;
        // 后输出低位字节
        *pDst++ = wchar[i] & 0xff;
    }
    
    // 返回目标编码串长度
    return nDstLength * 2;
}
 
    
// UCS2解码
// pSrc: 源编码串指针
// pDst: 目标字符串指针
// nSrcLength: 源编码串长度
// 返回: 目标字符串长度
int gsmDecodeUcs2(const unsigned char* pSrc, char* pDst, int nSrcLength)
{
    int nDstLength;        // UNICODE宽字符数目
    WCHAR wchar[128];      // UNICODE串缓冲区
    
    // 高低字节对调,拼成UNICODE
    for(int i=0; i<nSrcLength/2; i++)
    {
        // 先高位字节
        wchar[i] = *pSrc++ << 8;
    
        // 后低位字节
        wchar[i] |= *pSrc++;
    }
    
    // UNICODE串-->字符串
    nDstLength = ::WideCharToMultiByte(CP_ACP, 0, wchar, nSrcLength/2, pDst, 160, NULL, NULL);
    
    // 输出字符串加个结束符    
    pDst[nDstLength] = '\0';    
    
    // 返回目标字符串长度
    return nDstLength;
}


// 正常顺序的字符串转换为两两颠倒的字符串,若长度为奇数,补'F'凑成偶数
// 如:"8613851872468" --> "683158812764F8"
// pSrc: 源字符串指针
// pDst: 目标字符串指针
// nSrcLength: 源字符串长度
// 返回: 目标字符串长度
int gsmInvertNumbers(const char* pSrc, char* pDst, int nSrcLength)
{
    int nDstLength;   // 目标字符串长度
    char ch;          // 用于保存一个字符

    // 复制串长度
    nDstLength = nSrcLength;

    // 两两颠倒
    for(int i=0; i<nSrcLength;i+=2)
    {
        ch = *pSrc++;        // 保存先出现的字符
        *pDst++ = *pSrc++;   // 复制后出现的字符
        *pDst++ = ch;        // 复制先出现的字符
    }

    // 源串长度是奇数吗?
    if(nSrcLength & 1)
    {
        *(pDst-2) = 'F';     // 补'F'
        nDstLength++;        // 目标串长度加1
    }

    // 输出字符串加个结束符
    *pDst = '\0';

    // 返回目标字符串长度
    return nDstLength;
}

/*
// 发送短消息
// pSrc: 源PDU参数指针
BOOL gsmSendMessage(const SM_PARAM* pSrc)
{
    int nPduLength;        // PDU串长度
    unsigned char nSmscLength;    // SMSC串长度
    int nLength;           // 串口收到的数据长度
    char cmd[16];          // 命令串
    char pdu[512];         // PDU串
    char ans[128];         // 应答串


	CString at;
	at.Format("AT+CMGF=0\r");
	m_Com.SetOutput(COleVariant(at));

	VARIANT	MyInBuffer = m_Com.GetInput();
			
	CString	str = (CString)MyInBuffer.bstrVal;

    nPduLength = gsmEncodePdu(pSrc, pdu);    // 根据PDU参数,编码PDU串
    strcat(pdu, "\x01a");        // 以Ctrl-Z结束

    gsmString2Bytes(pdu, &nSmscLength, 2);    // 取PDU串中的SMSC信息长度
    nSmscLength++;        // 加上长度字节本身

    // 命令中的长度,不包括SMSC信息长度,以数据字节计
    sprintf(cmd, "AT+CMGS=%d\r", nPduLength / 2 - nSmscLength);    // 生成命令

//   WriteComm(cmd, strlen(cmd));    // 先输出命令串
//	m_Com.SetOutput(COleVariant(cmd, strlen(cmd)));
	m_Com.SetOutput(COleVariant(cmd));
			

	VARIANT MyInBuffer1= m_Com.GetInput();
			
	str = (CString)MyInBuffer1.bstrVal;
		//	MessageBox(str);
  
	memcpy(ans,str.GetBuffer(128),128);
		
	nLength=strlen(ans);
   // nLength = ReadComm(ans, 128);   // 读应答数据
 // VARIANT MyInBuffer = m_Com.GetInput();
    // 根据能否找到"\r\n> "决定成功与否
  // if(nLength == 4 && strncmp(ans, "\r\n> ", 4) == 0)
    {
       // m_Com.SetOutput(COleVariant(pdu, strlen(pdu)));
		m_Com.SetOutput(COleVariant(pdu));

	   //WriteComm(pdu, strlen(pdu));        // 得到肯定回答,继续输出PDU串
//	 VARIANT MyInBuffer = m_Com.GetInput();
        // VARIANT pDU
       VARIANT MyInBuffer2 = m_Com.GetInput();
			
		CString	str = (CString)MyInBuffer2.bstrVal;
		//	MessageBox(str);
  
     	memcpy(ans,str.GetBuffer(128),128);
		nLength=strlen(ans);
		//   nLength = ReadComm(ans, 128);       // 读应答数据

		// 根据能否找到"+CMS ERROR"决定成功与否
		// if(nLength > 0 && strncmp(ans, "+CMS ERROR", 10) != 0)
		// {
		//    return TRUE;
		//}
    }

	return TRUE;
   // return FALSE;
}
// 读取短消息
// 用+CMGL代替+CMGR,可一次性读出全部短消息
// pMsg: 短消息缓冲区,必须足够大
// 返回: 短消息条数
int gsmReadMessage(SM_PARAM* pMsg)
{
    int nLength = 0;        // 串口收到的数据长度
    int nMsg;           // 短消息计数值
    char* ptr;          // 内部用的数据指针
    char cmd[16];       // 命令串
    char ans[1024];     // 应答串

    nMsg = 0;
    ptr = ans;
    
    sprintf(cmd, "AT+CMGL\r");    // 生成命令
    
///   WriteComm(cmd, strlen(cmd));    // 输出命令串

   // nLength = ReadComm(ans, 1024);    // 读应答数据
    // 根据能否找到"+CMS ERROR"决定成功与否
    if(nLength > 0 && strncmp(ans, "+CMS ERROR", 10) != 0)
    {
        // 循环读取每一条短消息, 以"+CMGL:"开头
        while((ptr = strstr(ptr, "+CMGL:")) != NULL)
        {
            ptr += 6;        // 跳过"+CMGL:"
            sscanf(ptr, "%d", &pMsg->index);    // 读取序号
            TRACE("  index=%d\n",pMsg->index);

            ptr = strstr(ptr, "\r\n");    // 找下一行
            ptr += 2;        // 跳过"\r\n"

            gsmDecodePdu(ptr, pMsg);    // PDU串解码
            pMsg++;        // 准备读下一条短消息
            nMsg++;        // 短消息计数加1
        }
    }
    return nMsg;
}
*/

// PDU编码,用于编制、发送短消息
// pSrc: 源PDU参数指针
// pDst: 目标PDU串指针
// 返回: 目标PDU串长度
int gsmEncodePdu(const SM_PARAM* pSrc, char* pDst)
{
    int nLength;             // 内部用的串长度
    int nDstLength;          // 目标PDU串长度
    unsigned char buf[256];  // 内部用的缓冲区

    // SMSC地址信息段
    nLength = strlen(pSrc->SCA);    // SMSC地址字符串的长度
    buf[0] = (char)((nLength & 1) == 0 ? nLength : nLength + 1) / 2 + 1;    // SMSC地址信息长度
    buf[1] = 0x91;        // 固定: 用国际格式号码
    nDstLength = gsmBytes2String(buf, pDst, 2);        // 转换2个字节到目标PDU串
    nDstLength += gsmInvertNumbers(pSrc->SCA, &pDst[nDstLength], nLength);    // 转换SMSC到目标PDU串

    // TPDU段基本参数、目标地址等
    nLength = strlen(pSrc->TPA);    // TP-DA地址字符串的长度
    buf[0] = 0x11;            // 是发送短信(TP-MTI=01),TP-VP用相对格式(TP-VPF=10)
    buf[1] = 0;               // TP-MR=0
    buf[2] = (char)nLength;   // 目标地址数字个数(TP-DA地址字符串真实长度)
    buf[3] = 0x91;            // 固定: 用国际格式号码
    nDstLength += gsmBytes2String(buf, &pDst[nDstLength], 4);  // 转换4个字节到目标PDU串
    nDstLength += gsmInvertNumbers(pSrc->TPA, &pDst[nDstLength], nLength); // 转换TP-DA到目标PDU串

    // TPDU段协议标识、编码方式、用户信息等
    nLength = strlen(pSrc->TP_UD);    // 用户信息字符串的长度
    buf[0] = pSrc->TP_PID;        // 协议标识(TP-PID)
    buf[1] = pSrc->TP_DCS;        // 用户信息编码方式(TP-DCS)
    buf[2] = 0;            // 有效期(TP-VP)为5分钟
    if(pSrc->TP_DCS == GSM_7BIT)
    {
        // 7-bit编码方式
        buf[3] = nLength;            // 编码前长度
        nLength = gsmEncode7bit(pSrc->TP_UD, &buf[4], nLength+1) + 4;    // 转换TP-DA到目标PDU串
    }
    else if(pSrc->TP_DCS == GSM_UCS2)
    {
        // UCS2编码方式
        buf[3] = gsmEncodeUcs2(pSrc->TP_UD, &buf[4], nLength);    // 转换TP-DA到目标PDU串
        nLength = buf[3] + 4;        // nLength等于该段数据长度
    }
    else
    {
        // 8-bit编码方式
       // buf[3] = gsmEncode8bit(pSrc->TP_UD, &buf[4], nLength);    // 转换TP-DA到目标PDU串
       // nLength = buf[3] + 4;        // nLength等于该段数据长度
    }
    nDstLength += gsmBytes2String(buf, &pDst[nDstLength], nLength);        // 转换该段数据到目标PDU串

    // 返回目标字符串长度
    return nDstLength;
}
//gsmString2Bytes
//gsmSerializeNumbers
//gsmDecode7bit
//gsmDecodeUcs2


void CSendmessageDlg::OnTimer(UINT nIDEvent) 
{
	// TODO: Add your message handler code here and/or call default
	if (nIDEvent==1)
	{
		receiveMessage();
	}
	CDialog::OnTimer(nIDEvent);
}

void CSendmessageDlg::OnOK() 
{
	// TODO: Add extra validation here
	UpdateData(TRUE);
	if (m_strPhone.GetLength()==0)
	{
		MessageBox("请输入电话号码!");
	}
	else if (m_strContent.GetLength()==0)
	{
		MessageBox("发送内容不能为空!");
	}
	else
	{
		
		OnSendMessage();
		MessageBox("发送短信成功");
	}
//	OnSendMessage();
//	CDialog::OnOK();
}

void CSendmessageDlg::OnSendMessage()
{
	SM_PARAM * smparm=new SM_PARAM;
//	char sca[]  = "8613800100500";//北京移动
	char sca[]  = "8613010112500";//北京连通
	
	strcpy(smparm->SCA, sca);

	UpdateData(TRUE);
//	char tpa[] = "8613810574104";
	CString strPhone = m_strPhone;
	strPhone.TrimLeft();
	strPhone.TrimRight();
	strPhone = "86" + strPhone;
	const char* tpa = strPhone;
	strcpy(smparm->TPA, tpa);

	smparm->TP_PID = 0;
	smparm->TP_DCS = GSM_UCS2;
	
	const char *ud = m_strContent;
	strcpy(smparm->TP_UD,ud);
	
	UpdateData(FALSE);
	gsmSendMessage(smparm);

	delete smparm;

}

BOOL CSendmessageDlg::gsmSendMessage(const SM_PARAM *pSrc)
{
    int nPduLength;        // PDU串长度
    unsigned char nSmscLength;    // SMSC串长度
    int nLength;           // 串口收到的数据长度
    char cmd[16];          // 命令串
    char pdu[512];         // PDU串
    char ans[128];         // 应答串


	CString at;
	at.Format("AT+CMGF=0\r");
	m_Com.SetOutput(COleVariant(at));

	VARIANT	MyInBuffer = m_Com.GetInput();
			
	CString	str = (CString)MyInBuffer.bstrVal;

    nPduLength = gsmEncodePdu(pSrc, pdu);    // 根据PDU参数,编码PDU串
    strcat(pdu, "\x01a");        // 以Ctrl-Z结束

    gsmString2Bytes(pdu, &nSmscLength, 2);    // 取PDU串中的SMSC信息长度
    nSmscLength++;        // 加上长度字节本身

    // 命令中的长度,不包括SMSC信息长度,以数据字节计
    sprintf(cmd, "AT+CMGS=%d\r", nPduLength / 2 - nSmscLength);    // 生成命令

//   WriteComm(cmd, strlen(cmd));    // 先输出命令串
//	m_Com.SetOutput(COleVariant(cmd, strlen(cmd)));
	m_Com.SetOutput(COleVariant(cmd));
			

	VARIANT MyInBuffer1= m_Com.GetInput();
			
	str = (CString)MyInBuffer1.bstrVal;
		//	MessageBox(str);
  
	memcpy(ans,str.GetBuffer(128),128);
		
	nLength=strlen(ans);
   // nLength = ReadComm(ans, 128);   // 读应答数据
 // VARIANT MyInBuffer = m_Com.GetInput();
    // 根据能否找到"\r\n> "决定成功与否
  // if(nLength == 4 && strncmp(ans, "\r\n> ", 4) == 0)
    {
       // m_Com.SetOutput(COleVariant(pdu, strlen(pdu)));
		m_Com.SetOutput(COleVariant(pdu));

	   //WriteComm(pdu, strlen(pdu));        // 得到肯定回答,继续输出PDU串
//	 VARIANT MyInBuffer = m_Com.GetInput();
        // VARIANT pDU
       VARIANT MyInBuffer2 = m_Com.GetInput();
			
		CString	str = (CString)MyInBuffer2.bstrVal;
		//	MessageBox(str);
  
     	memcpy(ans,str.GetBuffer(128),128);
		nLength=strlen(ans);
		//   nLength = ReadComm(ans, 128);       // 读应答数据

		// 根据能否找到"+CMS ERROR"决定成功与否
		// if(nLength > 0 && strncmp(ans, "+CMS ERROR", 10) != 0)
		// {
		//    return TRUE;
		//}
    }

	return TRUE;
   // return FALSE;
}

void CSendmessageDlg::OnCancel() 
{
	// TODO: Add extra cleanup here
	CDialog::OnCancel();
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -