Laya 文本相关类(TextArea高度自适应)

一、位于laya.display包下的两个类

1.class Text extends Sprite
内部使用了graphics.fillBorderText绘制文本

2.class Input extends Text
内部封装了原生的文本输入框,并且是全局唯一的

private static function _createInputElement():void {
    _initInput(area = Browser.createElement("textarea"));
    _initInput(input = Browser.createElement("input"));
    
    inputContainer = Browser.createElement("div");
    inputContainer.style.position = "absolute";
    inputContainer.style.zIndex = 1E5;
    Browser.container.appendChild(inputContainer);
    //[IF-SCRIPT] inputContainer.setPos = function(x:int, y:int):void
    { inputContainer.style.left = x + 'px'; inputContainer.style.top = y + 'px'; };
}

在focusin时把原生的textarea或input添加到div中显示出来,在focusout时再移除。

//Input.as
public function set focus(value:Boolean):void {
    var input:* = nativeInput;
    
    if (_focus !== value) {
        if (value) {
            if (input.target) {
                input.target._focusOut();
            } else {
                _setInputMethod();
            }
            input.target = this;
            
            _focusIn();
        } else {
            input.target = null;
            _focusOut();
            Browser.document.body.scrollTop = 0;
            input.blur();
            
            if (Render.isConchApp) {
                input.setPos(-10000, -10000);
            } else if (inputContainer.contains(input))
                inputContainer.removeChild(input);
        }
    }
}

private function _setInputMethod():void {
    input.parentElement && (inputContainer.removeChild(input));
    area.parentElement && (inputContainer.removeChild(area));
    
    inputElement = (_multiline ? area : input);
    inputContainer.appendChild(inputElement);
    if (Text.RightToLeft)
    {
        inputElement.style.direction = "rtl";
    }
}

3.自动获取焦点,并弹出键盘
参考弹出一个面板 想让焦点自动在输入文本上且弹出手机键盘,官方答复是:“软键盘弹出必须有触发行为,不支持直接默认弹出虚拟键盘!”
其实在安卓上可以做到的,设置完input.focus=true之后,再调用_popupInputMethod即可。但IOS上无效。

public focusInInput() {
    this.chatInput.inputChat.focus = true;
    //移动平台在单击事件触发后弹出输入法
    var layaInput:any = Laya.Input;
    layaInput._popupInputMethod();
}

// 移动平台在单击事件触发后弹出输入法
private static function _popupInputMethod(e:*):void {
    //e.preventDefault();
    if (!Input.isInputting) return;
    
    var input:* = Input.inputElement;
    
    // 弹出输入法。
    input.focus();
}
二、位于laya.ui包下的两个类

1.class Component extends Sprite

2.class Label extends Component
//内部封装了一个Text,全是调用Text的方法,基本没有新代码。
protected var _tf:Text;

3.class TextInput extends Label

/**@inheritDoc */
override protected function createChildren():void {
    addChild(_tf = new Input());
    _tf.padding = Styles.inputLabelPadding;
    _tf.on(Event.INPUT, this, _onInput);
    _tf.on(Event.ENTER, this, _onEnter);
    _tf.on(Event.BLUR, this, _onBlur);
    _tf.on(Event.FOCUS, this, _onFocus);
}

在这段代码中,把_tf实例化为一个Input(Input是Text的一个子类,没有毛病)。所以可以看作TextInput是对Input的一个封装,比如很多方法是这样的:

public function get maxChars():int {
    return Input(_tf).maxChars;
}

public function set maxChars(value:int):void {
    Input(_tf).maxChars = value;
}

4.class TextArea extends TextInput
内部封装了两个滚动条,还有一些控制逻辑。

/**@private */
protected var _vScrollBar:VScrollBar;
/**@private */
protected var _hScrollBar:HScrollBar;
三、在处理高度自适应时,追踪了height属性

在class TextInput extends Label看到这段:

//TextInput.as
override protected function initialize():void {
    width = 128;
    height = 22;
}

override public function set height(value:Number):void {
    super.height = value;
    _bg && (_bg.height = value);
}

调用了super.height,所以去Label.as中看一看:

//Label.as
override public function set height(value:Number):void {
    super.height = value;
    _tf.height = value;
}

调用了_tf.height,所以去了Text.as中:

//Text.as
override public function set height(value:Number):void {
    if (value != _height) {
        super.height = value;
        isChanged = true;
    }
}

Text.as的super.height就是指向Sprite.as类了

//Sprite.as
public function set height(value:Number):void {
    if (this._height !== value) {
        this._height = value;
        conchModel && conchModel.size(this._width, value);
        repaint();
    }
}

再往回看,因为class Input extends Text,所以Input类继承了Text类的 set height,但是Input类的构造方法中,又指定了宽高值:

Input.as
/**创建一个新的 <code>Input</code> 类实例。*/
public function Input() {
    _width = 100;
    _height = 20;
    
    multiline = false;
    overflow = Text.SCROLL;
    
    on(Event.MOUSE_DOWN, this, _onMouseDown);
    on(Event.UNDISPLAY, this, _onUnDisplay);
}
四、进入正题,textarea高度自适应

在上述引擎源码分析中,可以看到class Input extends Text类中,是有一个原生的textarea的。作为静态属性,是全局唯一的,在Laya.init()中就会将其初始化。在HTML页面中也能看到基本属性:

<textarea style="position: absolute; overflow: hidden; resize: none;
 transform-origin: 0px 0px 0px; background-color: transparent; border: none;
 outline: none; z-index: 1; white-space: pre-wrap; color: rgb(51, 51, 51);
 font-size: 32px; font-family: Arial; line-height: 32px; font-style: normal;
 font-weight: normal; text-align: left; padding: 0px; width: 532px;
 height: 62px;" maxlength="100000" placeholder="">
</textarea>

所以要找到原生JS是如何处理textarea高度自适应,这里参考SegmentFault 如何创建一个高度自适应的textarea
第一种方式是把textarea替换成一个div,再把div的contentEditable设置为true。

第二种就是利用scrollHeight,设置到height上。这个可以参考http://www.jacklmoore.com/autosize/,我最终使用的就是这个库。当然也有手写的,这个网上帖子也比较多,可以参考textarea如何实现高度自适应(不出现滚动条)?

第三种是1楼高赞回复,在Textarea同级放一个pre 和span标签,然后把输入内容实时同步到span里。pre会随内容的高度变化而变化,expandingArea的高度又随pre变化,因为textarea的高度100% textarea的高度会随expandingArea变化,只要同步textarea的内容到pre中,就达到一个textarea随内容高度变化的目的了。

<div class="expandingArea">
    <pre><span></span><br></pre>
    <textarea placeholder="输入文字"></textarea>
</div>
五、使用jackmoore autosize库遇到的一些问题

1.autosize库源码中的resize方法有这样两行:

ta.style.height = '';
ta.style.height = ta.scrollHeight + heightOffset + 'px';

将height置为空串这个操作,经过我的测试,如果注释掉的话,在删除一行时,textarea不会自动缩回去。但是这行代码是有副作用的,那就是在输入第一行时,会多出来一个空行。后来参考论坛回复,把rows设置为1解决了。在Laya中的代码可以在Laya.init执行后去操作Textarea的属性。

//Input.as
/**@private */
protected static var input:*;
/**@private */
protected static var area:*;
/**@private */
protected static var inputElement:*;
/**@private */
protected static var inputContainer:*;

private static function _createInputElement():void {
    _initInput(area = Browser.createElement("textarea"));
    _initInput(input = Browser.createElement("input"));
    ……

area无法直接访问,所以只能这样写了:

Laya.init(750, 1334, Laya.WebGL);
Laya.Input["area"].rows = 1;

2._syncInputTransform在手机上不执行
在Input.as的_focusIn方法中,最后有这么一段:

// 输入框重定位。
_syncInputTransform();
if (!Render.isConchApp && Browser.onPC)
    Laya.timer.frameLoop(1, this, _syncInputTransform);

为什么只有在Browser.onPC时,才执行这个重定位呢,原因不明。那么只能侦听Laya.Event.INPUT事件,自己再手动执行_syncInputTransform方法了。

/**
 * 在输入期间,如果 Input 实例的位置改变,调用_syncInputTransform同步输入框的位置。
 */
private function _syncInputTransform():void {
    var inputElement:Object = nativeInput;
    var transform:Object = 
    Utils.getTransformRelativeToWindow(this, padding[3], padding[0]);
    var inputWid:int = _width - padding[1] - padding[3];
    var inputHei:int = _height - padding[0] - padding[2];
    if (Render.isConchApp) {
        inputElement.setScale(transform.scaleX, transform.scaleY);
        inputElement.setSize(inputWid, inputHei);
        inputElement.setPos(transform.x, transform.y);
    } else {
        //[IF-SCRIPT]inputContainer.style.transform = 
        inputContainer.style.webkitTransform = 
        "scale(" + transform.scaleX + "," + 
        transform.scaleY + ") rotate(" + (Laya.stage.canvasDegree) + "deg)";
        //[IF-SCRIPT]inputElement.style.width = inputWid + 'px';
        //[IF-SCRIPT]inputElement.style.height = inputHei + 'px';
        //[IF-SCRIPT]inputContainer.style.left = transform.x + 'px';
        //[IF-SCRIPT]inputContainer.style.top  = transform.y + 'px';
    }
}

这里原始的_syncInputTransform会考虑padding后,去设置style.width和height。因为我们已经在autosize中计算了宽高,所以_syncInputTransform方法中就不用再计算了,只要设置inputContainer.style.left和top即可。

六、弹出键盘不遮挡输入框

参考H5移动端弹出键盘时遮挡输入框,使用了文中第一种方式

private initLayaInput():void{
    var ta:any = Laya.Input["area"];
    //避免textarea出现空行
    ta.rows = 1;
    ta.onfocus = this.taFocusIn;
    ta.onblur = this.taFocusOut;
}

private taFocusIn: any = this.delayScrollBody.bind(this);
private delayScrollBody(): void {
    Laya.timer.loop(500,this,this.scrollDocBody);
}

private scrollDocBody():void{
    Laya.Browser.document.body.scrollTop = 
    Laya.Browser.document.body.scrollHeight;
}

private taFocusOut: any = this.removeDelayScrollBody.bind(this);
private removeDelayScrollBody(): void {
    Laya.timer.clear(this,this.scrollDocBody);
}

另外,也可以参考移动端iOS第三方输入法遮挡底部input及android键盘回落后留白问题

七、焦点控制
//Input.as
/**创建一个新的 <code>Input</code> 类实例。*/
public function Input() {
    _width = 100;
    _height = 20;
    
    multiline = false;
    overflow = Text.SCROLL;
    
    on(Event.MOUSE_DOWN, this, _onMouseDown);
    on(Event.UNDISPLAY, this, _onUnDisplay);
}

private function _onUnDisplay(e:Event = null):void {
    focus = false;
}

private function _onMouseDown(e:Event):void {
    focus = true;
}

// 移动平台最后单击画布才会调用focus
// 因此 调用focus接口是无法都在移动平台立刻弹出键盘的
public function set focus(value:Boolean):void {
    var input:* = nativeInput;
    
    if (_focus !== value) {
        if (value) {
            if (input.target) {
                input.target._focusOut();
            } else {
                _setInputMethod();
            }
            input.target = this;
            
            _focusIn();
        } else {
            input.target = null;
            _focusOut();
            Browser.document.body.scrollTop = 0;
            input.blur();
            
            if (Render.isConchApp) {
                input.setPos(-10000, -10000);
            } else if (inputContainer.contains(input))
                inputContainer.removeChild(input);
        }
    }
}

这里看到,侦听Event.MOUSE_DOWN获得了焦点。那么点击INPUT之外的区域,是怎么失去焦点的呢。经过查找,在MouseManager.as中找到了:

private function onMouseDown(ele:*):void {
    if (Input.isInputting && Laya.stage.focus && 
    Laya.stage.focus["focus"] && !Laya.stage.focus.contains(_target)) {
        // 从UI Input组件中取得Input引用
        // _tf 是TextInput的属性
        var pre_input:* = Laya.stage.focus['_tf'] || Laya.stage.focus;
        var new_input:Input = ele['_tf'] || ele;
        
        // 新的焦点是Input的情况下,不需要blur;
        // 不过如果是Input和TextArea之间的切换,还是需要重新弹出输入法;
        if (new_input is Input && new_input.multiline == pre_input.multiline)
            pre_input['_focusOut']();
        else
            pre_input.focus = false;
    }
    TouchManager.I.onMouseDown(ele, _tTouchID, _isLeftMouse);
}

参考一下Input.focus代码,可以看出执行focus=false与执行_focusOut方法的区别。最重要一点是,focus=false会额外执行input.blur(),这将导致收起软键盘

// 移动平台最后单击画布才会调用focus
// 因此 调用focus接口是无法都在移动平台立刻弹出键盘的
public function set focus(value:Boolean):void {
    var input:* = nativeInput;
    
    if (_focus !== value) {
        if (value) {
            if (input.target) {
                input.target._focusOut();
            } else {
                _setInputMethod();
            }
            input.target = this;
            
            _focusIn();
        } else {
            input.target = null;
            _focusOut();
            Browser.document.body.scrollTop = 0;
            input.blur();
            
            if (Render.isConchApp) {
                input.setPos(-10000, -10000);
            } else if (inputContainer.contains(input))
                inputContainer.removeChild(input);
        }
    }
}

默认情况下,在点击聊天发送按钮后,输入框input会失去焦点,导致软键盘收起。这时继续输入,就要重新点击输入框input,是不是很烦。但是想更改这个默认设定,也只能改源码了……

if ((new_input instanceof laya.display.Input )&& new_input.multiline==pre_input.multiline)
    pre_input['_focusOut']();
else if((new_input instanceof laya.ui.Button )&& new_input.name == "sendBtn"){
    //聊天点击发送按钮(name==sendBtn),输入框不会失去焦点
}else{
    pre_input.focus=false;
}
八、没有MOUSE_UP事件

在输入框输入文本时,直接点击发送按钮,在部分浏览器上会出现只是缩回键盘,文本并未发出的情况。经过检查,是没有抛出MOUSE_UP事件,进而导致没有CLICK事件。将发送按钮改为侦听MOUSE_DOWN事件即可。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容

  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,747评论 1 92
  • 一、《保质期》 任何东西都有保质期 比如刚刚做好的冰淇淋 比如你和我之间的感情 二、《青蛙王子》 像最后一次分别前...
    不敢说爱你阅读 250评论 0 2
  • 午加餐:饼干干晚水果:香蕉 参考目标: 1份豆2份肉3份“新鲜”水果4份谷物/薯5份蔬菜,深绿色叶菜最好6杯水 今...
    静趣_儿童心理师阅读 264评论 0 0
  • 约不可失 魏文侯与虞人期猎。是日,饮酒乐,天雨。文侯将出,左右曰:“今日饮酒乐,天又雨,公将焉之?”文侯曰:“吾与...
    学霸爱学习阅读 307评论 0 0