×

如何为LCD1602显示器创建自定义字符

消耗积分:2 | 格式:zip | 大小:0.29 MB | 2022-10-20

分享资料个

描述

在本教程中,您将学习如何为 LCD 1602 显示器创建自定义字符,以及如何将这些字符转换为完整的动画。

自定义字符

1602 LCD 显示器(16 个字符 x 2 行)使用 HD44780 驱动芯片。该芯片内置一个 5 x 8 像素的 ASCII 字符集。它还允许您向该字符集添加多达 8 个自定义字符。

自定义字符定义为 8 字节数组。只有每个字节的低 5 位用于组成 5 x 8 自定义字符。下面是 Heart 字符的定义。

const byte heart[8] =
{
    B01010,
    B11111,
    B11111,
    B11111,
    B01110,
    B00100,
    B00000,
    B00000
};

要将此自定义字符存储在 LCD 显示屏中,您需要传递一个介于 0 和 7 之间的值,该值标识您希望设置的自定义字符槽以及定义字符的字节数组。

#define HEART 0
lcd.createChar(HEART, heart);

要显示自定义字符,请使用 write 函数并将自定义字符标识符传递给它。

lcd.write(HEART);

动画角色

自定义字符可以在使用前随时初始化或更改。限制是只能同时显示八个自定义字符。

那就是你不能做这样的事情:

lcd.createChar(0, heart);
lcd.setCursor(0, 0);
lcd.write(0);
lcd.createChar(0, club);
lcd.setCursor(1, 0);
lcd.write(0);

上面的示例显示了试图在不同位置同时显示在屏幕上的心形符号和俱乐部符号,但两者都使用相同的自定义字符“槽”。这行不通。

但是,您可以覆盖一个字符,前提是被覆盖的字符未在屏幕上的其他位置使用。

lcd.createChar(0, heart);
lcd.setCursor(0, 0);
lcd.write(0);
delay(1000);
lcd.createChar(0, club);
lcd.setCursor(0, 0);
lcd.write(0);

The上面的示例显示了 1 秒钟的心形字符,然后将其替换为俱乐部字符。

在老虎机中,每个“卷轴”包含 10 个符号。

#define REEL_SYMBOLS 10
#define TOTAL_REEL_SYMBOL_ROWS (REEL_SYMBOLS  * 8)
const byte reel[TOTAL_REEL_SYMBOL_ROWS] =
{
//Heart
  B00000,
  B01010,
  B11111,
  B11111,
  B11111,
  B01110,
  B00100,
  B00000,

  //Diamond
  B00000,
  B00100,
  B01110,
  B11111,
  B11111,
  B01110,
  B00100,
  B00000,

    :

  //Face
  B00000,
  B11011,
  B11011,
  B00000,
  B10001,
  B01110,
  B00000,
  B00000,
};

每个轮子都有自己的 8 字节内存缓冲区,卷轴数据将被复制到其中。

byte wheel[8];

通过在卷轴数组中保留索引,将 8 个字节复制到轮数组,然后设置为自定义字符并显示。索引递增,并在短暂延迟后重复该过程。

pYYBAGNPRLeAFkFEAAGtJfybpA0310.png
如何为动画的每一帧创建自定义角色。
 

这是从卷轴数据中填充车轮字符的代码。row变量是一个全局变量,指向要复制的下一行显示新形成的字符后,行变量递增。一旦到达列表的末尾,它又会重新开始。

#define SLOT 0
#define CURSOR_X 3
#define CURSOR_Y 0

int row = 0;

//Copy eight bytes into a character RAM from the reel array
// - global variable "row" points to top row to display
// - if "row + i" exceeds the array size, wrap round to the start
while (true)
{
  for (int i = 0; i < 8; i++)
  {
    wheel[i] = reel[(row + i) % TOTAL_REEL_SYMBOL_ROWS];
  }

  //This example uses custom character slot 0. If there is more than one wheel to
  //spin, each wheel will need its own unique custom character slot and cursor
  //position.
  lcd.createChar(SLOT, wheel);
  lcd.setCursor(CURSOR_X, CURSOR_Y);
  lcd.write(SLOT);

  //Increase row for next animation frame. Wrap round to start if it gets past
  //the end of the reel rows.
  row = (row + 1) % TOTAL_REEL_SYMBOL_ROWS;

  delay(100);
}

面包板赌场

实际的电子设备相当简单。需要的零件有:

  • Arduino UNO, Nano, Pro Mini or Mega
  • 1602 LCD screen
  • 220 ohm resistor for the LCD backlight
  • 10K trim pot for the LCD contrast adjustment
  • 4 tactile push buttons
  • 1 piezo electric passive buzzer
poYBAGNPRLqAUY8BAAKjc9nf-Ak784.jpg
接线图
 

如果您想知道为什么选择 Arduino 引脚连接,那是为了简化 PCB 布局。

连接 Arduino 后,将提供的草图加载到 Arduino IDE 中并对 Arduino 板进行编程。

玩游戏

有四个按钮。从左边开始,它们是 HIT、STAND、LESS 和 MORE。

HIT - 只要您可以选择投注,此按钮将带您进入顶层菜单,让您选择不同的游戏。在二十一点中,它也用于表示玩家想要另一张牌。

STAND - 一般来说,这会开始一个新的游戏。在老虎机中,它旋转轮子,在二十一点中,它开始新手,也用于表示玩家已完成抽牌并希望庄家玩,而在掷骰子中,它用于掷骰子。

LESS - 减少您想要下注的金额。在顶级菜单上,它选择上一个菜单项。

更多 - 增加您想要下注的金额。在顶级菜单上,它选择下一个菜单项。

演示视频

 

赌场控制台

将您的构建从面包板变成完整的控制台将需要印刷威廉希尔官方网站 板和一些额外的组件。如果您希望将 PCB 商业化制造或像我一样自己制造,请附上 Eagle 文件。我使用了碳粉方法。

使用我在车间周围的零件,将 Arduino UNO 替换为 ATMega328 DIL 芯片,并通过使用 DIL 28 针 IC 插座并将其引脚压平安装在威廉希尔官方网站 板背面。除 47uF/16V 钽 3528 电容器和 1206 10uF 陶瓷电容器外,所有电容器均为 0805 SMD 变体。晶体是一个 16Mhz 通孔变体,也安装在背面。轻触开关为 12x12mm,带有圆形按钮帽。

pYYBAGNPRLyAQsCHAAE9SXuF1H4671.jpg
PCB 顶视图和底视图
 

外壳是 3D 打印的,层高为 0.2 毫米,没有支撑。使用 2.5 毫米钻头钻出 PCB 安装孔,并使用 3 毫米丝锥创建螺纹。使用 M3 6mm 螺钉将板固定到位。我还将 PCB 上的四个安装孔钻出 4 毫米,以便在固定威廉希尔官方网站 板时进行任何必要的调整,以防止按钮顶部粘在外壳上。

poYBAGNPRMGAI78eAAnh2SXcntU662.jpg
带 PCB 的 3D 打印外壳
 

如果 ATMega328 芯片没有 Arduino 引导加载程序,则需要先添加一个。我使用我的AVR ISP 程序员添加引导加载程序,然后使用 FTDI 程序员上传草图。

poYBAGNPRMSAPbIzAACCZVSV9ts679.png
FTDI程序员
 

结论

老虎机动画效果非常好。希望您可以在未来的项目之一中使用此处描述的动画技术。


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

评论(0)
发评论

下载排行榜

全部0条评论

快来发表一下你的评论吧 !

'+ '

'+ '

'+ ''+ '
'+ ''+ ''+ '
'+ ''+ '' ); $.get('/article/vipdownload/aid/'+webid,function(data){ if(data.code ==5){ $(pop_this).attr('href',"/login/index.html"); return false } if(data.code == 2){ //跳转到VIP升级页面 window.location.href="//m.obk20.com/vip/index?aid=" + webid return false } //是会员 if (data.code > 0) { $('body').append(htmlSetNormalDownload); var getWidth=$("#poplayer").width(); $("#poplayer").css("margin-left","-"+getWidth/2+"px"); $('#tips').html(data.msg) $('.download_confirm').click(function(){ $('#dialog').remove(); }) } else { var down_url = $('#vipdownload').attr('data-url'); isBindAnalysisForm(pop_this, down_url, 1) } }); }); //是否开通VIP $.get('/article/vipdownload/aid/'+webid,function(data){ if(data.code == 2 || data.code ==5){ //跳转到VIP升级页面 $('#vipdownload>span').text("开通VIP 免费下载") return false }else{ // 待续费 if(data.code == 3) { vipExpiredInfo.ifVipExpired = true vipExpiredInfo.vipExpiredDate = data.data.endoftime } $('#vipdownload .icon-vip-tips').remove() $('#vipdownload>span').text("VIP免积分下载") } }); }).on("click",".download_cancel",function(){ $('#dialog').remove(); }) var setWeixinShare={};//定义默认的微信分享信息,页面如果要自定义分享,直接更改此变量即可 if(window.navigator.userAgent.toLowerCase().match(/MicroMessenger/i) == 'micromessenger'){ var d={ title:'如何为LCD1602显示器创建自定义字符',//标题 desc:$('[name=description]').attr("content"), //描述 imgUrl:'https://'+location.host+'/static/images/ele-logo.png',// 分享图标,默认是logo link:'',//链接 type:'',// 分享类型,music、video或link,不填默认为link dataUrl:'',//如果type是music或video,则要提供数据链接,默认为空 success:'', // 用户确认分享后执行的回调函数 cancel:''// 用户取消分享后执行的回调函数 } setWeixinShare=$.extend(d,setWeixinShare); $.ajax({ url:"//www.obk20.com/app/wechat/index.php?s=Home/ShareConfig/index", data:"share_url="+encodeURIComponent(location.href)+"&format=jsonp&domain=m", type:'get', dataType:'jsonp', success:function(res){ if(res.status!="successed"){ return false; } $.getScript('https://res.wx.qq.com/open/js/jweixin-1.0.0.js',function(result,status){ if(status!="success"){ return false; } var getWxCfg=res.data; wx.config({ //debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。 appId:getWxCfg.appId, // 必填,公众号的唯一标识 timestamp:getWxCfg.timestamp, // 必填,生成签名的时间戳 nonceStr:getWxCfg.nonceStr, // 必填,生成签名的随机串 signature:getWxCfg.signature,// 必填,签名,见附录1 jsApiList:['onMenuShareTimeline','onMenuShareAppMessage','onMenuShareQQ','onMenuShareWeibo','onMenuShareQZone'] // 必填,需要使用的JS接口列表,所有JS接口列表见附录2 }); wx.ready(function(){ //获取“分享到朋友圈”按钮点击状态及自定义分享内容接口 wx.onMenuShareTimeline({ title: setWeixinShare.title, // 分享标题 link: setWeixinShare.link, // 分享链接 imgUrl: setWeixinShare.imgUrl, // 分享图标 success: function () { setWeixinShare.success; // 用户确认分享后执行的回调函数 }, cancel: function () { setWeixinShare.cancel; // 用户取消分享后执行的回调函数 } }); //获取“分享给朋友”按钮点击状态及自定义分享内容接口 wx.onMenuShareAppMessage({ title: setWeixinShare.title, // 分享标题 desc: setWeixinShare.desc, // 分享描述 link: setWeixinShare.link, // 分享链接 imgUrl: setWeixinShare.imgUrl, // 分享图标 type: setWeixinShare.type, // 分享类型,music、video或link,不填默认为link dataUrl: setWeixinShare.dataUrl, // 如果type是music或video,则要提供数据链接,默认为空 success: function () { setWeixinShare.success; // 用户确认分享后执行的回调函数 }, cancel: function () { setWeixinShare.cancel; // 用户取消分享后执行的回调函数 } }); //获取“分享到QQ”按钮点击状态及自定义分享内容接口 wx.onMenuShareQQ({ title: setWeixinShare.title, // 分享标题 desc: setWeixinShare.desc, // 分享描述 link: setWeixinShare.link, // 分享链接 imgUrl: setWeixinShare.imgUrl, // 分享图标 success: function () { setWeixinShare.success; // 用户确认分享后执行的回调函数 }, cancel: function () { setWeixinShare.cancel; // 用户取消分享后执行的回调函数 } }); //获取“分享到腾讯微博”按钮点击状态及自定义分享内容接口 wx.onMenuShareWeibo({ title: setWeixinShare.title, // 分享标题 desc: setWeixinShare.desc, // 分享描述 link: setWeixinShare.link, // 分享链接 imgUrl: setWeixinShare.imgUrl, // 分享图标 success: function () { setWeixinShare.success; // 用户确认分享后执行的回调函数 }, cancel: function () { setWeixinShare.cancel; // 用户取消分享后执行的回调函数 } }); //获取“分享到QQ空间”按钮点击状态及自定义分享内容接口 wx.onMenuShareQZone({ title: setWeixinShare.title, // 分享标题 desc: setWeixinShare.desc, // 分享描述 link: setWeixinShare.link, // 分享链接 imgUrl: setWeixinShare.imgUrl, // 分享图标 success: function () { setWeixinShare.success; // 用户确认分享后执行的回调函数 }, cancel: function () { setWeixinShare.cancel; // 用户取消分享后执行的回调函数 } }); }); }); } }); } function openX_ad(posterid, htmlid, width, height) { if ($(htmlid).length > 0) { var randomnumber = Math.random(); var now_url = encodeURIComponent(window.location.href); var ga = document.createElement('iframe'); ga.src = 'https://www1.elecfans.com/www/delivery/myafr.php?target=_blank&cb=' + randomnumber + '&zoneid=' + posterid+'&prefer='+now_url; ga.width = width; ga.height = height; ga.frameBorder = 0; ga.scrolling = 'no'; var s = $(htmlid).append(ga); } } openX_ad(828, '#berry-300', 300, 250);