找回密码
 立即注册
首页 业界区 业界 [C#] 软硬结合第二篇——酷我音乐盒的逆天玩法 ...

[C#] 软硬结合第二篇——酷我音乐盒的逆天玩法

崔竹 2025-5-29 17:00:44
 
1、灵感来源:
LZ是纯宅男,一天从早上8:00起一直要呆在电脑旁到晚上12:00左右吧~平时也没人来闲聊几句,刷空间暑假也没啥动态,听音乐吧...~有些确实不好听,于是就不得不打断手头的工作去点击下一曲或是找个好听的歌来听...但是,[移动手锁定鼠标-->移动鼠标关闭当前页面选择音乐软件页面-->选择合适的音乐-->恢复原来的界面] 这一过程也会烦人不少,如果说软件的设计要在用户体验上做足功夫,感觉这一点是软件设计人员很难管住的方面,毕竟操作系统也就这样安排的嘛(当然,有些机智的开发人员加了几个热键,确实方便了不少!)。于是我想能不能设计一个软件能尽量少打断我们正常的工作简单操作去触发下一曲~
 
2、需求分析:

  • 下图左一是传统的操作模式,在这里要人的眼、手并用而且还必须等待记忆,可能我们平时感觉不到,但是这个过程却是比较浪费时间且分散注意力!
  • 下图右一是想改为的操作模式,在这里我们只需要外部触发(如:摇一下头或者微笑一下,甚至只要想一下就可以啦),让切歌任务在后台进行,这样就能不打断前台工作(这里的前台和后台只是当前工作窗口和非当前窗口,和专业的有差别!)
1.png
  
 
3、解决方案
根据上面分析我们需要这些条件:

  • 外部硬件设备,可以接收特殊信号并传给PC
  • PC上的软件能够读取硬件传来的信号并分析信息,做出切歌任务
结合我现有设备,做出如下方案:

  •   硬件采用STC89C52单片机最小系统占用P1.0和P1.1两个端口和超声波测距模块HC-SR04,通过根据遮挡物在超声波测距范围内停留的时间来发出触发“下一曲”,“暂停”,“上一曲”事件的信号。
       
2.png
   

  • 软件采用C#从串口读取单片机发送的触发事件信号消息,然后调用WinAPI对音乐盒窗口进行识别计算以及发送点击消息,来控制切换歌曲。
   
3.png

PS:这里根据手在超声波范围内停留的时间来分出3种信号:

  • 短暂停留在区域内-->下一曲信号
  • 稍长停留在区域内-->上一曲信号
  • 超长停留在区域内-->暂停信号
 
4、作品提前展示及相关介绍:
哈哈,秒懂啦吧!图中那个像望远镜的东西就是超声波测距模块,它的前面辐射状的空间(我设置为40cm)就是有效范围,那个黑色的像蜈蚣的东西就是单片机(就相当于电脑里的CPU),插在USB里面的不用介绍就是USB转TTL啦!主要就是负责采集传感器信号然后将距离信息通过USB发送给电脑。最终达到达到的效果是:你的手只要在区域内挥一下,就能切歌啦!手停长一点时间就能暂停啦!这个玩法没试过吧,哈哈!
4.jpeg

下面这个图就是基于C#的电脑端软件,其主要功能就是连接串口进行数据接收、数据处理、以及查找音乐盒的窗口、计算该点击的按钮位置、发出点击消息、在不同窗口中切换(因为要实现少打扰当前活动的目的)。这里为了测试方便所以加了3个功能按钮:上一曲、暂停、下一曲,通过点击这些按钮能实现控制酷我音乐盒歌曲的切换,然后右边加了个下拉框用来枚举当前可用串口,LINK按钮就是连接该串口的触发按钮。下面一个文本显示区是用来显示串口传过来的距离的数据的(便于调试哈~)
5.png

 
5、C#软件部分技术详解
该部分要用到很多Windows API,主要功能就是查找窗口句柄、控制窗口显示、计算窗口位置、聚焦窗口、窗口切换....算是把窗口有关的常用API都用上啦~此外,还用到了鼠标光标位置设定、鼠标点击消息发送最终达到模拟鼠标点击事件。当然,串口通信绝对不能少滴!
5.1、C#串口通信
5.1.1、获取当前可用串口列表
  1. 1 //Get all port list for selection
  2. 2 //获得所有的端口列表,并显示在列表内
  3. 3 PortList.Items.Clear();
  4. 4 string[] Ports = SerialPort.GetPortNames();
  5. 5
  6. 6 for (int i = 0; i < Ports.Length; i++)
  7. 7 {
  8. 8     string s = Ports[i].ToUpper();
  9. 9     Regex reg = new Regex("[^COM\\d]", RegexOptions.IgnoreCase | RegexOptions.Multiline);//正则表达式
  10. 10     s = reg.Replace(s, "");
  11. 11
  12. 12     PortList.Items.Add(s);
  13. 13 }
  14. 14 if (Ports.Length >1) PortList.SelectedIndex = 1;
复制代码

  • 调用串口要引用  using System.IO.Ports;  
  • 第9行的正则表达式要引用 using System.Text.RegularExpressions;
  • 第3行的PortList是那个下拉框; 
  • 整体的功能就是通过第4行的函数获取所有可用串口,然后加入下拉框显示,如果有可用的就把第一个选中;
5.1.2、串口连接按钮事件
  1. 1 private void btn_link_Click(object sender, EventArgs e)
  2. 2 {
  3. 3     if (!Connection.IsOpen)
  4. 4     {
  5. 5         //Start
  6. 6         Status = "正在连接...";
  7. 7         Connection = new SerialPort();
  8. 8         btn_link.Enabled = false;
  9. 9         Connection.PortName = PortList.SelectedItem.ToString();
  10. 10         Connection.Open();
  11. 11         Connection.ReadTimeout = 10000;
  12. 12         Connection.DataReceived += new SerialDataReceivedEventHandler(PortDataReceived);
  13. 13         Status = "连接成功";
  14. 14     }
  15. 15 }
复制代码
PS:整体很好理解就是把下拉框选中的串口号连接上,这里第12行比较重要,它调用SerialDataReceivedEventHandler(Func Name)来定义一个数据接收函数的句柄,这里PortDataReceived你可以随便写,但是接下来你要写对应的实现函数:(这里说句柄比较难理解,你就理解成一个函数,绑定串口的函数,一旦串口有数据发动过来就执行这个函数....)
  1. 1 //接收串口数据
  2. 2 private int num=0;          //障碍物进入范围的时间
  3. 3 private bool enter=false;   //是否有障碍物进入
  4. 4 private int signal=0;       //对每次进入范围的时间分段形成控制信号
  5. 5 private void PortDataReceived(object o, SerialDataReceivedEventArgs e)
  6. 6 {
  7. 7     int length = 1;
  8. 8     byte[] data = new byte[length];
  9. 9     Connection.Read(data, 0, length);
  10. 10     for (int i = 0; i < length; i++)
  11. 11     {
  12. 12         ReceivedData = string.Format("{0}",data[i]);
  13. 13     }
  14. 14
  15. 15     //数据滤波转换为控制信号
  16. 16     if (data[0] != 136 && !enter){      //当有障碍物进入时,传过来数据不是136并且是第一个
  17. 17         enter = true;
  18. 18         num = 1;
  19. 19     }else if (data[0] == 136 && enter){ //当障碍物离开时,传过来数据变为136且是第一个
  20. 20         enter = false;
  21. 21         if (num > 1 && num < 6){
  22. 22             signal = 1;
  23. 23         }else if (num > 5 && num < 10){
  24. 24             signal = 2;
  25. 25         }else if (num > 9){
  26. 26             signal = 3;
  27. 27         }
  28. 28         num = 0;
  29. 29     }else if (data[0] != 136 && data[0] >= 0 && enter){
  30. 30         num++;
  31. 31     }
  32. 32 }
复制代码
PS:这就是串口数据接收函数实现,先别看其他内容,因为里面涉及滤波算法和控制信号生成的算法,只要看第7~13行的代码核心部分就是第9行从缓冲区读取串口数据放到data[]数组中,这样串口数据就放在data[]中啦!怎么处理是下面的事啦~
5.1.3、重量级功能函数:
  1. 1 /// <summary>
  2. 2 /// 模拟鼠标点击函数
  3. 3 /// </summary>
  4. 4 /// <param name="n_control_type">0是上一曲,1是暂停,2是下一曲</param>
  5. 5 public void func(int n_control_type)
  6. 6 {
  7. 7     //bool isVisabled;                                      //窗口原来状态,隐藏还是显示
  8. 8     IntPtr hCurWin = GetForegroundWindow();                 //获取当前激活窗口
  9. 9
  10. 10     IntPtr hMusic = FindWindow("kwmusicmaindlg", null);     //找到窗口句柄
  11. 11     if (hMusic == null)
  12. 12     {
  13. 13         return;
  14. 14     }
  15. 15     Point pt;                                               //获取鼠标当前位置
  16. 16     GetCursorPos(out pt);
  17. 17     ShowWindow(hMusic,SW_SHOWNORMAL);         //如果是隐藏的就让他正常显示出来
  18. 18     SetForegroundWindow(hMusic);                            //将音乐盒窗口放在最上层
  19. 19
  20. 20     RECT rect = new RECT();                                 //获取窗口矩形
  21. 21     GetWindowRect(hMusic, ref rect);
  22. 22     int width = rect.Right - rect.Left;                     //窗口的宽度
  23. 23     int height = rect.Bottom - rect.Top;                    //窗口的高度
  24. 24     int x = rect.Right;                                     //窗口的位置
  25. 25     int y = rect.Top;
  26. 26
  27. 27     int X=0,Y=0;
  28. 28     if(n_control_type==0)//坐标[-20,200]:第3列表     [-120,200]:第2列表        [-220,200]第1列表
  29. 29     {                    //坐标[-200,100]:上一曲     [-170,100]暂停            [-145,100]下一曲
  30. 30         X = x - 200;
  31. 31         Y = y + 100;
  32. 32     }
  33. 33     else if (n_control_type == 1)
  34. 34     {
  35. 35         X = x - 170;
  36. 36         Y = y + 100;
  37. 37     }
  38. 38     else
  39. 39     {
  40. 40         X = x - 145;
  41. 41         Y = y + 100;
  42. 42     }
  43. 43                                                                    
  44. 44     SetCursorPos(X, Y);                                                             //移动鼠标
  45. 45     mouse_event(MOUSEEVENTF_LEFTDOWN, X * 65536 / 1024, X * 65536 / 768, 0, 0);     //发送鼠标信息
  46. 46     mouse_event(MOUSEEVENTF_LEFTUP, Y * 65536 / 1024, Y * 65536 / 768, 0, 0);
  47. 47     SetCursorPos(pt.X, pt.Y);                                                       //移动鼠标回到原位置
  48. 48
  49. 49     //if (isVisabled == 24) ShowWindow(hMusic, SW_HIDE);
  50. 50     //SetParent(hMusic, this.Handle);
  51. 51     //EnableWindow((IntPtr)this.Handle, true);               
  52. 52     SetWindowPos(hMusic, (IntPtr)this.Handle, x, y, width, height, SWP_NOMOVE);     //使能窗口聚焦原窗口
  53. 53     SetForegroundWindow(hCurWin);                                                   //将原来窗口放在最上层
  54. 54 }
复制代码
PS:这个函数负责找到酷我音乐盒的窗口(第10行)、顶层窗口切换(第18行、第52行、第53行)、鼠标位置设置(第16行、第44行、第47行)、鼠标点击消息的生成(第45行、第46行)、点击区域计算(第27~42行)

  • GetForegroundWindow(); 获取当前顶层窗口句柄,不懂百度一下,就windows API介绍很多,初学者知道怎么用就行啦![在调用它之前要写这些代码,下面说的调用API都要这样的!]
  1. 1 [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
  2. 2 public static extern IntPtr GetForegroundWindow();
复制代码

  • FindWindow("kwmusicmaindlg", null);根据窗口类名或者窗口名获得窗口句柄。PS:该如何知道某个窗口的类名或者窗口名呢?一般是用VC6.0或者是VS系列软件的Tool-->Spy++,具体请见我写的一篇博文,里面有详细介绍:http://www.cnblogs.com/zjutlitao/p/3889900.html
  1. 1 [DllImport("user32.dll", EntryPoint = "FindWindow")]
  2. 2 public static extern IntPtr FindWindow(
  3. 3     string lpClassName,
  4. 4     string lpWindowName
  5. 5 );
复制代码

  • GetCursorPos(out pt);获取当前鼠标的位置,保存在Point结构体内,这里因为我们想让鼠标点击一下按钮然后回到原来的位置,所以要保存原来的位置!
  1. 1 [DllImport("user32.dll")]
  2. 2 public static extern bool GetCursorPos(out Point pt);
复制代码

  • ShowWindow(hMusic,SW_SHOWNORMAL);根据句柄显示窗口,这里第二个参数是设定窗口以哪种方式显示的,主要有以最小化显示、最大化显示、正常显示.....具体参见度娘~我们这里是为了避免有时候音乐盒最小化,我们得把它打开才能触发点击事件有效。(我本来想用个标记来标记它原来的状态然后在处理之后恢复音乐盒自身的状态,但是觉得还得写些代码,没时间啦,调试这个浪费了很长时间~)
  1. 1 //private readonly int SW_HIDE = 0;     //隐藏
  2. 2 private readonly int SW_SHOWNORMAL = 1; //还原
  3. 3 [DllImport("user32.dll", EntryPoint = "ShowWindow", SetLastError = true)]
  4. 4 private static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);
复制代码

  • SetForegroundWindow(hMusic); 将活动窗口切换到句柄所指窗口,这样鼠标点击对应区域窗口才能接收到鼠标点击消息!
  1. 1 [DllImport("user32.dll")]
  2. 2 private static extern bool SetForegroundWindow(IntPtr hWnd);
复制代码

  • GetWindowRect(hMusic, ref rect); 获取指定窗口的在桌面上的矩形坐标(这样就能根据这个值计算目标窗口的大小和位置啦:20~25行就是干这个的)
  1. 1 [DllImport("user32.dll")]
  2. 2 [return: MarshalAs(UnmanagedType.Bool)]
  3. 3 static extern bool GetWindowRect(IntPtr hWnd, ref RECT lpRect);
  4. 4
  5. 5 [StructLayout(LayoutKind.Sequential)]
  6. 6 public struct RECT
  7. 7 {
  8. 8     public int Left; //最左坐标
  9. 9     public int Top; //最上坐标
  10. 10     public int Right; //最右坐标
  11. 11     public int Bottom; //最下坐标
  12. 12 }
复制代码

  • SetCursorPos(X, Y); 设置鼠标光标位置(X,Y)
  1. 1 [DllImport("user32.dll", EntryPoint = "SetCursorPos")]
  2. 2 private static extern int SetCursorPos(int x, int y);
复制代码

  • mouse_event(MOUSEEVENTF_LEFTDOWN, X * 65536 / 1024, X * 65536 / 768, 0, 0); 发送消息函数,我们知道windows是消息机制的,你点一下鼠标其实就是光标移到指定位置,然后向系统发送一个鼠标按动消息,这里我仿制一个鼠标左击时间,第45行负责在指定位置发送个鼠标左键按下的消息,第46行发送个对应的鼠标左键抬起的消息,这样一按一抬就组成了一个点击事件。
  1. 1 private readonly int MOUSEEVENTF_LEFTDOWN = 0x2;
  2. 2 private readonly int MOUSEEVENTF_LEFTUP = 0x4;
  3. 3 [DllImport("user32")]
  4. 4 public static extern void mouse_event(int dwFlags, int dx, int dy, int dwData, int dwExtraInfo);
复制代码

  • SetWindowPos(hMusic, (IntPtr)this.Handle, x, y, width, height, SWP_NOMOVE); 这个函数和ShowWindow有点像,只是这个可以设置窗口的三维显示,为什么是三维?平面窗口还有一维是窗口的叠放顺序,具体可以问度娘~(这里删了这句好像也没啥影响,当初因为没有下面那句,所以需要这个函数将焦点放到C#软件窗口)
  1. 1 static readonly IntPtr HWND_TOP = new IntPtr(0);
  2. 2 const UInt32 SWP_NOMOVE = 0x0002;
  3. 3 [System.Runtime.InteropServices.DllImport("user32.dll", EntryPoint = "SetWindowPos", SetLastError = true)]
  4. 4 private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int x, int y, int cx, int cy, uint uFlags);
复制代码
5.1.4、时间函数TImer
往窗口里加一个Timer控件:[长下面那个模样,属性设置为Interval:100,然后给它一个消息函数,属性中的那个闪电的标志],C#比MFC要方便的多,MFC要自己写这货,有点麻烦,但是对于打基础的童鞋还是建议从win32学起,然后再学MFC这样你对windows消息机制会有比较清晰的理解!嘿嘿,撤远啦!其实这个Timer对应的消息函数就像一个会定时执行的函数一样,你只要在里面写些逻辑,它会每隔一定的时间执行的。比如你想做动画效果,让一个小球移动,那么小球的坐标的改变的计算可以放在这里面写。下面看一下我的这个函数中写了什么:
6.png
  1. 1 private string Status, ReceivedData;
  2. 2 private void timer1_Tick(object sender, EventArgs e)
  3. 3 {
  4. 4     StatusMessage.Text = Status;
  5. 5     StatusMessage.Text = ReceivedData;
  6. 6     //当有有效信号过来触发控制
  7. 7     if (signal == 1) func(2);//下一曲
  8. 8     if (signal == 2) func(0);//上一曲
  9. 9     if (signal == 3) func(1);//暂  停
  10. 10     signal = 0;
  11. 11 }
复制代码
PS:其实就是更新那个文本显示区的内容和根据上面串口收来的数据进行处理然后产生的3种不同的控制命令,来调用func函数执行不同的点击命令!

>_30)             //超出测量或无效数据 72     { 73         flag=0; 74         SeriPushSend(0x88); 75     }     76     else 77     { 78         SeriPushSend(S); 79     } 80 } 81 /*-------------------------------------------- 82 毫秒延时函数 83 ---------------------------------------------*/  84 void delayms(unsigned int ms) 85 { 86     unsigned char i=100,j; 87     for(;ms;ms--) 88     { 89         while(--i) 90         { 91             j=10; 92             while(--j); 93         } 94     } 95 } 96 /*-------------------------------------------- 97 超声波测距中断函数[计时用] 98 ---------------------------------------------*/ 99 void zd0() interrupt 1          //T0中断用来计数器溢出,超过测距范围100 {101     flag=1;                             //中断溢出标志102 }103 /*--------------------------------------------104 超声波测距启动函数105 ---------------------------------------------*/106 void  StartModule()                  //T1中断用来扫描数码管和计800MS启动模块107 {108     TX=1;                             //800MS  启动一次模块109     _nop_(); 110     _nop_(); 111     _nop_(); 112     _nop_(); 113     _nop_(); 114     _nop_(); 115     _nop_(); 116     _nop_(); 117     _nop_(); 118     _nop_(); 119     _nop_(); 120     _nop_(); 121     _nop_(); 122     _nop_(); 123     _nop_(); 124     _nop_(); 125     _nop_();126     _nop_(); 127     _nop_(); 128     _nop_(); 129     _nop_();130     TX=0;131 }132 /*--------------------------------------------133 main函数134 ---------------------------------------------*/135 void main(void)136 {  137     USRT_init();138     while(1)139     {140         StartModule();141         while(!RX);        //当RX为零时等待142         TR0=1;                //开启计数143         while(RX);            //当RX为1计数并等待144         TR0=0;                //关闭计数145         Conut();            //计算146         delayms(10);        //10MS147     }148 }              [/code]TaoTao.c
>_ 1 && num < 6){2     signal = 1;3 }else if (num > 5 && num < 10){4     signal = 2;5 }else if (num > 9){6     signal = 3;7 }[/code]
7.png

 
7、总结:

哈哈,终于写完啦!>_
您需要登录后才可以回帖 登录 | 立即注册