写在前面

Github: https://github.com/AlloyTeam/AlloyGameEngine

在dom元素里,自带了input标签,设置其type为text,它就是一个文本框。

那么在Canvas中模拟input文本框是不是闲的没事找事?绝对不是!
因为在游戏当中可以统一化像素管理,具体统一化像素管理有什么好处,以后新开文章详细讨论。

演示

上面的文本框就是使用AlloyRenderingEngine渲染出来的。

使用

; (function () {
var Stage = ARE.Stage, Textbox = ARE.Textbox;
var stage = new Stage("#ourCanvas", true);
var textbox = new ARE.Textbox({
fontSize: 22,
color: "red",
width: 200,
height: 26
});
textbox.x = 50;
textbox.y = 50;
textbox.focus();
stage.add(textbox);
})();

原理(都在注释里)

; (function () {
//先把要使用类的赋给临时变量,以后就不用打点了:)
var Stage = ARE.Stage, Container = ARE.Container, Graphics = ARE.Graphics, Text = ARE.Text;
//文本框集成自容器
ARE.Textbox = Container.extend({
//构造函数
ctor: function (option) {
//把容器的属性和方法搞给自己
this._super();
//鼠标移上去指针的形状,AlloyRenderingEngine会自动帮你显示鼠标移上去时候的形状
this.cursor = "text";
//文本框的边框
this.box = new Graphics()
//直接根据传进的宽和高画个矩形
this.box.strokeRect(0, 0, option.width, option.height);
//文本框的背景,这里接近透明,为什么要设置背景是因为鼠标一上去要触发一个事件,
//而AlloyRenderingEngine的默认触发是像素级别,
//会根据getImageData得到该点的rgba的a是否为0去判断是否触发事件
//所以铺一个接近透明的背景
//主要是为了触发的事件是:鼠标移到文本框上面,鼠标形状要变成cursor:text
this.box.fillStyle("rgba(255,255,255,0.1)").fillRect(0, 0, option.width, option.height);
//把边框添加到自身(因为自身就是容器,继承自Container,所以有了add方法)
this.add(this.box);
//绑定事件
this._bindEvent();
//合并默认配置
this.option = {
fontSize: option.fontSize || 12,
fontFamily: option.fontFamily || "arial",
color: option.color || "black",
width: option.width
};
//cursorText代表文本框中闪烁的光标,自己用黑色的Text去模拟
this.cursorText = new Text("|", this.option.fontSize + "px " + this.option.fontFamily, "black");
//真正的input!!!!哈哈,玄机就在于此 = =!
this.realTextbox = document.createElement("input");
this.realTextbox.type = "text";
this.realTextbox.style.position = "fixed";
this.realTextbox.style.left= "-200px"
this.realTextbox.style.top= "0px"
document.body.appendChild(this.realTextbox);
//canvas中显示的文本
this.text = new Text("", this.option.fontSize + "px " + this.option.fontFamily, this.option.color);
//measureCtx是专门用于测量canvas中文本宽度的
this.measureCtx = document.createElement("canvas").getContext("2d");
this.measureCtx.font = this.option.fontSize + "px " + this.option.fontFamily;
this.add(this.text, this.cursorText);
//tickFPS是该容器tick执行的频率,AlloyRenderingEngine会自动帮你执行tick方法
this.tickFPS = 20;
},
//获取焦点
focus: function () {
var self = this;
//真正的input也同时获取焦点
this.realTextbox.focus();
//Canvas中的光标闪烁
this.loop = setInterval(function () {
self.cursorText.visible = !self.cursorText.visible;
}, 500);
},
//失去焦点
blur: function () {
clearInterval(this.loop);
//真正的input也同时失去焦点
this.realTextbox.blur();
//隐藏Canvas中的光标
this.cursorText.visible = false;
},
_bindEvent: function () {
var self = this;
this.onClick(function (evt) {
//真正的input也同时获取焦点
self.realTextbox.focus();
//显示光标
self.cursorText.visible = true;
//自己也假装获取焦点
self.focus();
//阻止冒泡
evt.stopPropagation();
});
//点击文本框的其他区域触发失去焦点
document.addEventListener("mousedown", function () {
//失去焦点
self.blur();
}, false);
},
//计算合适的显示文本,这主要是解决文本超出了文本框的宽度时候的显示问题
getFitStr: function (str, index) {
//利用measureText计算文本宽度
var width = this.measureCtx.measureText(str.substring(index, str.length - 1)).width;
if (width < this.option.width - this.option.fontSize) {
return this.getFitStr(str, --index);
} else {
return str.substring(index++, str.length - 1)
}
},
tick: function () {
//利用measureText计算文本宽度,并把该宽度赋值给光标的偏移
this.cursorText.x = this.measureCtx.measureText(this.realTextbox.value).width;
//如果宽度超了
if (this.cursorText.x > this.option.width) {
this.text.value = this.getFitStr(this.realTextbox.value, this.realTextbox.value.length - 2);
this.cursorText.x = this.measureCtx.measureText(this.text.value).width;
} else {//如果宽度没超
this.text.value = this.realTextbox.value;
}
}
});
})();

大部分代码都做了解释,不再重复阐述。

Github: https://github.com/AlloyTeam/AlloyGameEngine

AlloyRenderingEngine文本框组件的更多相关文章

  1. Flutter学习笔记(21)--TextField文本框组件和Card卡片组件

    如需转载,请注明出处:Flutter学习笔记(21)--TextField文本框组件和Card卡片组件 今天来学习下TextField文本框组件和Card卡片组件. 只要是应用程序就少不了交互,基本上 ...

  2. Flutter——TextField组件(文本框组件)

    TextField组件的常用属性: 属性 描述 maxLines 设置此参数可以把文本框改为多行文本框 onChanged 文本框改变的时候触发的事件 decoration hintText 类似 h ...

  3. javaSwing文本框组件

    public class JTextFieldTest extends JFrame{    private static final long serialVersionUID = 1L;    p ...

  4. xmlplus 组件设计系列之三 - 文本框

    文本框是页面中最常用的输入组件,它的默认使用方式如下: <input type='text'/> 当然,这里的 `type='text' 可以略去不写.大部分情况下,使用默认的文本框作为输 ...

  5. Python Tkinter 文本框(Entry)

    Python Tkinter 文本框用来让用户输入一行文本字符串. 你如果需要输入多行文本,可以使用 Text 组件. 你如果需要显示一行或多行文本且不允许用户修改,你可以使用 Label 组件. 语 ...

  6. [js开源组件开发]js文本框计数组件

    js文本框计数组件 先上效果图: 样式可以自行调整 ,它的功能提供文本框的实时计数,并作出对应的操作,比如现在超出了,点击下面的按钮后,文本框会闪动两下,阻止提交.具体例子可以点击demo:http: ...

  7. Android开发10.2:UI组件AutoCompleteTextView(自动完成文本框)

    概述 AutoCompleteTextVeiw(自动完成文本框)从 EditText派生而出  PS :EditText用法介绍           当用户输入一定字符后,自动完成自动完成文本框会显示 ...

  8. 『Asp.Net 组件』第一个 Asp.Net 服务器组件:自己的文本框控件

    代码: using System.Web; using System.Web.UI; using System.Web.UI.WebControls; namespace DemoWebControl ...

  9. 使用ErrorProvider组件验证文本框输入

    实现效果: 知识运用: ErrorProvider组件的BlinkStyle属性 //指示错误图标的闪烁时间 public ErrorBlinkStyle BlinkStyle{ get;set; } ...

随机推荐

  1. 闭区间套定理(Nested intervals theorem)

    ① ②这里用到了极限与不等关系 ③如果a≠b,那么便不会有$\lim _{n\rightarrow \infty }\left| I_n \right| =0$ ④如果还存在一点c在内,那么同样也不会 ...

  2. Immutable(不可变)集合

    不可变集合,顾名思义就是说集合是不可被修改的.集合的数据项是在创建的时候提供,并且在整个生命周期中都不可改变. 为什么要用immutable对象?immutable对象有以下的优点: 对不可靠的客户代 ...

  3. 用Go实现的简易TCP通信框架

    接触到GO之后,GO的网络支持非常令人喜欢.GO实现了在语法层面上可以保持同步语义,但是却又没有牺牲太多性能,底层一样使用了IO路径复用,比如在LINUX下用了EPOLL,在WINDOWS下用了IOC ...

  4. 2000条你应知的WPF小姿势 基础篇<69-73 WPF Freeze机制和Template>

    在正文开始之前需要介绍一个人:Sean Sexton. 来自明尼苏达双城的软件工程师.最为出色的是他维护了两个博客:2,000ThingsYou Should Know About C# 和 2,00 ...

  5. ABP框架 - 设置管理

    文档目录 本节内容: 简介 关于ISettingStore 定义设置 setting scope(设置范围) 重写设置定义 获取设置值 服务端 客户端 修改设置 关于缓存 简介 每个应用必需存储一些设 ...

  6. esayui

  7. WCF学习之旅—WCF服务配置(十四)

    一.概述 我们在前面章节中讲了寄宿,在前面的实例中也用到了配置文件,这一篇主要讲讲如何在应用配置文件,提高WCF程序的灵活性.在编写WCF服务应用程序时,编写配置项也是其中一项主要工作,在前面的几个示 ...

  8. 基于Caffe的DeepID2实现(下)

    小喵的唠叨话:这次的博客,真心累伤了小喵的心.但考虑到知识需要巩固和分享,小喵决定这次把剩下的内容都写完. 小喵的博客:http://www.miaoerduo.com 博客原文: http://ww ...

  9. 【WPF】闲着没事,写了个支持数据列表分页的帮助类

    支持分页的MVVM组件大家可以网上找,老周这个类只是没事写来娱乐一下的,主要是功能简单,轻量级,至少它满足了我的需求,也许还有未知的 bug . 这个类支持对数据列表进行分页处理,原理是利用 Skip ...

  10. 使用AWS亚马逊云搭建Gmail转发服务(二)

    title: 使用AWS亚马逊云搭建Gmail转发服务(二) author:青南 date: 2014-12-31 14:44:27 categories: [Python] tags: [Pytho ...