javascript - VueJS 组件 ref 在所有阶段都是未定义的

标签 javascript vue.js vuejs2 vue-component

我正在学习 VueJS,将其与 Rails 应用程序一起使用,但在尝试从父组件访问组件时遇到问题。我可能做错了,但我不知道问题出在哪里。

我试图将我的问题归结为一个简单的 html 示例:

<html>
  <head>
    <meta charset='utf-8' />
    <script src="https://cdn.jsdelivr.net/npm/vue@2.5.21/dist/vue.js"></script>
  </head>
  <body>
    <div id="app">
      <outer>
        <div>
          <form>
            <inner ref="testref"><input/></inner>
          </form>
        </div>
      </outer>
    </div>
    <script>
      let inner = Vue.component('inner', {
        template: `
          <div>
            <span hidden="true"><slot></slot></span>
            <input v-model="searchText" />
          </div>
        `,
        props: ['placeholder'],
        data: function() {
          return {
            searchText: this.placeholder
          }
        }
      })
      let outer = Vue.component('outer', {
        template: `
          <div>
            <h3 v-on:click="testos">Hello: {{ client_id }}</h3>
            <slot></slot>
          </div>
        `,
        data: function() {
          console.log('data', this.$refs.testref)
          return {
            client_id: ''
          }
        },
        mounted: function() {
          console.log('mounted', this.$refs.testref)
        },
        methods: {
          testos: function() {
            console.log('method', this.$refs.testref)
          }
        }
      })
      new Vue({
        el: '#app',
        components: {
          'inner': inner,
          'outer': outer
        }
      })
    </script>
  </body>
</html>

我收到以下控制台日志:

data undefined
mounted undefined
method undefined

我尝试遵循 https://v2.vuejs.org/v2/api/#ref 中的语法,但我也尝试过使用 :ref 而不是 ref,但效果并没有更好。

我知道存在“引用注册计时”的复杂性,即使我不完全理解它。但是点击 H3 的时机不应该是好的吗?我希望它甚至可以通过 mounted 来解决,就像 https://stackoverflow.com/a/40884455/2730032 中那样。 .

额外问题:我的问题实际上是关于 ref 的工作原理。然而,我确实有一个更普遍的问题,如果不使用 ref 的建议可能会受益。

我有一个令我满意的 inner 组件,它将 client_id 的输入字段与搜索字段包装在一起,该搜索字段从 API 中提供一些类似选择的选项,并将输入字段 client_id 设置为选定的选项。我想在 outer 组件中重用该 inner 组件,该组件将使用 client_id(每当它被 inner 更改时)来调用API 并填充 outer 组件中的一堆其他表单字段(这些字段通过槽从 innerouter 给出)。如果这有任何意义的话。

我认为最好的方法是将 client_id 作为 inner 的数据字段,并让 outer 通过 ref 访问它。所以在这里我试图让 ref 工作。

编辑1:有一个复制错误,对此深表歉意。 test1 标签来自以前的版本。但无论如何我都会遇到这个问题。

EDIT2:我接受了 Roy J 的第二个答案,因为我觉得它最好地回答了问题,尽管它不是一个好的设计,但对于某些人来说可能是一个有效的解决方案。然而,在我的实际实现中,我使用了 Roy J 第一个答案,任何阅读这个问题的人可能也应该这样做(我还设法将 inner 添加到 outer 的模板中,以避免使用应用程序)。

最佳答案

由于您无法使用 ref 的唯一原因是它是在父级中定义的,因此您可以转到 $parent 并使用其refs。请注意,这使得 outer 依赖于(纠缠)inner 的结构以及是否存在 ref 来获取它。这是糟糕的设计,但它是对代码进行的最小更改,您必须让它完成您想要做的事情。

const inner = Vue.component('inner', {
  template: `
          <div>
            <span hidden="true"><slot></slot></span>
            <input v-model="searchText" />
          </div>
        `,
  props: ['placeholder'],
  data: function() {
    return {
      searchText: this.placeholder
    }
  }
});
const outer = Vue.component('outer', {
  template: `
          <div>
            <h3 v-on:click="testos">Hello: {{ client_id }}</h3>
            <slot></slot>
          </div>
        `,
  data: function() {
    return {
      client_id: ''
    }
  },
  methods: {
    testos: function() {
      this.client_id = this.$parent.$refs.testref.searchText;    
      console.log('method', this.client_id);
    }
  }

})
new Vue({
  el: '#app',
  components: {
    'inner': inner,
    'outer': outer
  }
})
<script src="https://unpkg.com/vue@latest/dist/vue.js"></script>
<div id="app">
  <outer>
    <div>
      <form>
        <inner ref="testref"></inner>
      </form>
    </div>
  </outer>
</div>

关于javascript - VueJS 组件 ref 在所有阶段都是未定义的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53820217/

相关文章:

javascript - 使 div 像模态一样弹出

javascript - vue.js 2.0 : Data gotten from backend via ajax can not be shown in view

laravel - 使用 SweetAlert、Vue 和 Laravel 进行确认

javascript - Vue2-使用 :checked on checkbox input breaks default behaviour

javascript - 在 Vue 2.4 中将数据从子组件传输到父组件的正确方法是什么?

javascript - 移动设备上的 onClick 事件后函数没有反应

javascript - 我的代码有什么问题? javascript图表位置

typescript - Vitest - 引用错误 : File Is Not Defined

vue.js - Vuetify 数据表操作行

javascript - 为什么我在 jsPdf 输出上得到空白页?