在WYSIWYG中敲除自定义绑定重置光标

时间:2013-05-14 20:29:50

标签: knockout.js wysiwyg jwysiwyg

我遇到了涉及KnockoutJS和jwysiwyg插件(https://github.com/jwysiwyg/jwysiwyg)的情况。我正在使用下面看到的自定义绑定。但是,每当用户将文本更新到第二行时,通过绑定更新内容会导致光标重置到编辑器的最开头。因此,每当用户在更新后暂停键入超过1秒钟时,他们就会被迫点击回到之前的状态继续。

    ko.bindingHandlers.wysiwyg = {
        init: function (element, valueAccessor, allBindingsAccessor) {
        var options = allBindingsAccessor().wysiwygOptions || {};
        var value = ko.utils.unwrapObservable(valueAccessor());
        var $e = $(element);
        $.extend(true, {
            initialContent : value
        }, options);

        $e.wysiwyg(options);

        //handle the field changing
        function detectFn() {
            var observable = valueAccessor();
            var newvalue = $e.wysiwyg("getContent");
            observable(newvalue);
        }

        var current = $e.wysiwyg('document');
        var timer;
        current.bind({    
            keyup: function(){
                clearTimeout(timer);
                timer = setTimeout(detectFn, 1000);
            }
        });

        //handle disposal (if KO removes by the template binding)
        ko.utils.domNodeDisposal.addDisposeCallback(element, function() {
            $e.wysiwyg('destroy');
        });
    },
    update: function (element, valueAccessor) {
        var value = ko.utils.unwrapObservable(valueAccessor());
        $(element).wysiwyg("setContent", value);
        ko.bindingHandlers.value.update(element, valueAccessor);
    }
};

(然后使用绑定)

<textarea data-bind="wysiwyg: yourViewModelValue"></textarea>

更新完成后,如何获取插入位置以恢复光标?或者还有什么其他解决方案可以在更新时强制光标不移动?

编辑:一个证明问题的方法。 http://jsfiddle.net/797ZL/ 请注意当您键入第二行或更多行时会发生什么,然后暂停一秒钟,导致光标重置。

1 个答案:

答案 0 :(得分:3)

以下是您所看到的:

  1. 用户在WYSIWYG编辑器中键入一些文本
  2. 用户停止输入,超时在1秒后命中,触发detectFn
  3. detectFn从WYSIWYG编辑器中获取新值并更新您的observable
  4. 您的observable正在更新,会触发自定义活页夹的更新方法
  5. 自定义活页夹的更新方法从observable获取值,并在WYSIWYG编辑器上设置
  6. WYSIWYG编辑器设置了新值,将光标移动到顶行
  7. 你可以通过在第5步中间停止操作来解决这个问题,并且如果它与observable中的值不同,那么只更新WYSIWYG编辑器中的值 - 也就是说,如果observable是从WYSIWYG编辑。

    将更新方法更改为:

    update: function (element, valueAccessor) {
        var newValue = ko.utils.unwrapObservable(valueAccessor());
        var oldValue = $(element).wysiwyg("getContent");
        if (newValue !== oldValue) {
            $(element).wysiwyg("setContent", newValue);
        }
        // ko.bindingHandlers.value.update(element, valueAccessor);
    }
    

    我评论了该方法的最后一行......它似乎无关紧要,我不确定它为什么会存在。

    此外,这些代码行似乎也没有做任何事情:

    $.extend(true, {
        initialContent : value
    }, options);
    

    我认为您要做的是将initialContent属性添加到options对象。这可以通过这种方式完成:

    $.extend(options, {initialContent: value});
    

    但是,即使这样也不需要,因为在init方法之后调用update方法,并且此时observable的值被加载到WYSIWYG编辑器中。

    以下是更新代码的小提琴:http://jsfiddle.net/tlarson/797ZL/2/