oled.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975
  1. #include "oled.h"
  2. #include "oledfont.h"
  3. #include "delay.h"
  4. #include "QR_Encode.h"
  5. #include <stdarg.h> // 标准头文件
  6. #include "stdio.h"
  7. #include "w25qxx.h"
  8. //OLED的显存
  9. //存放格式如下.
  10. //[0]0 1 2 3 ... 127
  11. //[1]0 1 2 3 ... 127
  12. //[2]0 1 2 3 ... 127
  13. //[3]0 1 2 3 ... 127
  14. //[4]0 1 2 3 ... 127
  15. //[5]0 1 2 3 ... 127
  16. //[6]0 1 2 3 ... 127
  17. //[7]0 1 2 3 ... 127
  18. /**********************************************
  19. //IIC Start
  20. **********************************************/
  21. /**********************************************
  22. //IIC Start
  23. **********************************************/
  24. void IIC_Start()
  25. {
  26. OLED_SCLK_Set() ;
  27. OLED_SDIN_Set();
  28. OLED_SDIN_Clr();
  29. OLED_SCLK_Clr();
  30. }
  31. /**********************************************
  32. //IIC Stop
  33. **********************************************/
  34. void IIC_Stop()
  35. {
  36. OLED_SCLK_Set() ;
  37. // OLED_SCLK_Clr();
  38. OLED_SDIN_Clr();
  39. OLED_SDIN_Set();
  40. }
  41. void IIC_Wait_Ack()
  42. {
  43. //GPIOB->CRH &= 0XFFF0FFFF; //设置PB12为上拉输入模式
  44. //GPIOB->CRH |= 0x00080000;
  45. // OLED_SDA = 1;
  46. // delay_us(1);
  47. //OLED_SCL = 1;
  48. //delay_us(50000);
  49. /* while(1)
  50. {
  51. if(!OLED_SDA) //判断是否接收到OLED 应答信号
  52. {
  53. //GPIOB->CRH &= 0XFFF0FFFF; //设置PB12为通用推免输出模式
  54. //GPIOB->CRH |= 0x00030000;
  55. return;
  56. }
  57. }
  58. */
  59. OLED_SCLK_Set() ;
  60. OLED_SCLK_Clr();
  61. }
  62. /**********************************************
  63. // IIC Write byte
  64. **********************************************/
  65. void Write_IIC_Byte(unsigned char IIC_Byte)
  66. {
  67. unsigned char i;
  68. unsigned char m,da;
  69. da=IIC_Byte;
  70. OLED_SCLK_Clr();
  71. for(i=0;i<8;i++)
  72. {
  73. m=da;
  74. // OLED_SCLK_Clr();
  75. m=m&0x80;
  76. if(m==0x80)
  77. {OLED_SDIN_Set();}
  78. else OLED_SDIN_Clr();
  79. da=da<<1;
  80. OLED_SCLK_Set();
  81. OLED_SCLK_Clr();
  82. }
  83. }
  84. /**********************************************
  85. // IIC Write Command
  86. **********************************************/
  87. void Write_IIC_Command(unsigned char IIC_Command)
  88. {
  89. IIC_Start();
  90. Write_IIC_Byte(0x78); //Slave address,SA0=0
  91. IIC_Wait_Ack();
  92. Write_IIC_Byte(0x00); //write command
  93. IIC_Wait_Ack();
  94. Write_IIC_Byte(IIC_Command);
  95. IIC_Wait_Ack();
  96. IIC_Stop();
  97. }
  98. /**********************************************
  99. // IIC Write Data
  100. **********************************************/
  101. void Write_IIC_Data(unsigned char IIC_Data)
  102. {
  103. IIC_Start();
  104. Write_IIC_Byte(0x78); //D/C#=0; R/W#=0
  105. IIC_Wait_Ack();
  106. Write_IIC_Byte(0x40); //write data
  107. IIC_Wait_Ack();
  108. Write_IIC_Byte(IIC_Data);
  109. IIC_Wait_Ack();
  110. IIC_Stop();
  111. }
  112. void OLED_WR_Byte(unsigned dat,unsigned cmd)
  113. {
  114. if(cmd)
  115. {
  116. Write_IIC_Data(dat);
  117. }
  118. else {
  119. Write_IIC_Command(dat);
  120. }
  121. }
  122. /********************************************
  123. // fill_Picture
  124. ********************************************/
  125. void fill_picture(unsigned char fill_Data)
  126. {
  127. unsigned char m,n;
  128. for(m=0;m<8;m++)
  129. {
  130. OLED_WR_Byte(0xb0+m,0); //page0-page1
  131. OLED_WR_Byte(0x00,0); //low column start address
  132. OLED_WR_Byte(0x10,0); //high column start address
  133. for(n=0;n<128;n++)
  134. {
  135. OLED_WR_Byte(fill_Data,1);
  136. }
  137. }
  138. }
  139. /***********************Delay****************************************/
  140. void Delay_50ms(unsigned int Del_50ms)
  141. {
  142. unsigned int m;
  143. for(;Del_50ms>0;Del_50ms--)
  144. for(m=6245;m>0;m--);
  145. }
  146. void Delay_1ms(unsigned int Del_1ms)
  147. {
  148. unsigned char j;
  149. while(Del_1ms--)
  150. {
  151. for(j=0;j<123;j++);
  152. }
  153. }
  154. //坐标设置
  155. void OLED_Set_Pos(unsigned char x, unsigned char y)
  156. {
  157. OLED_WR_Byte(0xb0+y,OLED_CMD);
  158. OLED_WR_Byte(((x&0xf0)>>4)|0x10,OLED_CMD);
  159. OLED_WR_Byte((x&0x0f),OLED_CMD);
  160. }
  161. //开启OLED显示
  162. void OLED_Display_On(void)
  163. {
  164. OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令
  165. OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON
  166. OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON
  167. }
  168. //关闭OLED显示
  169. void OLED_Display_Off(void)
  170. {
  171. OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令
  172. OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF
  173. OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF
  174. }
  175. //清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!
  176. void OLED_Clear(void)
  177. {
  178. u8 i,n;
  179. for(i=0;i<8;i++)
  180. {
  181. OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7)
  182. OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址
  183. OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址
  184. for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA);
  185. } //更新显示
  186. }
  187. void OLED_On(void)
  188. {
  189. u8 i,n;
  190. for(i=0;i<8;i++)
  191. {
  192. OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7)
  193. OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址
  194. OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址
  195. for(n=0;n<128;n++)OLED_WR_Byte(1,OLED_DATA);
  196. } //更新显示
  197. }
  198. //在指定位置显示一个字符,包括部分字符
  199. //x:0~127
  200. //y:0~63
  201. //mode:0,反白显示;1,正常显示
  202. //size:选择字体 16/12
  203. void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 Char_Size)
  204. {
  205. unsigned char c=0,i=0;
  206. c=chr-' ';//得到偏移后的值
  207. if(x>Max_Column-1){x=0;y=y+2;}
  208. if(Char_Size ==16)
  209. {
  210. OLED_Set_Pos(x,y);
  211. for(i=0;i<8;i++)
  212. OLED_WR_Byte(F8X16[c*16+i],OLED_DATA);
  213. OLED_Set_Pos(x,y+1);
  214. for(i=0;i<8;i++)
  215. OLED_WR_Byte(F8X16[c*16+i+8],OLED_DATA);
  216. }
  217. else {
  218. OLED_Set_Pos(x,y);
  219. for(i=0;i<6;i++)
  220. OLED_WR_Byte(F6x8[c][i],OLED_DATA);
  221. }
  222. }
  223. //m^n函数
  224. u32 oled_pow(u8 m,u8 n)
  225. {
  226. u32 result=1;
  227. while(n--)result*=m;
  228. return result;
  229. }
  230. //显示2个数字
  231. //x,y :起点坐标
  232. //len :数字的位数
  233. //size:字体大小
  234. //num:数值(0~4294967295);
  235. void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size2)
  236. {
  237. u8 t,temp;
  238. u8 enshow=0;
  239. for(t=0;t<len;t++)
  240. {
  241. temp=(num/oled_pow(10,len-t-1))%10;
  242. if(enshow==0&&t<(len-1))
  243. {
  244. if(temp==0)
  245. {
  246. OLED_ShowChar(x+(size2/2)*t,y,' ',size2);
  247. continue;
  248. }else enshow=1;
  249. }
  250. OLED_ShowChar(x+(size2/2)*t,y,temp+'0',size2);
  251. }
  252. }
  253. /*显示一个字符号串
  254. x 取0 2 4 6 分为4行 y 每一个数字字母占9
  255. 汉字占18 Size字体的大小 可取12或者16
  256. */
  257. void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 Char_Size)
  258. {
  259. unsigned char j=0;
  260. while (chr[j]!='\0')
  261. { OLED_ShowChar(x,y,chr[j],Char_Size);
  262. x+=8;
  263. if(x>120){x=0;y+=2;}
  264. j++;
  265. }
  266. }
  267. //显示汉字
  268. /*
  269. x 取0 2 4 6 分为4行 y 每一个数字字母占9
  270. 汉字占18 Size字体的大小 可取12或者16
  271. no为Hzk[]数组中的哪一个元素
  272. */
  273. void OLED_ShowCHinese(u8 x,u8 y,u8 no)
  274. {
  275. u8 t,adder=0;
  276. OLED_Set_Pos(x,y);
  277. for(t=0;t<16;t++)
  278. {
  279. OLED_WR_Byte(Hzk[2*no][t],OLED_DATA);
  280. adder+=1;
  281. }
  282. OLED_Set_Pos(x,y+1);
  283. for(t=0;t<16;t++)
  284. {
  285. OLED_WR_Byte(Hzk[2*no+1][t],OLED_DATA);
  286. adder+=1;
  287. }
  288. }
  289. /***********
  290. 功能描述:
  291. 显示显示BMP图片128×64起始点坐标(x,y),
  292. x的范围0~127,
  293. y为页的范围0~7
  294. *****************/
  295. void OLED_DrawBMP(unsigned char x0, unsigned char y0,unsigned char x1, unsigned char y1,unsigned char BMP[])
  296. {
  297. unsigned int j=0;
  298. unsigned char x,y;
  299. if(y1%8==0) y=y1/8;
  300. else y=y1/8+1;
  301. for(y=y0;y<y1;y++)
  302. {
  303. OLED_Set_Pos(x0,y);
  304. for(x=x0;x<x1;x++)
  305. {
  306. OLED_WR_Byte(BMP[j++],OLED_DATA);
  307. }
  308. }
  309. }
  310. //初始化SSD1306
  311. void OLED_Init(void)
  312. {
  313. GPIO_InitTypeDef GPIO_InitStructure;
  314. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能A端口时钟
  315. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  316. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
  317. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
  318. GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化GPIOB
  319. GPIO_SetBits(GPIOB,GPIO_Pin_2);
  320. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF, ENABLE); //使能A端口时钟
  321. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  322. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
  323. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
  324. GPIO_Init(GPIOF, &GPIO_InitStructure); //初始化GPIOB
  325. GPIO_SetBits(GPIOF,GPIO_Pin_10);
  326. delay_ms(800);
  327. // OLED_WR_Byte(0xAE,OLED_CMD);//--display off
  328. // OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
  329. // OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
  330. // OLED_WR_Byte(0x40,OLED_CMD);//--set start line address
  331. // OLED_WR_Byte(0xB0,OLED_CMD);//--set page address
  332. // OLED_WR_Byte(0x81,OLED_CMD); // contract control
  333. // OLED_WR_Byte(0xFF,OLED_CMD);//--128
  334. // OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap
  335. // OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse
  336. // OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  337. // OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty
  338. // OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction
  339. // OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset
  340. // OLED_WR_Byte(0x00,OLED_CMD);//
  341. //
  342. // OLED_WR_Byte(0xD5,OLED_CMD);//set osc division
  343. // OLED_WR_Byte(0x80,OLED_CMD);//
  344. //
  345. // OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off
  346. // OLED_WR_Byte(0x05,OLED_CMD);//
  347. //
  348. // OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period
  349. // OLED_WR_Byte(0xF1,OLED_CMD);//
  350. //
  351. // OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion
  352. // OLED_WR_Byte(0x12,OLED_CMD);//
  353. //
  354. // OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh
  355. // OLED_WR_Byte(0x30,OLED_CMD);//
  356. //
  357. // OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable
  358. // OLED_WR_Byte(0x14,OLED_CMD);//
  359. //
  360. // OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel
  361. // OLED_WR_Byte(0xAE,OLED_CMD);//--display off 对应这种显示汉字的方式用以下初始化代码 OLED_ShowCHinese(4,0,0);
  362. // OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
  363. // OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
  364. // OLED_WR_Byte(0x40,OLED_CMD);//--set start line address
  365. // OLED_WR_Byte(0xB0,OLED_CMD);//--set page address
  366. // OLED_WR_Byte(0x81,OLED_CMD); // contract control
  367. // OLED_WR_Byte(0xFF,OLED_CMD);//--128
  368. // OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap
  369. // OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse
  370. // OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  371. // OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty
  372. // OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction
  373. // OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset
  374. // OLED_WR_Byte(0x00,OLED_CMD);//
  375. //
  376. // OLED_WR_Byte(0xD5,OLED_CMD);//set osc division
  377. // OLED_WR_Byte(0x80,OLED_CMD);//
  378. //
  379. // OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off
  380. // OLED_WR_Byte(0x05,OLED_CMD);//
  381. //
  382. // OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period
  383. // OLED_WR_Byte(0xF1,OLED_CMD);//
  384. //
  385. // OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion
  386. // OLED_WR_Byte(0x12,OLED_CMD);//
  387. //
  388. // OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh
  389. // OLED_WR_Byte(0x30,OLED_CMD);//
  390. //
  391. // OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable
  392. // OLED_WR_Byte(0x14,OLED_CMD);//
  393. //
  394. // OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel
  395. OLED_WR_Byte(0xAE,OLED_CMD); //关闭显示
  396. OLED_WR_Byte(0xD5,OLED_CMD); //设置时钟分频因子,震荡频率
  397. OLED_WR_Byte(80,OLED_CMD); //[3:0],分频因子;[7:4],震荡频率
  398. OLED_WR_Byte(0xA8,OLED_CMD); //设置驱动路数
  399. OLED_WR_Byte(0X3F,OLED_CMD); //默认0X3F(1/64)
  400. OLED_WR_Byte(0xD3,OLED_CMD); //设置显示偏移
  401. OLED_WR_Byte(0X00,OLED_CMD); //默认为0
  402. OLED_WR_Byte(0x40,OLED_CMD); //设置显示开始行 [5:0],行数.
  403. OLED_WR_Byte(0x8D,OLED_CMD); //电荷泵设置
  404. OLED_WR_Byte(0x14,OLED_CMD); //bit2,开启/关闭
  405. OLED_WR_Byte(0x20,OLED_CMD); //设置内存地址模式
  406. OLED_WR_Byte(0x02,OLED_CMD); //[1:0],00,列地址模式;01,行地址模式;10,页地址模式;默认10;
  407. OLED_WR_Byte(0xA1,OLED_CMD); //段重定义设置,bit0:0,0->0;1,0->127;
  408. OLED_WR_Byte(0xC0,OLED_CMD); //设置COM扫描方向;bit3:0,普通模式;1,重定义模式 COM[N-1]->COM0;N:驱动路数
  409. OLED_WR_Byte(0xDA,OLED_CMD); //设置COM硬件引脚配置
  410. OLED_WR_Byte(0x12,OLED_CMD); //[5:4]配置
  411. OLED_WR_Byte(0x81,OLED_CMD); //对比度设置
  412. OLED_WR_Byte(0xEF,OLED_CMD); //1~255;默认0X7F (亮度设置,越大越亮)
  413. OLED_WR_Byte(0xD9,OLED_CMD); //设置预充电周期
  414. OLED_WR_Byte(0xf1,OLED_CMD); //[3:0],PHASE 1;[7:4],PHASE 2;
  415. OLED_WR_Byte(0xDB,OLED_CMD); //设置VCOMH 电压倍率
  416. OLED_WR_Byte(0x30,OLED_CMD); //[6:4] 000,0.65*vcc;001,0.77*vcc;011,0.83*vcc;
  417. OLED_WR_Byte(0xA4,OLED_CMD); //全局显示开启;bit0:1,开启;0,关闭;(白屏/黑屏)
  418. OLED_WR_Byte(0xA6,OLED_CMD); //设置显示方式;bit0:1,反相显示;0,正常显示
  419. OLED_WR_Byte(0xAF,OLED_CMD); //开启显示
  420. }
  421. /**
  422. * OLED的显存
  423. * 存放格式如下.
  424. * [0]0 1 2 3 ... 127
  425. * [1]0 1 2 3 ... 127
  426. * [2]0 1 2 3 ... 127
  427. * [3]0 1 2 3 ... 127
  428. * [4]0 1 2 3 ... 127
  429. * [5]0 1 2 3 ... 127
  430. * [6]0 1 2 3 ... 127
  431. * [7]0 1 2 3 ... 127
  432. **/
  433. u8 OLED_GRAM[128][8];
  434. void OLED_Refresh_Gram(void)//更新显存到OLED
  435. {
  436. u8 i, n;
  437. for(i = 0; i < 8; i++)
  438. {
  439. OLED_WR_Byte(0xb0 + i,OLED_CMD); //设置页地址(0~7)
  440. OLED_WR_Byte(0x00,OLED_CMD); //设置显示位置—列低地址
  441. OLED_WR_Byte(0x10,OLED_CMD); //设置显示位置—列高地址
  442. for(n = 0; n < 128; n++)
  443. {
  444. OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA);
  445. }
  446. }
  447. }
  448. /**
  449. * @name void OLED_Clear(void)
  450. * @brief 清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!
  451. * @param none
  452. * @retval none
  453. */
  454. void MyOLED_Clear(void)
  455. {
  456. u8 i, n;
  457. for(i = 0; i < 8; i++)
  458. {
  459. for(n = 0; n < 128; n++)
  460. {
  461. OLED_GRAM[n][i] = 0X00;
  462. }
  463. }
  464. OLED_Refresh_Gram();//更新显示
  465. }
  466. /**
  467. * @name void OLED_DrawPoint(u8 x,u8 y,u8 t)
  468. * @brief 画点
  469. * @param x: 0~127 y: 0~63 t:1 填充 0 清空
  470. * @retval none
  471. */
  472. void OLED_DrawPoint(u8 x, u8 y, u8 t)
  473. {
  474. u8 pos, bx, temp = 0;
  475. if(x > 127 || y > 63)
  476. {
  477. return;//超出范围了.
  478. }
  479. pos = 7 - y/8;
  480. bx = y % 8;
  481. temp = 1 << (7 - bx);
  482. t ? (OLED_GRAM[x][pos] |= temp) : (OLED_GRAM[x][pos] &= ~temp);
  483. }
  484. /**
  485. * @name void OLED_DrawLine(u8 x, u8 y, u8 t)
  486. * @brief 画水平线
  487. * @param x0: 0~127 y: 0~63 x1: 0~127 t:1 填充 0 清空
  488. * @retval none
  489. */
  490. void OLED_DrawLine(u8 x0, u8 y, u8 x1, u8 t)
  491. {
  492. u8 x_0, x_1, i;
  493. if (x0 > x1) //判断x0与x1的大小
  494. {
  495. x_0 = x1;
  496. x_1 = x0;
  497. }
  498. else
  499. {
  500. x_0 = x0;
  501. x_1 = x1;
  502. }
  503. for (i = x_0 ; i <= x_1; i++) //依次画点
  504. {
  505. OLED_DrawPoint(i, y, t);
  506. }
  507. OLED_Refresh_Gram();//更新显示
  508. }
  509. /**
  510. * @name void OLED_DrawRow(u8 x0, u8 y, u8 x1, u8 t)
  511. * @brief 画竖直线
  512. * @param x: 0~127 y0: 0~63 y1: 0~63 t:1 填充 0 清空
  513. * @retval none
  514. */
  515. void OLED_DrawRow(u8 x, u8 y0, u8 y1, u8 t)
  516. {
  517. u8 y_0, y_1, i;
  518. if (y0 > y1) //判断y0与y1的大小
  519. {
  520. y_0 = y1;
  521. y_1 = y0;
  522. }
  523. else
  524. {
  525. y_0 = y0;
  526. y_1 = y1;
  527. }
  528. for (i = y_0 ; i <= y_1; i++) //依次画点
  529. {
  530. OLED_DrawPoint(x, i, t);
  531. }
  532. OLED_Refresh_Gram();//更新显示
  533. }
  534. /**
  535. * @name void OLED_DrawRow(u8 x0, u8 y, u8 x1, u8 t)
  536. * @brief 画矩形
  537. * @param x0: 0~127 y0: 0~63 x1: 0~127 y1: 0~63 t:1 填充 0 清空
  538. * @retval none
  539. */
  540. void OLED_DrawRectangle(u8 x0, u8 y0, u8 x1, u8 y1, u8 t)
  541. {
  542. u8 y_0, y_1, x_0, x_1, i;
  543. if (y0 > y1) //判断y0与y1的大小
  544. {
  545. y_0 = y1;
  546. y_1 = y0;
  547. }
  548. else
  549. {
  550. y_0 = y0;
  551. y_1 = y1;
  552. }
  553. if (x0 > x1) //判断x0与x1的大小
  554. {
  555. x_0 = x1;
  556. x_1 = x0;
  557. }
  558. else
  559. {
  560. x_0 = x0;
  561. x_1 = x1;
  562. }
  563. for (i = x_0 ; i <= x_1; i++) //依次画点
  564. {
  565. OLED_DrawPoint(i, y0, t);
  566. OLED_DrawPoint(i, y1, t);
  567. }
  568. for (i = y_0 ; i <= y_1; i++) //依次画点
  569. {
  570. OLED_DrawPoint(x0, i, t);
  571. OLED_DrawPoint(x1, i, t);
  572. }
  573. OLED_Refresh_Gram();//更新显示
  574. }
  575. /**
  576. * @name void OLED_Fill(u8 x0, u8 y0, u8 x1, u8 y1, u8 dot)
  577. * @brief 填充区域
  578. * @param x0,y0,x1,y1 填充区域的对角坐标
  579. * 确保x0<=x1;y0<=y1 0<=x0<=127 0<=y0<=63
  580. * dot:0,清空;1,填充
  581. * @retval none
  582. */
  583. void OLED_Fill2(u8 x0, u8 y0, u8 x1, u8 y1, u8 dot)
  584. {
  585. u8 x, y;
  586. for (x = x0; x < x1; x++)
  587. {
  588. for (y = y0; y < y1; y++)
  589. {
  590. OLED_DrawPoint(x, y, dot);
  591. }
  592. }
  593. // OLED_Refresh_Gram();
  594. }
  595. /**
  596. * @name u32 OLED_Pow(u8 m, u8 n)
  597. * @brief 求m^n
  598. * @param m:指数底 n:指数幂
  599. * @retval none
  600. */
  601. u32 OLED_Pow(u8 m, u8 n)
  602. {
  603. u32 result = 1;
  604. while (n--)
  605. {
  606. result *= m;
  607. }
  608. return result;
  609. }
  610. /**************************************************************************************************************
  611. 功能:在oled上显示二维码
  612. 参数:str->二维码内容; offset->二维码在X轴上的位置,范围为0-127; colour=1->二维码正显 colour=0->二维码反显
  613. **************************************************************************************************************/
  614. void OLED_QRcode_Display(char *str,uint8_t offset,uint8_t colour)//二维码的内容和第一个点再X轴的位置
  615. {
  616. uint32_t i,j,point;
  617. uint8_t exp = 1;//放大倍数
  618. uint8_t pos_X,pos_Y;
  619. if(colour)
  620. point = 1;
  621. else
  622. point = 0;
  623. EncodeData(str);
  624. exp = 64 / m_nSymbleSize; //根据屏幕尺寸自动计算最佳放大倍数
  625. pos_Y = (64 - exp*m_nSymbleSize)/2; //这是二维码左下角第一个点的纵坐标
  626. // exp = 128 / m_nSymbleSize; //根据屏幕尺寸自动计算最佳放大倍数
  627. // pos_Y = (128 - exp*m_nSymbleSize)/2; //这是二维码左下角第一个点的纵坐标
  628. pos_X = pos_Y + offset; //这是二维码左下角第一个点的横坐标
  629. if(point==0)
  630. OLED_Fill2(pos_X-2,pos_Y-2,pos_X + exp*m_nSymbleSize+2,pos_Y + exp*m_nSymbleSize+2,1);//给反显的二维码填充底色
  631. //exp*m_nSymbleSize为放大后二维码的边长(二维码是正方形)
  632. for(i=0;i<m_nSymbleSize;i++)
  633. {
  634. for(j=0;j<m_nSymbleSize;j++)
  635. {
  636. if(m_byModuleData[i][j] == 1)
  637. {
  638. OLED_Fill2(pos_X,pos_Y,pos_X+exp,pos_Y+exp,point);//画矩形并填充
  639. }
  640. if(m_byModuleData[i][j] == 0)
  641. {
  642. OLED_Fill2(pos_X,pos_Y,pos_X+exp,pos_Y+exp,1-point);//清空矩形区域
  643. }
  644. pos_Y += exp;
  645. }
  646. pos_X += exp;
  647. pos_Y -= m_nSymbleSize*exp;
  648. }
  649. OLED_Refresh_Gram();
  650. }
  651. void oled_printf(char x,char y,char *p,...)
  652. {
  653. char LcdBuff[16]={0};
  654. va_list ap;
  655. va_start(ap,p);
  656. vsprintf(LcdBuff,p,ap);
  657. va_end(ap);
  658. OLED_ShowString(x,y,(u8*)LcdBuff,16);
  659. //WriteLcd128_String(x,y,LcdBuff);
  660. }
  661. //画点
  662. //x:0~127
  663. //y:0~63
  664. //t:1 填充 0,清空
  665. void OLED_DrawPoint1(u8 x,u8 y,u8 t)
  666. {
  667. u8 pos,bx,temp=0;
  668. if(x>127||y>63)return;//超出范围了.
  669. pos=7-y/8;
  670. bx=y%8;
  671. temp=1<<(7-bx);
  672. if(t)OLED_GRAM[x][pos]|=temp;
  673. else OLED_GRAM[x][pos]&=~temp;
  674. }
  675. //u8 OLED_GRAM[128][8];
  676. //在指定位置显示一个字符,包括部分字符
  677. //x:0~127
  678. //y:0~63
  679. //mode:0,反白显示;1,正常显示
  680. //size:选择字体 12/16/24
  681. void OLED_ShowChar1(u8 x,u8 y,u8 chr,u8 size,u8 mode)
  682. {
  683. u8 temp,t,t1;
  684. u8 y0=y;
  685. u8 csize=(size/8+((size%8)?1:0))*(size/2); //得到字体一个字符对应点阵集所占的字节数
  686. chr=chr-' ';//得到偏移后的值
  687. for(t=0;t<csize;t++)
  688. {
  689. if(size==12)temp=asc2_1206_oled[chr][t]; //调用1206字体
  690. else if(size==16)temp=asc2_1608_oled[chr][t]; //调用1608字体
  691. else if(size==24)temp=asc2_2412_oled[chr][t]; //调用2412字体
  692. else return; //没有的字库
  693. for(t1=0;t1<8;t1++)
  694. {
  695. if(temp&0x80)OLED_DrawPoint1(x,y,mode);
  696. else OLED_DrawPoint1(x,y,!mode);
  697. temp<<=1;
  698. y++;
  699. if((y-y0)==size)
  700. {
  701. y=y0;
  702. x++;
  703. break;
  704. }
  705. }
  706. }
  707. // OLED_Refresh_Gram();
  708. }
  709. //显示字符串
  710. //x,y:起点坐标
  711. //size:字体大小
  712. //*p:字符串起始地址
  713. void OLED_ShowString1(u8 x,u8 y,const char *p,char size)
  714. {
  715. while((*p<='~')&&(*p>=' '))//判断是不是非法字符!
  716. {
  717. if(x>(128-(size/2))){x=0;y+=size;}
  718. if(y>(64-size)){y=x=0;OLED_Clear();}
  719. OLED_ShowChar1(x,y,*p,size,1);
  720. x+=size/2;
  721. p++;
  722. }
  723. }
  724. //更新显存到LCD
  725. void OLED_Refresh_Gram1(void)
  726. {
  727. u8 i,n;
  728. for(i=0;i<8;i++)
  729. {
  730. OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7)
  731. OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址
  732. OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址
  733. for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA);
  734. }
  735. }
  736. //x1,y1,x2,y2 填充区域的对角坐标
  737. //确保x1<=x2;y1<=y2 0<=x1<=127 0<=y1<=63
  738. //dot:0,清空;1,填充
  739. void OLED_Fill(u8 x1,u8 y1,u8 x2,u8 y2,u8 dot)
  740. {
  741. u8 x,y;
  742. for(x=x1;x<=x2;x++)
  743. {
  744. for(y=y1;y<=y2;y++)OLED_DrawPoint(x,y,dot);
  745. }
  746. OLED_Refresh_Gram();//更新显示
  747. }
  748. //void oled_printf(char x,char y,char *p,...)
  749. //{
  750. // char LcdBuff[20];
  751. // va_list ap;
  752. // va_start(ap,p);
  753. // vsprintf(LcdBuff,p,ap);
  754. // va_end(ap);
  755. // OLED_ShowString(x,y,LcdBuff,16);
  756. // //WriteLcd128_String(x,y,LcdBuff);
  757. //}
  758. //显示汉字
  759. /*
  760. x 取0 2 4 6 分为4行 y 每一个数字字母占9
  761. 汉字占18 Size字体的大小 可取12或者16
  762. no为Hzk[]数组中的哪一个元素
  763. */
  764. void OLED_ShowCHinese1(u8 x,u8 y,u8 no)
  765. {
  766. u8 t,adder=0;
  767. OLED_Set_Pos(x,y);
  768. for(t=0;t<16;t++)
  769. {
  770. // OLED_WR_Byte(Hzk[2*no][t],OLED_DATA);
  771. adder+=1;
  772. }
  773. OLED_Set_Pos(x,y+1);
  774. for(t=0;t<16;t++)
  775. {
  776. // OLED_WR_Byte(Hzk[2*no+1][t],OLED_DATA);
  777. adder+=1;
  778. }
  779. }
  780. //code 字符指针开始
  781. //从字库中查找出字模
  782. //code 字符串的开始地址,GBK码
  783. //mat 数据存放地址 (size/8+((size%8)?1:0))*(size) bytes大小
  784. //size:字体大小
  785. void OLED_Get_HzMat(unsigned char *code,unsigned char *mat,u8 size)
  786. {
  787. unsigned char qh,ql;
  788. unsigned char i;
  789. unsigned long foffset;
  790. u8 csize=(size/8+((size%8)?1:0))*(size);//得到字体一个字符对应点阵集所占的字节数
  791. qh=*code;
  792. ql=*(++code);
  793. if(qh<0x81||ql<0x40||ql==0xff||qh==0xff)//非 常用汉字
  794. {
  795. for(i=0;i<csize;i++)*mat++=0x00;//填充满格
  796. return; //结束访问
  797. }
  798. if(ql<0x7f)ql-=0x40;//注意!
  799. else ql-=0x41;
  800. qh-=0x81;
  801. foffset=((unsigned long)190*qh+ql)*csize; //得到字库中的字节偏移量
  802. switch(size)
  803. {
  804. case 12:
  805. // W25QXX_Read(mat,foffset+ftinfo.f12addr,csize);
  806. W25QXX_Read(mat,foffset+0,csize);//0-574560 实际大小574560字节 574560
  807. break;
  808. case 16:
  809. W25QXX_Read(mat,574560+foffset,csize);// foffset计算出来的偏移地址 574560->1340640 实际大小766080字节
  810. break;
  811. case 24:
  812. // W25QXX_Read(mat,foffset+ftinfo.f24addr,csize);
  813. break;
  814. }
  815. }
  816. /*显示单个汉字 */
  817. void Show_Font_Fury(u16 x,u16 y,u8 *font,u8 size,u8 mode)
  818. {
  819. u8 temp,t,t1;
  820. u16 y0=y;
  821. u8 dzk[72];
  822. u8 csize=(size/8+((size%8)?1:0))*(size);//得到字体一个字符对应点阵集所占的字节数
  823. if(size!=12&&size!=16&&size!=24)return; //不支持的size
  824. OLED_Get_HzMat(font,dzk,size); //得到相应大小的点阵数据
  825. for(t=0;t<csize;t++)
  826. {
  827. temp=dzk[t]; //得到点阵数据
  828. for(t1=0;t1<8;t1++)
  829. {
  830. if(temp&0x80)OLED_DrawPoint(x,y,1);
  831. else if(mode==0)OLED_DrawPoint(x,y,0);
  832. temp<<=1;
  833. y++;
  834. if((y-y0)==size)
  835. {
  836. y=y0;
  837. x++;
  838. break;
  839. }
  840. }
  841. }
  842. }
  843. /*显示汉字字符串 */
  844. void Show_Str_Fury(u16 x,u16 y,u16 width,u16 height,u8*str,u8 size)
  845. {
  846. u16 x0=x;
  847. u16 y0=y;
  848. u8 bHz=0; //字符或者中文
  849. while(*str!=0)//数据未结束
  850. {
  851. if(!bHz)
  852. {
  853. if(*str>0x80)bHz=1;//中文
  854. else //字符
  855. {
  856. if(x>(x0+width-size/2))//换行
  857. {
  858. y+=size;
  859. x=x0;
  860. }
  861. if(y>(y0+height-size))break;//越界返回
  862. if(*str==13)//换行符号
  863. {
  864. y+=size;
  865. x=x0;
  866. str++;
  867. }
  868. else OLED_ShowChar1(x,y,*str,size,1);//有效部分写入
  869. str++;
  870. x+=size/2; //字符,为全字的一半
  871. }
  872. }else//中文
  873. {
  874. bHz=0;//有汉字库
  875. if(x>(x0+width-size))//换行
  876. {
  877. y+=size;
  878. x=x0;
  879. }
  880. if(y>(y0+height-size))break;//越界返回
  881. Show_Font_Fury(x,y,str,size,0); //显示这个汉字
  882. str+=2;
  883. x+=size;//下一个汉字偏移
  884. }
  885. }
  886. }