VueJS从查询参数添加数据

时间:2016-12-21 05:35:09

标签: javascript pushstate vue-router vue.js vuejs2

我正在开发一个小的纯客户端VueJS应用程序,它使用VueRouter在数据输入应用程序时更新URL,这样用户就可以使用URL刷新页面并将数据重新加载到刷新页面时的应用程序。数据存储在查询参数中,例如: .../app.html#/?time=300&time=300&distance=300&distance=300&unit=m&unit=m

router根据需要解析数据,但是当我将数据添加到空的performance数组时,我的标记中的:class属性会引发

  

TypeError:无法读取未定义的属性“length”

其中undefinedperformances数组。我已将此问题缩小到:class性能循环内的v-for属性,方法是从应用的标记中删除:class并成功呈现该网站。我猜测performances数据数组尚未传递给:class

我在VueRouter文档中没有看到任何类型的路由。应该如何将数据从URL查询参数传递到应用程序中?这是我的相关代码(为了简洁而简化):

// js file
var router = new VueRouter({}); // empty because no routes, just query parameters, so no need for list

var app = new Vue({
    el: '#app',
    data: function() {
        return {
            performances: [],
            units: units,
            newPerformance: {
                distance: {
                    value: '',
                    units: units.distance.options[0]
                },
                time: {
                    formatted: '',
                    value: ''
                },
            },
            editingPerformance: null,
        }
    },
    router: router,
    watch: {
        performances: {
            handler: function() {
                this.drawChart();
                this.updateURL();
            },
            deep: true
        }
    },
    updateURL: function() {
        var times = this.performances.map(function(v) {
                return v.time.value
            }),
            distances = this.performances.map(function(v) {
                return v.distance.value
            }),
            units = this.performances.map(function(v) {
                return v.distance.units.value
            });
        router.push({
            query: {
                time: times,
                distance: distances,
                dunit: units
            }
        });
    }
    ....
});

HTML文件

...
<div v-for="(performance, index) in performances" :class="{'card-list': performances.length > 1 }">
...
</div>
...

我已阅读Reactivity in Depth文档,并尝试了以下解决方案: Vue.nextTick(function() { this.performances.push(performance); })

Vue.set(app, 'performances',performanceArray);

导致相同的错误。我可以根据要求提供更多代码。

编辑:添加堆栈跟踪:

[Vue warn]: Error when rendering root instance: warn @ vue.js:513
Vue._render @ vue.js:2934
(anonymous function) @ vue.js:2335
get @ vue.js:1643run @ vue.js:1712
flushSchedulerQueue @ vue.js:1530
(anonymous function) @ vue.js:465
nextTickHandler @ vue.js:414

紧接着是:

vue.js:427 TypeError: Cannot read property 'length' of undefined(…)
logError @ vue.js:427

1 个答案:

答案 0 :(得分:2)

似乎您可能会将其设置为代码中某个位置以外的其他内容。你将它实例化为一个数组,而.length只适用于数组和字符串,因此如果你将它设置为一个对象,json或其他任何东西,它将会出错。 Vue.nextTick有一个不会随身携带的回调&#34;这个&#34;除非你将它设置为与Vue.nextTick函数相同范围内的变量,否则转入它。

之类的东西
var self = this;

//inside nextTick

self.performances.push(performance);

但我想你不应该使用nextTick因为在重新渲染DOM之后nextTick会更新。 push()函数将触发dom重新渲染,这将使您的页面响应。如果您等待nextTick,那么在您更改其他内容之前它可能永远不会更新。

<div v-if="performances.length" v-for="(performance, index) in performances" 
:class="performances.length > 1 ? 'card-list' : ''">
...
</div>

v-if属性将确保在内容存在之前它不会呈现。

那应该解决你的问题

您还可以执行{{performance}}以查找对象中的确切内容,并且Google Chrome具有vue-debugger扩展名。它显示了所有数据属性的实时模型。我建议使用它。