在前面用C#开发完CRM的来电弹屏之后,有些客户有了新的要求,他们希望不但能够实现来电弹屏,更希望能够将呼入呼出的电话录音并上传到CRM服务器上,方便日后跟踪记录。于是便有了来电小秘书客户端的开发。 
  来电小秘书客户端的开发是基于纽曼USB来电通客户端的基础上进行开发的,由于纽曼USB来电通的硬件没有录音功能,于是硬件上使用了纽曼的另一个硬件产品来电小秘书,虽然是同一个厂家的产品,可是它们的API却是完全不兼容,更烦的是,来电小秘书API没有来电的回调接口,无法通过回调触发程序,也没有C#的Demo,很多功能只能通过一个不是那么详细的文档和一个Delphi的Demo摸索着做了,经历了一些挫折和困惑,终于完成了这个客户端程序。 
  首先,开发要做的就是与硬件的API进行沟通,依然通过C#的P/Invoke来完成,以下是来电小秘书的P/Invoke代码。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Runtime.InteropServices;
  5. namespace WindowsApplication1
  6. {
  7. class LDT1
  8. {
  9. [DllImport("usbms.dll", EntryPoint = "LoadDRV")]
  10. public static extern int LoadDRV();
  11. [DllImport("usbms.dll", EntryPoint = "EnableCard")]
  12. public static extern int EnableCard();
  13. [DllImport("usbms.dll", EntryPoint = "StopSigCheck")]
  14. public static extern int StopSigCheck(int Handle);
  15. [DllImport("usbms.dll", EntryPoint = "ReSetUsb")]
  16. public static extern int ReSetUsb(int Handle);
  17. [DllImport("usbms.dll", EntryPoint = "HangUp")]
  18. public static extern int HangUp(int Handle);
  19. [DllImport("usbms.dll", EntryPoint = "InitDtmfBuf")]
  20. public static extern int InitDtmfBuf(int Handle);
  21. [DllImport("usbms.dll", EntryPoint = "SetDialPara")]
  22. public static extern int SetDialPara(UInt16 RingBack1, UInt16 RingBack0, UInt16 BusyLen, UInt16 RingTimes, UInt16 SendNoSignalLen);
  23. [DllImport("usbms.dll", EntryPoint = "DisableCard")]
  24. public static extern int DisableCard();
  25. [DllImport("usbms.dll", EntryPoint = "FreeDRV")]
  26. public static extern int FreeDRV();
  27. [DllImport("usbms.dll", EntryPoint = "GetDtmfCode")]
  28. public static extern int GetDtmfCode(UInt16 Line);
  29. [DllImport("usbms.dll", EntryPoint = "IsRing")]
  30. public static extern bool IsRing(UInt16 Line);
  31. [DllImport("usbms.dll", EntryPoint = "GetCallerIDStr")]
  32. public static extern UInt16 GetCallerIDStr(UInt16 Line, StringBuilder IDStr);
  33. [DllImport("usbms.dll", EntryPoint = "IsOffHook")]
  34. public static extern bool IsOffHook(UInt16 Line);
  35. [DllImport("usbms.dll", EntryPoint = "StartRecordFile")]
  36. public static extern bool StartRecordFile(UInt16 Line, string FileName, UInt32 dwRecordLen);
  37. [DllImport("usbms.dll", EntryPoint = "CheckRecordEnd")]
  38. public static extern bool CheckRecordEnd(UInt16 Line);
  39. [DllImport("usbms.dll", EntryPoint = "StopRecordFile")]
  40. public static extern bool StopRecordFile(UInt16 Line);
  41. [DllImport("usbms.dll", EntryPoint = "PCMtoWave")]
  42. public static extern int PCMtoWave(string SourceFileName, string TargetFileName);
  43. [DllImport("usbms.dll", EntryPoint = "ReadCheckResult")]
  44. public static extern int ReadCheckResult(int line, int mode);
  45. [DllImport("usbms.dll", EntryPoint = "StartSigCheck")]
  46. public static extern void StartSigCheck(int line);
  47. [DllImport("usbms.dll", EntryPoint = "ReadUsbState")]
  48. public static extern bool ReadUsbState(int line);
  49. [DllImport("usbms.dll", EntryPoint = "GetRingNum")]
  50. public static extern int GetRingNum(int line);
  51. [DllImport("usbms.dll", EntryPoint = "InitRingNum")]
  52. public static extern void InitRingNum(int line);
  53. [DllImport("usbms.dll", EntryPoint = "ReadSerialNo")]
  54. public static extern int ReadSerialNo(int line,StringBuilder serialNo);
  55. }
  56. }

然后就是关于设备状态检测了,由于没有API直接支持来电回调,所以只能自己手动的检测设备状态来判断,要实现这一部分一般有两种方式,使用Timer或者使用Thread,Delphi的Demo中使用了Timer,可是Timer实现的弊端需要使用异步的思考方式,不符合我的思维模式,灵活度也不够,而且C#创建线程太方便了,而线程是通过同步方式思考的,所以使用了Thread模式。 
  然后在特定的时刻,记录电话号码、弹屏(如果是来电)、电话结束后录音和上传文件和信息到CRM服务器,其中来电号码可以很容易的获取,可是播出的号码获取就比较的麻烦了,C#中可以使用如下代码:

  1. while (LDT1.IsOffHook((ushort)this.line))
  2. {
  3. int temp = LDT1.GetDtmfCode((ushort)this.line);
  4. if (temp > 0)
  5. {
  6. phonenum = phonenum + this.convertInt(temp);
  7. }
  8. Thread.Sleep(300);
  9. }
  10. private string convertInt(int code)
  11. {
  12. string ret="";
  13. switch (code)
  14. {
  15. case 10:
  16. ret = "0";
  17. break;
  18. case 11:
  19. ret = "*";
  20. break;
  21. case 12:
  22. ret = "#";
  23. break;
  24. case 13:
  25. ret = "A";
  26. break;
  27. case 14:
  28. ret = "B";
  29. break;
  30. case 15:
  31. ret = "C";
  32. break;
  33. case 16:
  34. ret = "D";
  35. break;
  36. default:
  37. ret = code.ToString();
  38. break;
  39. }
  40. return ret;
  41. }

下面说一下C#中的大文件上传吧,网上有很多例子了,我参考了如下blog的代码进行开发http://www.cnblogs.com/bccu/archive/2009/01/05/1363771.html,可是无法上传成功,于是我读了一下代码,发现他将信息中的\r\n用空字符代替了,导致服务器无法识别,于是我更改了他的代码,解决了问题,代码如下:

  1. public static string UploadFileEx(string uploadfile, string url,
  2. string fileFormName, string contenttype, NameValueCollection querystring,
  3. CookieContainer cookies)
  4. {
  5. if ((fileFormName == null) ||
  6. (fileFormName.Length == 0))
  7. {
  8. fileFormName = "file";
  9. }
  10. if ((contenttype == null) ||
  11. (contenttype.Length == 0))
  12. {
  13. contenttype = "application/octet-stream";
  14. }
  15. string postdata;
  16. postdata = "?";
  17. if (querystring != null)
  18. {
  19. foreach (string key in querystring.Keys)
  20. {
  21. postdata += key + "=" + querystring.Get(key) + "&";
  22. }
  23. }
  24. Uri uri = new Uri(url + postdata);
  25. string boundary = "----------" + DateTime.Now.Ticks.ToString("x");
  26. HttpWebRequest webrequest = (HttpWebRequest)WebRequest.Create(uri);
  27. //webrequest.CookieContainer = cookies;
  28. webrequest.ContentType = "multipart/form-data; boundary=" + boundary;
  29. webrequest.Method = "POST";
  30. string huanhang = "\r\n";
  31. byte[] huanhangbyte = Encoding.UTF8.GetBytes(huanhang);
  32. // Build up the post message header
  33. StringBuilder sb = new StringBuilder();
  34. sb.Append("--");
  35. sb.Append(boundary);
  36. sb.Append("\r\n");
  37. sb.Append("Content-Disposition: form-data; name=\"");
  38. sb.Append(fileFormName);
  39. sb.Append("\"; filename=\"");
  40. sb.Append(Path.GetFileName(uploadfile));
  41. sb.Append("\"");
  42. sb.Append("\r\n");
  43. sb.Append("Content-Type: ");
  44. sb.Append(contenttype);
  45. sb.Append("\r\n");
  46. sb.Append("\r\n");
  47. string postHeader = sb.ToString();
  48. byte[] postHeaderBytes = Encoding.UTF8.GetBytes(postHeader);
  49. // Build the trailing boundary string as a byte array
  50. // ensuring the boundary appears on a line by itself
  51. byte[] boundaryBytes =
  52. Encoding.ASCII.GetBytes("--" + boundary + "");
  53. FileStream fileStream = new FileStream(uploadfile,
  54. FileMode.Open, FileAccess.Read);
  55. long length = postHeaderBytes.Length + fileStream.Length +
  56. boundaryBytes.Length + huanhangbyte.Length;
  57. webrequest.ContentLength = length;
  58. Stream requestStream = webrequest.GetRequestStream();
  59. // Write out our post header
  60. requestStream.Write(postHeaderBytes, 0, postHeaderBytes.Length);
  61. // Write out the file contents
  62. byte[] buffer = new Byte[checked((uint)Math.Min(4096,
  63. (int)fileStream.Length))];
  64. int bytesRead = 0;
  65. while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0)
  66. requestStream.Write(buffer, 0, bytesRead);
  67. requestStream.Write(huanhangbyte, 0, huanhangbyte.Length);
  68. // Write out the trailing boundary
  69. requestStream.Write(boundaryBytes, 0, boundaryBytes.Length);
  70. fileStream.Dispose();
  71. requestStream.Dispose();
  72. WebResponse responce = webrequest.GetResponse();
  73. Stream s = responce.GetResponseStream();
  74. StreamReader sr = new StreamReader(s);
  75. string retval=sr.ReadToEnd();
  76. sr.Dispose();
  77. if (File.Exists(uploadfile))
  78. {
  79. try
  80. {
  81. File.Delete(uploadfile);
  82. }catch(Exception e)
  83. {
  84. }
  85. }
  86. return retval;
  87. }

CRM来电小秘书客户端完成了,当然要配合这个功能,服务器端CRM系统也要做一些修改,不过不是这篇文章的主要内容,关于服务器端的修改的小节,就等下次再说吧。

使用C#开发纽曼USB来电小秘书客户端小结的更多相关文章

  1. 【项目实例】使用C#开发纽曼USB来电通来电弹屏客户端小结

    基于CRM客户和咨询者的普遍需求,老板决定在CRM系统上加入来电弹屏功能,所谓来电弹屏,就是当一个电话打入时,电脑会弹出该电话号码对应的客户.联系人或者供应商详细信息,如果是新号码,则添加一个新的客户 ...

  2. Android开发的16条小经验总结

    Android开发的16条小经验总结,希望对各位搞Android开发的朋友有所帮助. 1. TextView中的getTextSize返回值是以像素(px)为单位的, 而setTextSize()是以 ...

  3. 开发H5程序或者小程序的时候,后端Web API项目在IISExpress调试中使用IP地址,便于开发调试

    在我们开发开发H5程序或者小程序的时候,有时候需要基于内置浏览器或者微信开发者工具进行测试,这个时候可以采用默认的localhost进行访问后端接口,一般来说没什么问题,如果我们需要通过USB基座方式 ...

  4. TODO:即将开发的第一个小程序

    TODO:即将开发的第一个小程序 微信小程序是一种全新的连接用户与服务的方式,它可以在微信内被便捷地获取和传播,同时具有出色的使用体验.个人理解小程序是寄宿在微信平台上的一个前端框架,具有跨平台功能, ...

  5. 移动开发之【微信小程序】的原理与权限问题以及相关的简易教程

    这几天圈子里到处都在传播着这样一个东西,微信公众平台提供了一种新的开放能力,开发者可以快速开发一个小程序,取名曰:微信公众平台-小程序 据说取代移动开发安卓和苹果,那这个东东究竟是干吗用的?但很多人觉 ...

  6. 微信小程序的开发:通过微信小程序看前端

    前言 2016年9月22日凌晨,微信官方通过"微信公开课"公众号发布了关于微信小程序(微信应用号)的内测通知.整个朋友圈瞬间便像炸开了锅似的,各种揣测.介绍性文章在一夜里诞生.而真 ...

  7. 零基础开发一款微信小程序商城

    零基础开发一款微信小程序商城 一个朋友问我能不能帮忙做个商城?我一个完整网页都写不出的 菜鸟程序员,我该怎么拒绝呢?好吧,看在小程序这么火的形势下,我还是答应了!找了个开源项目,差不多花了三天时间搞定 ...

  8. 微信小程序开发教程 #043 - 在小程序开发中使用 npm

    本文介绍了如何在微信小程序开发中使用 npm 中包的功能,大大提高微信小程序的开发效率,同时也是微信小程序系列教程的视频版更新. 微信小程序在发布之初没有对 npm 的支持功能,这也是目前很多前端开发 ...

  9. .Net Core ORM选择之路,哪个才适合你 通用查询类封装之Mongodb篇 Snowflake(雪花算法)的JavaScript实现 【开发记录】如何在B/S项目中使用中国天气的实时天气功能 【开发记录】微信小游戏开发入门——俄罗斯方块

    .Net Core ORM选择之路,哪个才适合你   因为老板的一句话公司项目需要迁移到.Net Core ,但是以前同事用的ORM不支持.Net Core 开发过程也遇到了各种坑,插入条数多了也特别 ...

随机推荐

  1. [Everyday Mathematics]20150202

    设 $f:\bbR^2\to \bbR$ 为连续函数, 且满足条件 $$\bex f(x+1,y)=f(x,y+1)=f(x,y),\quad\forall\ (x,y)\in \bbR^2. \ee ...

  2. UVA 12532-Interval Product(BIT)

    题意: 给n个数字的序列,C v p 把v位上的数字置成p , S l r代表求区间[l,r]各数字相乘得数的符号,对于每个S输出所得符号(’+‘,’-‘,’0‘) 分析: 开两个数组表示区间负数的个 ...

  3. UIButton 设置为圆形,并且使用图片(UIImage)当做背景

    -(UIButton *)shareButtonWithIcon:(NSString *)iconName { UIButton *button = [UIButtonbuttonWithType:U ...

  4. Ye.云狐J2刷机笔记 | 完美切换内部存储卡和SD卡的改法.vold.fstab

    ================================================================================Ye.完美切换内部存储卡和SD卡成功.v ...

  5. TortoiseHg简单的入门使用说明

    参考资料: 互普的 TortoiseHg使用说明_百度文库 Mercurial(Hg)基本操作 - Tim Gong - 博客园 Mercurial与TortoiseHg使用入门教程(转) - mee ...

  6. libpomelo的cocos2d-x客户端使用总结

    这几天看了libpomelo的cocos2dx客户端这是个聊天室,由2个场景构成,登录场景LoginScene,聊天场景ChatScene. 一. LoginScene 客户端是以Login场景来启动 ...

  7. 网页抓取:PHP实现网页爬虫方式小结

    来源:http://www.ido321.com/1158.html 抓取某一个网页中的内容,需要对DOM树进行解析,找到指定节点后,再抓取我们需要的内容,过程有点繁琐.LZ总结了几种常用的.易于实现 ...

  8. C语言——递归练习

    1.炮弹一样的球状物体,能够堆积成一个金字塔,在顶端有一个炮弹,它坐落在一个4个炮弹组成的层面上,而这4个炮弹又坐落在一个9个炮弹组成的层面上,以此类推.写一个递归函数CannonBall,这个函数把 ...

  9. 机器学习中的数学(1)-回归(regression)、梯度下降(gradient descent)

    版权声明: 本文由LeftNotEasy所有,发布于http://leftnoteasy.cnblogs.com.如果转载,请注明出处,在未经作者同意下将本文用于商业用途,将追究其法律责任. 前言: ...

  10. 视频资源下载方法[download video resources]

    笔者做视频时钟分析,需要用到大量特殊的视频,如何获取需要的视频是一个问题? 以下载NBA视频为例: Tools:①腾讯视频软件 (自行下载)  ②批处理文件(下文会给出代码) convert.bat ...