標(biāo)題: TCS34725顏色傳感器實(shí)現(xiàn)色彩RGB還原(PWM)含電腦PC還原加12864液晶顯示 [打印本頁]

作者: 冰冰潔潔    時(shí)間: 2018-12-31 01:14
標(biāo)題: TCS34725顏色傳感器實(shí)現(xiàn)色彩RGB還原(PWM)含電腦PC還原加12864液晶顯示
弄了幾天終于實(shí)現(xiàn)了,就是傳感器照在什么顏色物質(zhì)上,RGB燈還還原出來,并上傳電腦,和12864液晶顯示器。哈哈哈哈,機(jī)器人有眼睛了。關(guān)鍵一點(diǎn)還是全開源。

制作出來的實(shí)物圖如下:


單片機(jī)源碼如下:
  1. #include <reg52.h>
  2. #include "usart.h"
  3. #include "iic.h"
  4. #include  <intrins.h>
  5. #define uchar unsigned char
  6. #define uint  unsigned int
  7. /*
  8. 硬件接法:
  9. GY-33---c51
  10. 1、GY-33_RX---c51_TX,c51復(fù)位將發(fā)送A5 01 A6 給模塊
  11. 2、c51_TX---FT232,STM32將數(shù)據(jù)上傳給上位機(jī)
  12. 3、GY-33_TX---c51_RX,接收模塊角度數(shù)據(jù)
  13. 軟件說明:
  14. 該程序采用串口方式獲取模塊rgb數(shù)據(jù),波特率9600
  15. 所以得用上位機(jī)先將模塊串口設(shè)置成9600,然后再進(jìn)行以上操作:
  16. 指令:A5 AE 53,模塊需復(fù)位生效
  17. 注:中斷函數(shù)位于stc_it.c
  18. */
  19. sfr T2MOD =0xC9;  //定時(shí)器中斷2TMOD必須定義




  20. /*-----------------------PWM定義---------------------------*/


  21. uchar PWM_T=0;//PWM脈沖周期PWM_T(max)=256*中斷定時(shí)時(shí)間(1ms)


  22. //PWM物理地址通道定義,物理地址根據(jù)需要自由定義
  23. sbit PWM_IO0 = P1^0;
  24. sbit PWM_IO1 = P1^1;
  25. sbit PWM_IO2 = P1^2;


  26. //PWM脈沖占空比
  27. uchar PWM_IO0_ZKB=0;  //脈沖占空比,0~255,=0時(shí)輸出最大
  28. uchar PWM_IO1_ZKB=0;
  29. uchar PWM_IO2_ZKB=0;


  30. /*---------------定時(shí)器2初始化-------------------*/
  31. void timer2_init()//定時(shí)器2初始化
  32. {
  33.         RCAP2H=0xff;  //16位計(jì)數(shù)器,定時(shí)39us
  34.         RCAP2L=0xdc;  //X=0xffdc=65500;T=(65536-X)*機(jī)器周期(11.0592MHz,1.085us)
  35.         //TH2=RCAP2H;     //會(huì)自動(dòng)重裝
  36.         //TL2=RCAP2L;
  37.         T2CON=0;
  38.         T2MOD=0;
  39.         EA=1;
  40.         ET2=1;
  41.         TR2=1;
  42. }


  43. /*--------------定時(shí)器2中斷服務(wù)程序----------------------------*/
  44. void timer2() interrupt 5
  45. {        
  46.         TF2=0; //注意:定時(shí)器2必須由軟件對溢出標(biāo)志位清零,
  47.                //這里與定時(shí)器0和定時(shí)器1不同
  48.         //PWM處理
  49.            PWM_T++;
  50.         if(PWM_T == 255)//取到最大值
  51.         {PWM_T = PWM_IO0 = PWM_IO1 = PWM_IO2  = 0;}        //所有通道脈沖拉至低電平
  52.         
  53.         //通道脈沖電平拉高
  54.         if(PWM_T == PWM_IO0_ZKB)  //通道0脈沖得高電平
  55.                 PWM_IO0 = 1;        
  56.         
  57.         if(PWM_T == PWM_IO1_ZKB)  //通道1脈沖得高電平
  58.                 PWM_IO1 = 1;
  59.         
  60.         if(PWM_T == PWM_IO2_ZKB)  //通道2脈沖得高電平
  61.         PWM_IO2 = 1;
  62.                
  63.         
  64. }










  65. void send_com(u8 datas)
  66. {
  67.         u8 bytes[3]={0};
  68.         bytes[0]=0xa5;
  69.         bytes[1]=datas;//功能字節(jié)
  70.         USART_Send(bytes,3);//發(fā)送幀頭、功能字節(jié)、校驗(yàn)和
  71. }
  72. typedef struct
  73. {
  74.     uint8_t Red;
  75.     uint8_t Green;
  76.     uint8_t Blue;
  77. } RGB;


  78. /***************************功能定義***************************/
  79. #define LCD12864_W_COM                        0xF8        //寫指令
  80. #define LCD12864_R_COM                        0xFC        //讀指令
  81. #define LCD12864_W_DAT                        0xFA        //寫數(shù)據(jù)
  82. #define LCD12864_R_DAT                        0xFE        //讀數(shù)據(jù)
  83. #ifndef __LCD12864_H__
  84. #define __LCD12864_H__
  85. #define LCD12864_COM 0        //輸入指令
  86. #define LCD12864_DAT 1        //輸入數(shù)據(jù)
  87. #define LCD12864_CLEAR                        0x01        //清除顯示
  88. #define LCD12864_CURSOR                        0x02        //位址歸位,游標(biāo)回原點(diǎn)
  89. #define LCD12864_CURSOR_RLF                0x04        //光標(biāo)左移,整體顯示不移動(dòng)
  90. #define LCD12864_CURSOR_RRF                0x06        //光標(biāo)右移,整體顯示不移動(dòng)
  91. #define LCD12864_CURSOR_RLN                0x05        //光標(biāo)左移,整體顯示移動(dòng)
  92. #define LCD12864_CURSOR_RRN                0x07        //光標(biāo)右移,整體顯示移動(dòng)
  93. #define LCD12864_DIS_OFF                0x08        //顯示狀態(tài)全關(guān)
  94. #define LCD12864_DIS_ON                        0x0E        //顯示狀態(tài)開
  95. #define LCD12864_DIS_CUR_ON                0x0E        //游標(biāo)顯示開
  96. #define LCD12864_DIS_CURS_ON        0x0F        //游標(biāo)位置顯示開
  97. #define LCD12864_CURSOR_RL                0x10        //游標(biāo)左移
  98. #define LCD12864_CURSOR_RR                0x14        //游標(biāo)右移
  99. #define LCD12864_CURDIS_RL                0x18        //整體顯示左移,光標(biāo)跟隨
  100. #define LCD12864_CURDIS_RR                0x1C        //整體顯示又移,光標(biāo)跟隨
  101. #define LCD12864_COM_BASIC                0x30        //基本指令集動(dòng)作
  102. #define LCD12864_COM_PLOT_ON        0x36        //擴(kuò)展指令集動(dòng)作,繪圖顯示ON
  103. #define LCD12864_COM_PLOT_OFF        0x34        //擴(kuò)展指令集動(dòng)作,繪圖顯示OFF
  104. #define LCD12864_STAND_TO                0x31        //進(jìn)入待命模式
  105. #define LCD12864_PLACE_ROLL                0x03        //允許輸入卷動(dòng)位址
  106. #define LCD12864_PLACE_IRAM                0x02        //允許輸入IRAM位址
  107. #define LCD12864_REVERES_13                0x04        //選擇一、三行同時(shí)作反白顯示
  108. #define LCD12864_REVERES_24                0x05        //選擇二、四行同時(shí)作反白顯示
  109. #define LCD12864_REST_ON                0x08        //進(jìn)入睡眠模式
  110. #define LCD12864_REST_OFF                0x0C        //脫離睡眠模式
  111. #define LCD12864_W_COM                        0xF8        //寫指令
  112. #define LCD12864_R_COM                        0xFC        //讀指令
  113. #define LCD12864_W_DAT                        0xFA        //寫數(shù)據(jù)
  114. #define LCD12864_R_DAT                        0xFE        //讀數(shù)據(jù)
  115. //======串口模式=======
  116. sbit LCD12864_CS = P2^7;   //片選信號輸入
  117. sbit LCD12864_STD = P2^6;  //串行數(shù)據(jù)輸入
  118. sbit LCD12864_SCLK = P2^5; //串行移位脈沖輸入
  119. sbit LCD12864_PSB = P2^4;  //串并模式選擇 0:串口,1:并口
  120. sbit LCDRST  = P2^3;
  121. unsigned char code tabe[10]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};//查表法//0-9數(shù)字
  122. /*****************************向12864寫入8位數(shù)據(jù)*************************************/
  123. void LCD12864_Wbyte(uchar byte)
  124. {
  125.         uchar i;


  126.         for(i=0;i<8;i++)
  127.                 {
  128.                 LCD12864_SCLK = 0;
  129.                 byte <<= 1;                         //左移待寫入數(shù)據(jù),使最高位移至CY
  130.                 LCD12864_STD = CY;        //寫入一位數(shù)據(jù)
  131.                 LCD12864_SCLK = 1;
  132.                 LCD12864_SCLK = 0;   
  133.                 }
  134. }
  135. /***************************從12864讀出8位數(shù)據(jù)***返回:sdat************************************/
  136. uchar LCD12864_Rbyte(void)
  137. {
  138.      uchar i,sdat,temp1,temp2;
  139.      temp1 = 0;
  140.      temp2 = 0;
  141. //==========12864的串行數(shù)據(jù)輸出形式是D7-D6-D5-D4-0-0-0-0-D3-D2-D1-D0-0-0-0-0,故要度兩個(gè)八位整合==============
  142.      for(i=0;i<8;i++)
  143.                 {
  144.                 temp1=temp1 << 1;
  145.                 LCD12864_SCLK = 0;
  146.                 LCD12864_SCLK = 1;            
  147.                 LCD12864_SCLK = 0;
  148.                 if( LCD12864_STD )        temp1|=0x01;        //tempe1:D7-D6-D5-D4-0-0-0-0
  149.                 }
  150.      for(i=0;i<8;i++)
  151.                 {
  152.                 temp2=temp2 << 1;
  153.                 LCD12864_SCLK = 0;
  154.                 LCD12864_SCLK = 1;            
  155.                 LCD12864_SCLK = 0;
  156.                 if( LCD12864_STD ) temp2|=0x01;   //tempe2:D3-D2-D1-D0-0-0-0-0
  157.                 }
  158.         sdat = ((temp1 & 0xf0)|(temp2 >> 4));  //取出D7\D6\D5\D4\D3\D2\D1\D0                    
  159.         return sdat;  //返回一個(gè)值(sdat)
  160. }
  161. /****************************讀忙狀態(tài)**************************************/
  162. void LCD12864_busy( void )
  163. {
  164.         do        LCD12864_Wbyte( LCD12864_R_COM );                //LCD12864_RW=11111,RW(1),RS(0),0
  165.         while( LCD12864_Rbyte() & 0x80 );         //讀BF,直到BF == 0
  166. }
  167. /*************************dat_com=0寫指令,dat_com=1寫數(shù)據(jù)*****************************************/
  168. void LCD12864_W (bit dat_com,uchar byte)
  169. {
  170.         uchar temp;
  171.         if(dat_com == 0)         //為零,寫入指令
  172.                 temp = LCD12864_W_COM;        //11111,RS(0),RW(0),0
  173.         else                            //否則,寫入數(shù)據(jù)
  174.                 temp = LCD12864_W_DAT;        //11111,RS(1),RW(0),0
  175.         LCD12864_CS = 1;                //片選使能
  176.         LCD12864_busy();                //LCD忙檢測
  177.         LCD12864_Wbyte(temp);        //寫入模式選擇,指令還是數(shù)據(jù)
  178.         LCD12864_Wbyte(byte&0xf0);        //寫入字節(jié)
  179.         LCD12864_Wbyte((byte<<4)&0xf0);        //寫入字節(jié)
  180.     LCD12864_CS = 0;                //片選關(guān)閉
  181. }
  182. /****************************LCD12864初始化**************************************/


  183. void LCD12864_Init(void)
  184. {
  185.         LCDRST=1;
  186.         LCD12864_PSB = 0;        //選擇串口模式
  187.         LCD12864_W(LCD12864_COM,0x30);        //基本指令功能.
  188.         LCD12864_W(LCD12864_COM,0x03);        //允許輸入卷動(dòng)位址
  189.         LCD12864_W(LCD12864_COM,0x0c);                //脫離隨眠狀態(tài),顯示打開,關(guān)光標(biāo),反白關(guān).
  190.         LCD12864_W(LCD12864_COM,0x01);                //清屏指令.
  191.         LCD12864_W(LCD12864_COM,0x06);        //AC自動(dòng)加一,光標(biāo)右移,整體顯示不移動(dòng)   
  192. }
  193. /**************************LCD12864字符位置自定義顯示****************************************/
  194. void LCD12864_WPoss(uchar x,uchar y,uchar *buff)
  195. {
  196.         uchar addr,i=0;
  197.         switch (x)
  198.                 {
  199.                 case 0:addr=0x80;break;
  200.                 case 1:addr=0x90;break;
  201.                 case 2:addr=0x88;break;
  202.                 case 3:addr=0x98;break;
  203.                 }
  204.         addr=addr+y;


  205.         LCD12864_W(LCD12864_COM,LCD12864_COM_BASIC);        //8BitMCU,基本指令集合
  206.         LCD12864_W(LCD12864_COM,addr);                                        //顯示在自定義位置


  207.         while(buff[i]!='\0')[/i][i]
  208.                 {        
  209.                 LCD12864_W(LCD12864_DAT,buff[i++]);
  210.                 if(++y==16)
  211.                         {
  212.                         y=0;
  213.                         if(++x==4) x=0;


  214.                         switch (x)
  215.                                 {
  216.                                 case 0:addr=0x80;break;
  217.                                 case 1:addr=0x90;break;
  218.                                 case 2:addr=0x88;break;
  219.                                 case 3:addr=0x98;break;
  220.                                 }
  221.                         addr=addr+y;
  222.                         LCD12864_W(LCD12864_COM,addr);
  223.                         }
  224.         
  225.                 }
  226. }
  227. void Disp_img(unsigned char  *img)        //圖形方式12864顯示字模221 橫向取膜
  228. {
  229.         unsigned char i,j;
  230.         unsigned int k = 0;


  231.         LCD12864_W(0,0x36); //圖形方式
  232.         for(i=0;i<32;i++)
  233.           {
  234.                 LCD12864_W(0,0x80+i);
  235.                 LCD12864_W(0,0x80);
  236.             for(j=0;j<16;j++)
  237.                 {
  238.                         LCD12864_W(1,img[k++]);
  239.                 }
  240.           }


  241.          for(i=0;i<32;i++)
  242.           {
  243.                 LCD12864_W(0,0x80+i);
  244.                 LCD12864_W(0,0x88);
  245.             for(j=0;j<16;j++)
  246.                 {
  247.                         LCD12864_W(1,img[k++]);
  248.                   }
  249.         }
  250. }
  251. /************************顯示1************************************/


  252. void display1(long temp)
  253. {
  254.     uchar dt1,dt2,dt3,dt4,dt5,dt6,dt7,dt8;
  255.         //temp=12345678;
  256.         dt1 = temp%10;//1位
  257.         dt2 = temp%100/10;//2位
  258.         dt3 = temp%1000/100;//3位
  259.         dt4 = temp%10000/1000;//4位


  260.         dt5 = temp%100000/10000;//5位
  261.         dt6 = temp%1000000/100000;//6位
  262.         dt7 = temp%10000000/1000000;//7位
  263.         dt8 = temp/10000000;//8位             最高位  


  264.         LCD12864_WPoss(1,0,"紅色R:");//第1行顯示表1
  265.     // LCD12864_W(0,0x80+3);//LCD12864_W ((0或1),****)  0寫指令  1寫數(shù)據(jù)


  266.     // LCD12864_W(1,tabe[dt8]);//6位
  267.      //LCD12864_W(1,tabe[dt7]);//5位
  268.     // LCD12864_W(1,tabe[dt6]);//4位
  269.     // LCD12864_W(1,tabe[dt5]);//3位
  270.          //LCD12864_W(1,0x2e);
  271.            //LCD12864_W(1,tabe[dt4]);//4位
  272.      LCD12864_W(1,tabe[dt3]);//3位
  273.            LCD12864_W(1,tabe[dt2]);//4位
  274.      LCD12864_W(1,tabe[dt1]);//3位
  275.      LCD12864_WPoss(1,7,"%");//第1行顯示表1
  276. }
  277. /************************顯示2************************************/


  278. void display2(long temp)
  279. {
  280.     uchar dt1,dt2,dt3,dt4,dt5,dt6,dt7,dt8;
  281.         //temp=12345678;
  282.         dt1 = temp%10;//1位
  283.         dt2 = temp%100/10;//2位
  284.         dt3 = temp%1000/100;//3位
  285.         dt4 = temp%10000/1000;//4位


  286.         dt5 = temp%100000/10000;//5位
  287.         dt6 = temp%1000000/100000;//6位
  288.         dt7 = temp%10000000/1000000;//7位
  289.         dt8 = temp/10000000;//8位             最高位  


  290.         LCD12864_WPoss(2,0,"綠色G:");//第1行顯示表1
  291.     // LCD12864_W(0,0x80+3);//LCD12864_W ((0或1),****)  0寫指令  1寫數(shù)據(jù)


  292.     // LCD12864_W(1,tabe[dt8]);//6位
  293.     // LCD12864_W(1,tabe[dt7]);//5位
  294.     // LCD12864_W(1,tabe[dt6]);//4位
  295.     // LCD12864_W(1,tabe[dt5]);//3位
  296.          //LCD12864_W(1,0x2e);
  297.           // LCD12864_W(1,tabe[dt4]);//4位
  298.      LCD12864_W(1,tabe[dt3]);//3位
  299.            LCD12864_W(1,tabe[dt2]);//4位
  300.      LCD12864_W(1,tabe[dt1]);//3位
  301.      LCD12864_WPoss(2,7,"%");//第1行顯示表1
  302. }
  303. /************************顯示3************************************/


  304. void display3(long temp)
  305. {
  306.     uchar dt1,dt2,dt3,dt4,dt5,dt6,dt7,dt8;
  307.         //temp=12345678;
  308.         dt1 = temp%10;//1位
  309.         dt2 = temp%100/10;//2位
  310.         dt3 = temp%1000/100;//3位
  311.         dt4 = temp%10000/1000;//4位


  312.         dt5 = temp%100000/10000;//5位
  313.         dt6 = temp%1000000/100000;//6位
  314.         dt7 = temp%10000000/1000000;//7位
  315.         dt8 = temp/10000000;//8位             最高位  


  316.         LCD12864_WPoss(3,0,"蘭色B:");//第1行顯示表1
  317.     // LCD12864_W(0,0x80+3);//LCD12864_W ((0或1),****)  0寫指令  1寫數(shù)據(jù)


  318.      //LCD12864_W(1,tabe[dt8]);//6位
  319.      //LCD12864_W(1,tabe[dt7]);//5位
  320.      //LCD12864_W(1,tabe[dt6]);//4位
  321.      //LCD12864_W(1,tabe[dt5]);//3位
  322.          //LCD12864_W(1,0x2e);
  323.           // LCD12864_W(1,tabe[dt4]);//4位
  324.      LCD12864_W(1,tabe[dt3]);//3位
  325.            LCD12864_W(1,tabe[dt2]);//4位
  326.      LCD12864_W(1,tabe[dt1]);//3位
  327.      LCD12864_WPoss(3,7,"%");//第1行顯示表1
  328. }


  329. int main(void)
  330. {


  331.     u8 sum=0,i=0;
  332.         RGB rgb={0,0,0};
  333.         Usart_Int(9600);
  334.         send_com(0x01);//發(fā)送讀方位角指令
  335.         
  336.         LCD12864_Init();        
  337.         LCD12864_WPoss(0,1,"TCS34725顏色");//第1行顯示表1
  338.         
  339.         timer2_init();//定時(shí)器2初始化
  340.         
  341.          while(1)
  342.         {
  343.                 if(Receive_ok)//串口接收完畢
  344.                 {
  345.                         for(sum=0,i=0;i<(rgb_data[3]+4);i++)//rgb_data[3]=3
  346.                         sum+=rgb_data[i];[/i][i]
  347.                         if(sum==rgb_data[i])//校驗(yàn)和判斷[/i][i]
  348.                         {
  349.                                 rgb.Red=rgb_data[4];
  350.                                 rgb.Green=rgb_data[5];
  351.                 rgb.Blue=rgb_data[6];
  352.                                 send_3out(&rgb_data[4],3,0x45);//上傳給上位機(jī)
  353.                                 
  354.                                 
  355.                                 display1(rgb.Red);//顯示紅色R:
  356.                                 display2(rgb.Green);//顯示綠色G:G:
  357.                                 display3(rgb.Blue);//顯示蘭色B:         
  358.                                 
  359.                         //通過TCS34725傳感器感受的光線進(jìn)行PWM三基色彩還原、、
  360.                                                 
  361.                                 PWM_IO0_ZKB = 255-rgb.Red; //紅色R:
  362.                                 PWM_IO1_ZKB = 255-rgb.Green;//綠色G:
  363.                                 PWM_IO2_ZKB = 255-rgb.Blue;//蘭色B:
  364.                                 
  365.                                 
  366.                                 
  367.                         }
  368.                         Receive_ok=0;//處理數(shù)據(jù)完畢標(biāo)志
  369.                 }        
  370.         }
  371. }
復(fù)制代碼

全部資料51hei下載地址:
顏色傳感器電腦端程序.zip (1.28 MB, 下載次數(shù): 152)
128顯串STC89TCS34725顏色OK2.zip (71.07 KB, 下載次數(shù): 183)



作者: 冰冰潔潔    時(shí)間: 2018-12-31 01:30
論壇也沒有找到TCS34725顏色傳感器相關(guān)參考,只能自己弄了,希望各位指點(diǎn)迷津,本程序只是實(shí)現(xiàn)了功能,電腦顏色還原和12864液晶顯示都很精彩,就是RGB燈有點(diǎn)不準(zhǔn),因?yàn)槭怯靡幻X的小RGB燈珠自己裝進(jìn)LED燈泡內(nèi)做成的,下一步打算用一個(gè)微型投影機(jī)RGB投影頭試試,有點(diǎn)不忍心拆呀,太浪費(fèi)了,不個(gè)要是做個(gè)機(jī)器人,給她看一下東西能用燈光還原并投影在你面前并加個(gè)語音模塊說出來我好喜歡你送的紅色玫瑰花。那一定會(huì)開心❤❤❤❤❤。。!支持我拆投影機(jī)的點(diǎn)贊。

作者: admin    時(shí)間: 2018-12-31 03:02
好資料,51黑有你更精彩!!!
作者: 壞孩子ii    時(shí)間: 2018-12-31 14:33
這個(gè)能完全測出顏色嗎
作者: 冰冰潔潔    時(shí)間: 2018-12-31 17:27
壞孩子ii 發(fā)表于 2018-12-31 14:33
這個(gè)能完全測出顏色嗎

能的,TCS34725是最新芯片,能還原RGB三基色,但是每個(gè)人的眼都有色差,但是我們應(yīng)用主要取值RGB數(shù)值分分辨率是255,不用太再意,色彩感受主要是要有好的環(huán)境,不過還可以通過軟件軟件濾波,這個(gè)幾十塊錢的東西能給出255的分辨率就不錯(cuò)了,先解決智能識別的有無問題吧,慢慢來,另外我們的RGB也要能支持256級PWM調(diào)節(jié)才行,所以我想到了千元級的投影機(jī),哈哈哈哈
作者: thanksbaby    時(shí)間: 2020-6-23 11:54
謝謝分享,下載學(xué)習(xí)
作者: hj213204    時(shí)間: 2021-1-29 20:34
樓主:您好?戳四愕娘@示顏色的圖片,以為是你自己寫的TCS34725 單片機(jī)端的程序,下載工程文件,里面沒看到有關(guān)任何TCS34725的程序?赡苁俏依斫忮e(cuò)了。你買的別人就識別號的開發(fā)板,只是串口接收RGB結(jié)果嗎???
作者: 馮666    時(shí)間: 2021-8-22 11:33
/PWM物理地址通道定義   這個(gè)怎么理解
作者: allan_qing    時(shí)間: 2022-3-13 17:14
好資料,謝謝你的提供




歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1