找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 543|回复: 1
收起左侧

单片机pov彩灯旋转LED显示原理介绍与程序电路原理图设计

[复制链接]
ID:1164104 发表于 2025-12-4 23:09 | 显示全部楼层 |阅读模式
分享一个基于STC12C5A60S3单片机的pov彩灯设计,含代码
旋转LED显示原理介绍
如何让一列灯旋转起来感觉像稳定的字显示在空中呢?首先我们来分析下人的眼睛,其实人的眼睛非常好骗的,只要让电机扫描的快一点就行了,实际上肉眼在24帧/秒以上就不会看到闪烁的,所以要保证电机的速度能在一秒转24圈以上,这样的话人眼就觉得旋转的字很稳定很清晰的显示在空中了。
旋转LED旋转起来是一个圆,那么就需要有一个传感器来判断起点位置,有人用霍尔传感器,有人用红外对管,笔者觉得用红外对管来的便宜些,而且实现起来也容易。这个起点检测非常重要,单片机就是根据这个起点来判断是否要开始显示 数据的。如果起点检测不到 单片机就不开始显示。
如何让一列灯不断的送数据实现一个文字的显示呢?这个我们就要了解文字取模的原理了,这里以PC2002字幕软件为例,取一个16*16的中文字,见字幕选项设置:
从第一列开始向下每取8个点作为一个字节,如果最后不足8个点就补满8位。
   取模顺序是从低到高,即第一个点作为最低位。如*-------取为00000001
  
取模后如下表:
0xFF,0x7F,0xFF,0xBF,0x3F,0xC8,0xBF,0xFB,0xBF,0xFB,0xBF,0xEB,0x80,0x9B,0xB7,0xFB,0xB7,0xEB,0xB7,0x9B,0xB7,0xFB,0x37,0xF8,0xF7,0xEF,0xF7,0x1F,0xFF,0xFF,0xFF,0xFF,/*"点",0*/
现在我们知道16*16取模是一列一列取的了,一列有2个字节,一共16列,所以一个16*16的汉字就有32个字节,需要占用单片机的Code空间 32个字节。然后再结合硬件来分析,如下图:
一列灯16个 刚好对应16*16一个汉字的一列:2个字节,所以把取模到的数据依次送到P2口 和P4 P5口,这里硬件中P4 P5组成一个字节,所以显示程序如下:
for(i=0;i<16;i++) //送16列   显示 这里只显示一个字。
            {                  
              P2=zimo[i*2]; //送数据低位显示
              P4=(zimo[i*2+1]);//送数据高位显示 这里用了单片机P4 和P5口 是LQFP48脚才有的IO口
              P5=(zimo[i*2+1])>>4;//这里行和列 都是IO口独立驱动的LED
                DelayUs(200);  //延时让LED亮起来 每列延时的时间
                          P2=0XFF;
                          P4=P5=0XFF;
               
            }
在什么时候送显示呢?单片机IO一判断到 红外接收管接收到起点信号,就开始显示,显示完16列后等待下一次的起点信号。这样只要电机的速度够快就会稳定的把字显示字空中了。平面的文字显示同理。
如何让一组文字不断的移动?这就需要一个字幕计数器,旋转LED每旋转一圈,这个字幕计数器就加一,指向下一列,这样不断的刷新,感觉文字就在移动了,程序如下:
j就是字幕计数器,每转一圈j就会加1;
if(KEY==0) //红外接收管 判断起始位
        {
            j++;           
if(j>672) //根据显示的字数定义改数值672/16=42个字 显示完42个字后 重新开始
            {
                j=0 ;
            }           
for(i=j;i<128+j;i++) //每转一圈  前进一列  这里定义一圈中同时显示128/16=8个字,
            {                                                                             P2=zimo[i*2]; //送数据低位显示
             P4=(zimo[i*2+1]);//送数据高位显示  这里用了单片机P4 和P5口 是LQFP48脚才有的
             P5=(zimo[i*2+1])>>4;//这里行和列 都是IO口独立驱动的LED
                DelayUs(200);  //延时让LED亮起来 每列延时的时间
                         P2=0XFF;
                         P4=P5=0XFF;
               
            }
最后一点是供电的问题,旋转LED供电问题是比较麻烦的,这里我采用了无线供电方式,经过实践论证,功率很有限,需要改进的地方还很多,电路原理是把直流转成交流,然后经过初级线圈 ,最后次级线圈感应得电,经过整流滤波后给旋转部分供电,直流转交流部分电路是一个自激震荡电路。动手能力强的朋友经过改造电机电刷方法实现,如果有条件制作的欢迎用此种方式。





单片机源程序如下:
  1. /*************************************************************
  2.                       旋转LED综合功能程序 平面立体一起显示
  3. 单片机 STC12C5A32S2  
  4. 运用无线供电方法 旋转起来平静 (特别说明;旋转LED转接塑料需放点纸巾压进电机轴,这样旋转起来摩擦力才够大,够稳定。这个对旋转是否安静影响很大
  5.      一个就是配重的问题,利用两个铜柱可以配平衡,长铜柱固定在旋转方向那头,即后备电池正极旁边,短铜柱在另一边)        结构做的好旋转是非常静的,几乎是一台风扇!
  6. 字模软件用的PC2002
  7. 实现效果:1.立体显示旋转字 (通过 “暂停按键” 进行静态与移动显示8-29更新
  8.                   2.平面显示数字时钟 年月日 星期+\立体显示字(时钟可调 时钟芯片用DS1302 带后备电池 掉电时钟继续走)
  9.                   3.平面显示模拟表盘时钟  (如果表盘不全或者过多显示 请按配套的遥控器"快进"或者"快退"来调整单列显示时间 设置后参数保存到DS1302RAM 中 下次启用无需调整 2012-8-28)
  10.                   4.平面\立体显示字体模式 通过 “暂停按键” 进行静态与移动显示8-29更 新   
  11.                   5.立体下拉显示字+停顿
  12.                   6.两颗心闪动
  13.                   7.一颗心填满效果
  14.                   8.笑脸动画
  15.                   9.打印字效果 (打印一封写给亲爱的信)
  16.                  10.立体显示时间+显示文字
  17. 上述10种效果对应遥控器按键1、 2、 3、 4、 5、 6、 7、 8、 9 、0 原时钟复位按键 改到按键“Meun”

  18. 通过上位机改 立体显示字        平面显示字
  19. ***************************************************************/
  20. #include "NEW_8051.H"
  21. #include "task.h"
  22. #include <stddef.h>
  23. #define ucNumRows  224//224 //定义一周显示字个数 一个字16列 224/16=13个字
  24. #define uclineNumRows  104//定义立体显示时间中的立体显示文字

  25. sbit Led1=P3^0;//最里面LED
  26. sbit Led2=P3^1;
  27. sbit Led3=P3^4;

  28. uchar Delay=1 ;
  29. bit BIT_timeout=0;
  30. uint uitime=0;
  31. data uint uiicount=0;
  32. uint uiCountFontNumber=0;
  33. uint uiPrintFontNumber=0;
  34. uchar ucMode=1;
  35. data uint uiicountTwo=0;
  36. uchar BuffCount=0;
  37. uchar ucPCA0_TimeCount;
  38. data uint uiBaseAddress=0;
  39. data uint uiFontNumber=0;
  40. data uint uiFontNumber_P=0;

  41. data uchar ucNumRows_p=160;
  42. uchar DelayTimeCount=0;
  43. bit Stop_move=0;
  44. uchar Mod=0;
  45. uchar Buff[448]=0xFF;
  46. uchar ucLine_count=0;
  47. uchar Row;
  48. uchar k;
  49. bit Error=0;
  50. uchar Stop_time=0;
  51. uchar ucline=0;
  52. bit bDot=0;
  53. uchar code sw[8]={0XFE,0XFC,0XF8,0XF0,0XE0,0XC0,0X80,0X00};
  54. void delay_200ms(void);
  55. uchar code liushui[8]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};//流水扫描数据
  56. /***************模拟表盘变量**************/
  57. uint DISP_LINE=0;
  58. uchar DISP_LINE_ADJ=0;
  59. uchar DISP_TIME_ADJ=0;
  60. uchar Time0_count=0;
  61. //====参数初始化
  62. uchar DISP_TIME_SET = 100 ; //单列显示时间设置值

  63. /********************************************/
  64. /*********延时子程序**********/
  65. void delay(unsigned char n)        //每列显示的时间
  66. {
  67.         unsigned char a,b,c;
  68.           c=2;//+DelayTimeCount;
  69.         for(;n>0;n--)
  70.         {            
  71.                 for(b=45;b>0;b--)
  72.         for(a=c;a>0;a--);
  73.         }
  74. }
  75. /*********延时子程序**********/
  76. void delay_PIC(unsigned char n)        //每列显示的时间
  77. {
  78.         unsigned char a,b,c;
  79.           c=2;//+DelayTimeCount;
  80.         for(;n>0;n--)
  81.         {            
  82.                 for(b=38;b>0;b--)
  83.         for(a=c;a>0;a--);
  84.         }
  85. }

  86. /*********PCA中断子程序**********/
  87. void PCA() interrupt 7        //PCA定时器
  88. {

  89.         if( CCF0 )
  90.         {
  91.                 CCF0 = 0;
  92.                 CL = 0;
  93.             CH = 0;
  94.                 ucPCA0_TimeCount++;
  95.                 if(ucPCA0_TimeCount>12)
  96.                 {
  97.                         ucPCA0_TimeCount=0;        //调时闪烁
  98.                         if(ucMode==2)
  99.                          {
  100.                                  flag=~flag;
  101.                          }
  102.                 }
  103.         }
  104. }
  105. /************************************/
  106. void Timer0Interrupt(void) interrupt 1
  107. {                        
  108.          TF0=0;
  109.                  TR0=0;
  110.           TH0 =(65536- DISP_TIME_SET)/256;
  111.           TL0 =(65536- DISP_TIME_SET)%256;
  112.           TR0=1;
  113.   //if ( ++Time0_count > 250 ) Time0_count = 250 ; //溢出计数器+1,>最大值,=最大值

  114.   if ( ++DISP_LINE > 180 ) DISP_LINE = 0 ; //显示列计数器+1,>最大值,=0
  115.                 /***********清空显示************/
  116.                 P1=0XFF ;
  117.             P0=0XFF ;
  118.                 Led1=0;
  119.                 Led2=1;
  120.                 Led3=1;
  121.                 /*************形成表盘*******************/
  122.            switch (DISP_LINE)   
  123.        {
  124.                 case 180*1/12 :    //1点
  125.                 case 180*2/12 :    //2点
  126.                 case 180*4/12 :    //4点
  127.                 case 180*5/12 :    //5点
  128.                 case 180*7/12 :    //6点
  129.                 case 180*8/12 :    //7点
  130.                 case 180*10/12 :   //10点
  131.                 case 180*11/12 :   //11点
  132.                   P1=0x3F;
  133.                  break;
  134.                         case 180*3/12 :    //3点
  135.                 case 180*6/12 :    //6点
  136.                 case 180*9/12 :    //9点
  137.                         case 0:    //12点
  138.                           P1=0x1F;
  139.                           break;
  140.        }
  141.    /****************************************/   
  142.              //显示指针"时"
  143. //           i = Hour ;
  144. //       j = Hour ;
  145. //      if ( ++i > 179 ) i=0 ;    //i+1,结果>179,则清零
  146. //      if ( --j > 179 ) i=179 ;   //j-1,结果为负,则重新赋值179
  147.         if(( (Hour-3)<=DISP_LINE)&&((Hour+3)>=DISP_LINE) )
  148.         {
  149.                 if(((Hour-2)==DISP_LINE)||((Hour+2)==DISP_LINE))
  150.                 P0=0XFB;
  151.                 else if(((Hour-3)==DISP_LINE)||((Hour+3)==DISP_LINE))
  152.                 P0=0XFD;
  153.                         
  154.         }
  155.            if(Hour==DISP_LINE)//||(DISP_LINE==i)||(DISP_LINE==j
  156.     {
  157.         P1=P1|0X1F ;
  158.         P0=0XF0 ;
  159.                Led1=0;
  160.                 Led2=0;
  161.                 Led3=0;
  162.     }   
  163.     //显示指针"分"
  164.         if(( (Cent-2)<=DISP_LINE)&&((Cent+2)>=DISP_LINE) )
  165.         {
  166.                 if(((Cent-1)==DISP_LINE)||((Cent+1)==DISP_LINE))
  167.                 P0=0XBF;
  168.                 else if(((Cent-2)==DISP_LINE)||((Cent+2)==DISP_LINE))
  169.                 P0=0XDF;
  170.                         
  171.         }
  172.         if(Cent==DISP_LINE)
  173.     {
  174.         P1=P1|0X1F ;
  175.         P0=0X00 ;
  176.         Led1=0;
  177.                 Led2=0;
  178.                 Led3=0;
  179.                
  180.     }
  181.         //显示指针"秒"
  182.         if(( (Sec-2)<=DISP_LINE)&&((Sec+2)>=DISP_LINE) )
  183.         {
  184.                 if(((Sec-1)==DISP_LINE)||((Sec+1)==DISP_LINE))
  185.                 P1=P1&0XF7 ;
  186.                 else if(((Sec-2)==DISP_LINE)||((Sec+2)==DISP_LINE))
  187.                 P1=P1&0XFB ;
  188.                         
  189.         }
  190.     if(Sec==DISP_LINE)
  191.     {
  192.         P1=P1&0XE0 ;
  193.         P0=0X00 ;
  194.                Led1=0;
  195.                 Led2=0;
  196.                 Led3=0;
  197.       
  198.     }

  199.         //}
  200. }


  201. /*********中断子程序**********/
  202. void int0() interrupt 0        //使用外部中断0
  203. {
  204.         EX0=0;
  205.         BIT_timeout=1;
  206.         if(ucMode==0x03)
  207.         {
  208.                 //调整单列显示时间设置值(模糊控制)
  209. //           if ( Time0_count > 183 )
  210. //           {
  211. //                        if ( Time0_count > 220 ) DISP_TIME_SET = DISP_TIME_SET + 4 ;
  212. //                        else if ( Time0_count > 200 ) DISP_TIME_SET = DISP_TIME_SET + 3 ;
  213. //                        else if ( Time0_count > 185 ) DISP_TIME_SET = DISP_TIME_SET + 2 ;
  214. //                        else DISP_TIME_SET = DISP_TIME_SET + 1 ;
  215. //                        
  216. //                        if ( DISP_TIME_SET > 2000 ) DISP_TIME_SET = 2000 ;  //钳位
  217. //               
  218. //                        }
  219. //                        else if ( Time0_count < 180 )
  220. //                        {
  221. //                        if ( Time0_count < 140 ) DISP_TIME_SET = DISP_TIME_SET - 4 ;
  222. //                        else if ( Time0_count < 160 ) DISP_TIME_SET = DISP_TIME_SET - 3 ;
  223. //                        else if ( Time0_count < 175 ) DISP_TIME_SET = DISP_TIME_SET - 2 ;
  224. //                        else DISP_TIME_SET = DISP_TIME_SET - 1 ;
  225. //                                if ( DISP_TIME_SET < 100 ) DISP_TIME_SET = 100 ;  //钳位
  226. //               
  227. //                        }
  228. //                        
  229.                 //        DISP_TIME_SET=DISP_TIME_SET+DISP_TIME_ADJ;
  230.                 //        Time0_count=0; //溢出计数器 = 0
  231.                         TR0=0;
  232.                         TH0 =(65536- DISP_TIME_SET)/256;
  233.                           TL0 =(65536- DISP_TIME_SET)%256;
  234.                         TR0=1;
  235.                     DISP_LINE =0;// DISP_LINE_ADJ ;   //显示列计数器(0~179)=校正值
  236.                          du1302();//获取时间
  237.                         Sec=59-(Sec/16*10+Sec%16) ; //由于电机反转 所以需要用最大值减去当前值
  238.                 Cent=59-(Cent/16*10+Cent%16) ;
  239.                     Hour=Hour/16*10+Hour%16 ;
  240.                         if(Hour>11)
  241.                         {
  242.                                 Hour=Hour-12;        
  243.                         }
  244.                         Hour=12-Hour;
  245.                         Sec=Sec*3;//秒针定位="秒"*3
  246.                         Hour=(Hour*15)+(Cent/4);//时针定位="时"*15+"分"/4
  247.                         Cent=Cent*3;//分针定位="分"*3        
  248.                         
  249.                 }
  250.                 EX0=1;        
  251. }
  252. /********************************/
  253. void start(void)
  254. {
  255.         P1=0XFF;
  256.         P2=0XFF;
  257.         P0=0XFF;
  258.         P3=0XFF;
  259.         P4=0XFF;
  260.         P5=0XFF;

  261. //        uiFontNumber=(byte_read(0x0100)<<8)+byte_read(0x0101);
  262.         uiFontNumber=((byte_read(0)<<8)+byte_read(1))*16; //读取EEP立体数据总数
  263.         uiPrintFontNumber=uiFontNumber*2;
  264.         uiFontNumber_P=((byte_read(0x3600)<<8)+byte_read(0x3601))*16; //读取EEP平面数据总数
  265.         //ucNumRows=R1302(ucNumRows_adder+1)*16;        //读立体一圈中显示字总数
  266. //        if(ucNumRows<16)
  267. //        {
  268. //                ucNumRows=224;        
  269. //        }
  270. //        ucNumRows_p=R1302(ucNumRows_p_adder+1)*16;
  271. //        if(ucNumRows_p<16)
  272. //        {
  273. //                ucNumRows_p=96;        
  274. //        }
  275.         //Send_data(uiFontNumber>>8);
  276.         //Send_data(uiFontNumber);
  277.         //Send_data(BCD2DEC(R1302(LINE_ADJ_adder+1)));
  278. //        DISP_LINE_ADJ=BCD2DEC(R1302(LINE_ADJ_adder+1));//读取列计数器校正值
  279. //        if(DISP_LINE_ADJ>20)//防止值过大
  280. //        {
  281. //                DISP_LINE_ADJ=0;        
  282. //        }
  283.         //write_1302Data(DISP_TIME_adder,160);

  284. //        DISP_LINE = DISP_LINE_ADJ ; //显示列计数器(0~179)=校正值

  285. //        DISP_TIME_SET=DISP_TIME_SET+ DISP_TIME_ADJ;
  286.         //DISP_LINE = DISP_LINE_ADJ ; //显示列计数器(0~179)=校正值  
  287.         //uiAllRows=uiAllRows-128;;

  288. }
  289. /***************************************/
  290. void OS_ALLRun(void)
  291. {        
  292.         
  293.         //BIT_timeout=1;
  294.         if(BIT_timeout)//起始点
  295.          {                          
  296.                 BIT_timeout=0;
  297.                 Error=1;

  298.                 if( new_code ) //有红外按键
  299.                 {
  300.                         new_code=0;
  301.                         
  302.                         //Send_data(key_code);
  303.                         switch( key_code ) //根据不同的按键值执行不同的动作
  304.                         {
  305.                                 case 0x07:
  306.                                         if(ucMode==3)
  307.                                         {                                                
  308.                                                 if(DISP_TIME_SET<250)
  309.                                                     {
  310.                                                                 DISP_TIME_SET++;                                                
  311.                                                                 write_1302Data(DISP_TIME_adder,DISP_TIME_SET);//写入DS1302 RAM中保存
  312.                                                         }        
  313.                                         }
  314.                                         else
  315.                                         {
  316.                                                  DelayTimeCount++;//单列延时时间+
  317.                                         }
  318.                                 break;
  319.         
  320.                                 case 0x09://
  321.                                         if(ucMode==3)
  322.                                         {
  323.                                                 if(DISP_TIME_SET>20)
  324.                                                 {
  325.                                                         DISP_TIME_SET--;                                       
  326.                                                         write_1302Data(DISP_TIME_adder,DISP_TIME_SET);//写入DS1302 RAM中保存               
  327.                                             }
  328.                                         }
  329.                                         else
  330.                                         {
  331.                                                 if(DelayTimeCount>0)//单列延时时间-
  332.                                                 DelayTimeCount--;
  333.                                         }
  334.                                 break;

  335.                            /*********************************/
  336.                                 case 0x0C:
  337.                                         ucMode=0x01;//立体显示字 移动效果+静态显示
  338.                                         CR=0;
  339.                                         TR0=0;
  340.                                         uiCountFontNumber=0;
  341.                                         Led1=1;
  342.                                         Led2=1;
  343.                                         Led3=1;
  344.                                 break;
  345.                                    case 0x1C:
  346.                                         ucMode=0x05;//立体显示字 下拉效果+静态显示
  347.                                         CR=0;
  348.                                         TR0=0;
  349.                                         Led1=1;
  350.                                         Led2=1;
  351.                                         Led3=1;
  352.                                         uiCountFontNumber=0;
  353.                                 break;
  354.                                  case 0x5A:
  355.                                         ucMode=0x06;//立体显示简单动画
  356.                                         CR=0;
  357.                                         TR0=0;
  358.                                         uiCountFontNumber=0;
  359.                                         Led1=1;
  360.                                         Led2=1;
  361.                                         Led3=1;
  362.                                 break;
  363.                                 case 0x42:
  364.                                         ucMode=0x07;//立体显示心型变化
  365.                                         CR=0;
  366.                                         TR0=0;
  367.                                         uiCountFontNumber=0;
  368.                                         Led1=1;
  369.                                         Led2=1;
  370.                                         Led3=1;
  371.                                 break;
  372.                                 case 0x52:
  373.                                         ucMode=0x08;//笑脸变化
  374.                                         CR=0;
  375.                                         TR0=0;
  376.                                         uiCountFontNumber=0;
  377.                                         Led1=1;
  378.                                         Led2=1;
  379.                                         Led3=1;
  380.                              break;
  381.                                 case 0x4A:                                       
  382.                                         CR=0;
  383.                                         TR0=0;
  384.                                         uiCountFontNumber=0;
  385.                                         ucMode=0x09;//打印字效果
  386.                                         ucLine_count=0;
  387.                                         Led1=1;
  388.                                         Led2=1;
  389.                                         Led3=1;
  390.                              break;
  391.                                 case 0x16:                                       
  392.                                         CR=0;
  393.                                         TR0=0;
  394.                                         uiCountFontNumber=0;
  395.                                         ucMode=0x0A;//立体显示时间 与文字
  396.                                         Led1=1;
  397.                                         Led2=1;
  398.                                         Led3=1;
  399.                              break;
  400.                                 case 0x18:                                       
  401.                                         TR0=0;
  402.                                         CR=0;
  403.                                         Auto_Set1302(starts_time);
  404.                                         ucMode=0x02;//数字钟模式
  405.                                         Led1=1;
  406.                                         Led2=1;
  407.                                         Led3=1;
  408.                                 break;

  409.                                 case 0x5E:
  410.                                 
  411.                                          DISP_TIME_SET=R1302(DISP_TIME_adder+1);//读取单列显示时间
  412.                                          //Send_data(DISP_TIME_SET);
  413.                                         if((DISP_TIME_SET>250)||(DISP_TIME_SET<20)) //防止值过大        过小
  414.                                         {
  415.                                                 DISP_TIME_SET=115;        
  416.                                         }
  417.                                         ucMode=0x03;//指针时钟模式
  418.                                         TR0=1;
  419.                                          CR=0;
  420.                                         Led1=0;
  421.                                 break;
  422.                                 case 0x08:
  423.                                         ucMode=0x04;//平面显示字模式
  424.                                         CR=0;
  425.                                         TR0=0;
  426.                                         uiCountFontNumber=0;
  427.                                         Led1=1;
  428.                                         Led2=1;
  429.                                         Led3=1;
  430.                                 break;
  431.                                 /*********************************/
  432.                                 case 0x40:         //调时加
  433.                                         if(ucMode==0x02)
  434.                                         {
  435.                                                 Set(id,1);
  436.                                                 
  437.                                         }
  438.                                         else if(ucMode==0x03)
  439.                                         {
  440.                                                 if(++DISP_LINE_ADJ>179)        //盘面校正值
  441.                                                 {
  442.                                                         DISP_LINE_ADJ=0;
  443.                                                         
  444.                                                 }
  445.                                                 write_1302Data(LINE_ADJ_adder,DISP_LINE_ADJ);//写入DS1302 RAM中保存        
  446.                                         }
  447.                                 break;

  448.                                 case 0x19: //调时减
  449.                                         if(ucMode==0x02)
  450.                                         {                                                
  451.                                                 Set(id,0);
  452.                                         }
  453.                                         else if(ucMode==0x03)
  454.                                         {
  455.                                                 if(--DISP_LINE_ADJ==0) //盘面校正值
  456.                                                 {
  457.                                                         DISP_LINE_ADJ=179;
  458.                                                         
  459.                                                 }
  460.                                           write_1302Data(LINE_ADJ_adder,DISP_LINE_ADJ);//写入DS1302 RAM中保存               
  461.                                         }

  462.                                 break;
  463.                                 
  464.                                 case 0x15://调时切换
  465.                                                 if(ucMode==2)
  466.                                                 {
  467.                                                         id++;                                                
  468.                                                         //CCAPM1 = 0x49; //开PCA0中断
  469.                                                         //CCAPM0 = 0x00; //关PCA1中断
  470.                                                         CR=1;//启动PCA计数
  471.                                                            if(id>4)
  472.                                                         {
  473.                                                             id=0;
  474.                                                                 CR=0;//关闭PCA计数
  475.                                                         //        CCAPM0 = 0x00;//关PCA0中断
  476.                                                         }
  477.                                                 }
  478.                                                 else if((ucMode==4)||(ucMode==5)||(ucMode==1))
  479.                                                 {
  480.                                                         Stop_move=~Stop_move;               
  481.                                                 }
  482.                                 break;
  483.                                         case 0x47://时钟初始化
  484.                                                 Set1302(starts_time);    //初始化
  485.                                                 W1302(0x8e,0x00); //控制命令,WP=0,写操作               
  486.                                                 W1302(0x90,0xa5); //打涞二级? 一个二级管串联一个2K玷
  487.                                                 //write_1302Data(DISP_TIME_adder,160);
  488.                                 break;

  489.                                 default: break;
  490.                         }
  491.                         key_code=0;
  492.                 }
  493.                 //DelayTime=TimeCount/(ucNumRows+60);        
  494.                 //TimeCount=0;        //延时周期计数清零
  495.                 switch (ucMode)
  496.                 {
  497.                         case 0x01://立体显示字移动模式
  498.                                 //uiCountFontNumber++;
  499.                                 Led3=1;
  500.                                 P1=0XFF;
  501.                                 P0=0XFF;
  502.                                         if(Stop_move==0) //是否暂停移动
  503.                                         {
  504.                                                 uiCountFontNumber++;        
  505.                                         }                                          
  506.                     if(uiCountFontNumber >uiFontNumber ) //uiFontNumber
  507.                     {
  508.                         
  509.                                         uiCountFontNumber = 0;
  510.                                        
  511.                     }            
  512.                     for( uiicount = uiCountFontNumber;uiicount<ucNumRows+uiCountFontNumber+32;uiicount++)
  513.                     {                                                               
  514.                        
  515.                                         P2=byte_read(2+uiicount*2);//读取内部EEPROM字幕数据 数据在2地址后 所以要加上2
  516.                                         P4=byte_read(2+uiicount*2+1);
  517.                                         P5=byte_read(2+uiicount*2+1)>>4;
  518.                                         delay(1);
  519.                                         P2=P4=P5=0XFF;
  520.                        if(BIT_timeout)
  521.                                    {
  522.                                             
  523.                                           break;
  524.                                    }
  525.                                 }

  526.                                 break;
  527.                         case 0X0A://立体显示时间 与文字
  528.                                  du1302();
  529.                                  Show_line_time();//立体显示时间
  530.                                  delay(48);
  531.                                  uiCountFontNumber++;        
  532.                                 if(uiCountFontNumber >uiFontNumber ) //uiFontNumber
  533.                     {
  534.                         
  535.                                         uiCountFontNumber = 0;
  536.                                        
  537.                     }            
  538.                     for( uiicount = uiCountFontNumber;uiicount<uclineNumRows+uiCountFontNumber;uiicount++)
  539.                     {                                                               
  540.                        
  541.                                         P2=byte_read(2+uiicount*2);//读取内部EEPROM字幕数据 数据在2地址后 所以要加上2
  542.                                         P4=byte_read(2+uiicount*2+1);
  543.                                         P5=byte_read(2+uiicount*2+1)>>4;
  544.                                         delay(1);
  545.                                         P2=P4=P5=0XFF;
  546.                        if(BIT_timeout)
  547.                                    {
  548.                                             
  549.                                           break;
  550.                                    }
  551.                                 }
  552.                         break;
  553.                         case 0x05://立体显示字下拉模式
  554.                                 //uiCountFontNumber++;
  555.                                 {
  556.                                 if(uiCountFontNumber >uiFontNumber ) //uiFontNumber
  557.                     {
  558.                         
  559.                                         uiCountFontNumber = 0;
  560.                                        
  561.                     }            
  562.                     for( uiicount = 0;uiicount<ucNumRows;uiicount++)
  563.                     {                                                               
  564.                        
  565.                                         P2=Buff[uiicount*2];//读取内部EEPROM字幕数据 数据在2地址后 所以要加上2
  566.                                         P4=Buff[uiicount*2+1];
  567.                                         P5=Buff[uiicount*2+1]>>4;
  568.                                         delay(1);
  569.                                         P2=P4=P5=0XFF;
  570.                        if(BIT_timeout)
  571.                                    {
  572.                                             
  573.                                           break;
  574.                                    }
  575.                                 }
  576.                                  
  577.                                 /****************************************/
  578.                             switch (Mod) //根据当前状态显示
  579.                                 {
  580.                                         case 0://下拉显示
  581.                                                 for(k=0;k<=ucNumRows;k++)
  582.                                                 {
  583.                                                         if(Row<8)
  584.                                                         {
  585.                                                                 Buff[k*2]=(byte_read(2+uiCountFontNumber+k*2))|(sw[Row]);
  586.                                                                 Buff[k*2+1]=0xFF;
  587.                                                         }
  588.                                                         else
  589.                                                         {
  590.                                                                 Buff[k*2]=(byte_read(2+uiCountFontNumber+k*2));
  591.                                                                 Buff[k*2+1]=(byte_read(2+uiCountFontNumber+k*2+1))|(sw[Row-8]);                                                        
  592.                                                         }        
  593.                                                 }
  594.                                                 break;
  595.                                         case 1: //静态显示
  596.                                         case 2:                                                
  597.                                                 break;
  598.                                         case 3:        //擦除
  599.                                                 for(k=0;k<=ucNumRows;k++)
  600.                                                 {
  601.                                                         if(Row<8)
  602.                                                         {
  603.                                                                 Buff[k*2]=(byte_read(2+uiCountFontNumber+k*2))|(~sw[Row]);
  604.                                                         }
  605.                                                         else
  606.                                                         {
  607.                                                                 Buff[k*2+1]=(byte_read(2+uiCountFontNumber+k*2+1))|(~sw[Row-8]);                                                        
  608.                                                         }        
  609.                                                 }                 
  610.                                         break;
  611.                                         default:
  612.                                                 break;
  613.                                 }
  614.                                 if(++Row>=16)
  615.                                 {
  616.                                         Row=0;
  617.                                         Mod++;
  618.                                         if(Mod>3)
  619.                                         {
  620.                                                 Mod=0;
  621.                                                 if(Stop_move==0) //是否暂停移动
  622.                                                 {
  623.                                                         uiCountFontNumber+=64;        
  624.                                                 }
  625.                                         }
  626.                                        
  627.                                 }
  628.                                 }                                                                           
  629.                                 break;
  630.                                 case 0x06://立体显示两颗心动画
  631.                                          for(k=0;k<4;k++)
  632.                                          {
  633.                                                 for( uiicount = 0;uiicount<32;uiicount++)
  634.                                     {                                                               
  635.                                        
  636.                                                         P2=Buff[uiicount*2];
  637.                                                         P4=Buff[uiicount*2+1];
  638.                                                         P5=Buff[uiicount*2+1]>>4;
  639.                                                         delay(2);
  640.                                                         P2=P4=P5=0XFF;
  641.                                        if(BIT_timeout)
  642.                                                    {
  643.                                                             
  644.                                                           break;
  645.                                                    }
  646.                                                   }
  647.                                            }
  648.                                         for(k=0;k<32;k++)
  649.                                         {
  650.                                        
  651.                                                 Buff[k*2]=flash[uiCountFontNumber+k*2];
  652.                                                 Buff[k*2+1]=flash[uiCountFontNumber+k*2+1];
  653.                                        
  654.                                         }
  655.                                         if(++Row>=8)
  656.                                         {
  657.                                                 Row=0;
  658.                                                 uiCountFontNumber+=64;
  659.                                                 if(uiCountFontNumber>=192)
  660.                                                         {uiCountFontNumber=0;}        
  661.                                                 
  662.                                         }                                       
  663.                                         break;
  664.                                 case 0x07://立体显示心形动画
  665.                                          for(k=0;k<4;k++)
  666.                                          {
  667.                                                 for( uiicount = 0;uiicount<32;uiicount++)
  668.                                     {                                                               
  669.                                        
  670.                                                         P2=Buff[uiicount*2];
  671.                                                         P4=Buff[uiicount*2+1];
  672.                                                         P5=Buff[uiicount*2+1]>>4;
  673.                                                         delay(2);
  674.                                                         P2=P4=P5=0XFF;
  675.                                        if(BIT_timeout)
  676.                                                    {
  677.                                                             
  678.                                                           break;
  679.                                                    }
  680.                                                   }
  681.                                            }
  682.                                         for(k=0;k<32;k++)
  683.                                         {
  684.                                        
  685.                                                 Buff[k*2]=xinxing[uiCountFontNumber+k*2];
  686.                                                 Buff[k*2+1]=xinxing[uiCountFontNumber+k*2+1];
  687.                                        
  688.                                         }
  689.                                         if(++Row>=8)
  690.                                         {
  691.                                                 Row=0;
  692.                                                 uiCountFontNumber+=64;
  693.                                                 if(uiCountFontNumber>=576)
  694.                                                         {uiCountFontNumber=0;}        
  695.                                                 
  696.                                         }                                       
  697.                                         break;
  698.                                 case 0x08://立体显示笑脸动画
  699.                                          for(k=0;k<4;k++)
  700.                                          {
  701.                                                 for( uiicount = 0;uiicount<32;uiicount++)
  702.                                     {                                                               
  703.                                        
  704.                                                         P2=Buff[uiicount*2];
  705.                                                         P4=Buff[uiicount*2+1];
  706.                                                         P5=Buff[uiicount*2+1]>>4;
  707.                                                         delay(2);
  708.                                                         P2=P4=P5=0XFF;
  709.                                        if(BIT_timeout)
  710.                                                    {
  711.                                                             
  712.                                                           break;
  713.                                                    }
  714.                                                   }
  715.                                            }
  716.                                         for(k=0;k<32;k++)
  717.                                         {
  718.                                        
  719.                                                 Buff[k*2]=xiaolian[uiCountFontNumber+k*2];
  720.                                                 Buff[k*2+1]=xiaolian[uiCountFontNumber+k*2+1];
  721.                                        
  722.                                         }
  723.                                         if(++Row>=15)
  724.                                         {
  725.                                                 Row=0;
  726.                                                 uiCountFontNumber+=64;
  727.                                                 if(uiCountFontNumber>=256)
  728.                                                         {uiCountFontNumber=0;}        
  729.                                                 
  730.                                         }                                       
  731.                                         break;
  732.                                 case 0x09://打印字效果
  733.                                 
  734.                                                 for( uiicount = 0;uiicount<176;uiicount++)
  735.                                     {                                                               
  736.                                        
  737.                                                         P2=Buff[uiicount*2];
  738.                                                         P4=Buff[uiicount*2+1];
  739.                                                         P5=Buff[uiicount*2+1]>>4;
  740.                                                         delay(1);
  741.                                                         P2=P4=P5=0XFF;
  742.                                        if(BIT_timeout)
  743.                                                    {
  744.                                                             
  745.                                                           break;
  746.                                                    }
  747.                                                   }
  748.                                                 
  749.                                                 ucline=0;
  750.                                         for(k=0;k<176;k++)
  751.                                         {
  752.                                        
  753.                                                 if(k<ucLine_count)
  754.                                                 {
  755.                                                         Buff[k*2]=byte_read(2+uiCountFontNumber+k*2);
  756.                                                         Buff[k*2+1]=byte_read(2+uiCountFontNumber+k*2+1);
  757.                                                 }
  758.                                                 else if(k<(ucLine_count+16))
  759.                                                 {
  760.                                                         
  761.                                                         if((bDot==1)&&(ucLine_count==160))
  762.                                                         {
  763.                                                                 Buff[k*2]=0XFF;
  764.                                                                 Buff[k*2+1]=0XFF;        
  765.                                                         }
  766.                                                         else
  767.                                                         {
  768.                                                                 Buff[k*2]=line[ucline++];
  769.                                                                 Buff[k*2+1]=line[ucline++];        
  770.                                                         }
  771.                                                 }
  772.                                                 else
  773.                                                 {
  774.                                                         
  775.                                                         Buff[k*2]=0XFF;
  776.                                                         Buff[k*2+1]=0XFF;        
  777.                                                 }
  778.                                        
  779.                                         }
  780.                                         if(++Row>=10)                                                
  781.                                         {
  782.                                                 Row=0;
  783.                                                 bDot=~bDot;
  784.                                                 if(ucLine_count<160)
  785.                                                 {
  786.                                                         ucLine_count=ucLine_count+16;
  787.                                                 }                                                        
  788.                                                 if(ucLine_count>=160)
  789.                                                         {
  790.                                                                 if(++Stop_time>4)
  791.                                                                 {
  792.                                                                         Stop_time=0;                                                               
  793.                                                                         ucLine_count=0;
  794.                                                                         uiCountFontNumber+=320;
  795.                                                                         if(uiCountFontNumber>=uiPrintFontNumber)
  796.                                                                         {
  797.                                                                                 uiCountFontNumber=0;        
  798.                                                                         }
  799.                                                                 }
  800.                                                         }        
  801.                                                 
  802.                                         }                                                                                
  803.                                         break;
  804.                         case 0x02:        //数字时钟模式
  805.                                 du1302();
  806.                                 Led1=1;
  807.                                 Led3=1;
  808.                                 if( ++uiCountFontNumber >uiFontNumber ) //uiFontNumber
  809.                     {
  810.                         
  811.                                         uiCountFontNumber = 0;
  812.                                        
  813.                     }
  814.                                  uiicount = uiCountFontNumber;                                 
  815.                                 display();
  816.                                 break;
  817.                         /*********************/
  818.                
  819.                         case 0x04:        //平面显示字体模式
  820.                                 Led3=1;
  821.                                 if(Stop_move==0)
  822.                                 {
  823.                                         uiCountFontNumber++;
  824.                                 }                                            
  825.                     if( uiCountFontNumber>uiFontNumber_P )
  826.                     {
  827.                         uiCountFontNumber=0 ;
  828.                     }            
  829.                     for(uiicountTwo=uiCountFontNumber;uiicountTwo<ucNumRows_p+uiCountFontNumber;uiicountTwo++)
  830.                     {                              
  831.                         P1=byte_read(0x3602+uiicountTwo*2);
  832.                         P0=byte_read(0x3603+uiicountTwo*2);
  833.                                         P2=byte_read(2+uiicountTwo*4);//读取内部EEPROM字幕数据 数据在2地址后 所以要加上2
  834.                                         P4=byte_read(3+uiicountTwo*4);
  835.                                         P5=byte_read(3+uiicountTwo*4)>>4;
  836.                                         delay(1);
  837.                                         P2=P4=P5=0XFF;
  838.                                         P2=byte_read(4+uiicountTwo*4);//读取内部EEPROM字幕数据 数据在2地址后 所以要加上2
  839.                                         P4=byte_read(5+uiicountTwo*4);
  840.                                         P5=byte_read(5+uiicountTwo*4)>>4;                                
  841.                                         delay(1);
  842.                                         P1=0xFF;
  843.                                         P0=0XFF;
  844.                                         P2=P4=P5=0XFF;                                
  845.                        if(BIT_timeout)
  846.                                    {
  847.                                           break;
  848.                                    }
  849.                                 }
  850.                                 break;
  851.                         default:
  852.                                 break;
  853.                 }
  854.                   
  855.         }
  856.         if(Error==0) //上电后如果红外对管 不正常 (无起点检测)会以流水灯形式跑起来,流水灯也可检测LED焊接是否正常
  857.         {
  858.                 for(k=0;k<8;k++)
  859.                         {                                
  860.                                 P0=~liushui[k];
  861.                                 P1=~liushui[k];
  862.                                 P2=~liushui[k];
  863.                                 P3=~liushui[k]|0x0F;
  864.                                 P4=~liushui[k];
  865.                                 P5=~liushui[k];               
  866.                             delay_200ms();

  867.                         }               
  868.         }
  869. }
复制代码

代码下载: 基于STC12的pov彩灯程序设计.7z (1.19 MB, 下载次数: 0)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的奖励!

查看全部评分

回复

使用道具 举报

ID:1070293 发表于 2025-12-16 18:38 | 显示全部楼层
看着不错,好资料,51黑有你更精彩!!!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表