最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Vue.js trying to get DOM element by ID in computed returns null - Stack Overflow

programmeradmin1浏览0评论

I am wondering if it is posible to get elements by their id in Vue.js from a puted function. It is a silly question but for some reason it is giving me null as a response when I try to log this condition.

Let's say these are html tags:

<button id="dice1" class="revol"></button>
<button id="dice2" class="revol"></button>

then in one of my puted methods I try to access both ids

puted: {
  roll(){
    document.getElementById("dice1").className += "dic1";
    document.getElementById("dice2").className += "dic2";

    ...some code 
  }
}

Due to the error I checked in the created hook what was going on and realize that the document.getElementById of any id returns null

created() {
  console.log(document.getElementById("dice1"));
  console.log(document.getElementById("dice1"));
},

Also instead of referring straight to the DOM element I initialize variables and assigning the elementsById to them, but the results are the same

rollDice() {
  var  diceFirst= document.getElementById("dice1");
  var diceSecond= document.getElementById("dice2")
  
  diceFirst.className += "dic1";
  diceSecond.className += "dic2";

  ....some code
}

How can I improve this situation? Thanks in advance!

I am wondering if it is posible to get elements by their id in Vue.js from a puted function. It is a silly question but for some reason it is giving me null as a response when I try to log this condition.

Let's say these are html tags:

<button id="dice1" class="revol"></button>
<button id="dice2" class="revol"></button>

then in one of my puted methods I try to access both ids

puted: {
  roll(){
    document.getElementById("dice1").className += "dic1";
    document.getElementById("dice2").className += "dic2";

    ...some code 
  }
}

Due to the error I checked in the created hook what was going on and realize that the document.getElementById of any id returns null

created() {
  console.log(document.getElementById("dice1"));
  console.log(document.getElementById("dice1"));
},

Also instead of referring straight to the DOM element I initialize variables and assigning the elementsById to them, but the results are the same

rollDice() {
  var  diceFirst= document.getElementById("dice1");
  var diceSecond= document.getElementById("dice2")
  
  diceFirst.className += "dic1";
  diceSecond.className += "dic2";

  ....some code
}

How can I improve this situation? Thanks in advance!

Share Improve this question edited Jul 14, 2020 at 11:31 yarwest 9908 silver badges20 bronze badges asked Jul 14, 2020 at 10:37 Enrique GFEnrique GF 1,2954 gold badges20 silver badges39 bronze badges 6
  • 2 Vue lifecycle has no DOM in created hook. – Estradiaz Commented Jul 14, 2020 at 10:42
  • just checked in created to see what was going on , but since the very begginig the function breaks cause of this problem of null value of the elementById – Enrique GF Commented Jul 14, 2020 at 10:46
  • 3 You shouldn't be manually manipulating the DOM like that, just bind a value to each element's class. – Decade Moon Commented Jul 14, 2020 at 10:48
  • Any example on detail Decade Moon?....thanks – Enrique GF Commented Jul 14, 2020 at 10:56
  • 2 Use this.$refs and the ref attribute in the template to refer to elements in your template. Also, you should not be manipulating the class names in puted property. Is there a reason why you're doing that? For toggling of classes you should be using the v-bind:class directive instead. – Terry Commented Jul 14, 2020 at 10:59
 |  Show 1 more ment

2 Answers 2

Reset to default 4

Stemming from Decade Moon's ment, Vue doesn't like it when you manually manipulate the DOM (well, actually, Vue doesn't care, but it's bad practice since detaches your data from its representation in the DOM, which is not what you want when making a datadriven app). Instead, whenever you want to set the class of an element, do it with data (or with a puted property).

Your DOM should be representing your data, not the other way around.

new Vue({
  el: '#app',
  data: {
    class1: 'revol',
    class2: 'revol'
  },
  methods: {
    rollDice: function() {
      class1 += 'dic1';
      class2 += 'dic2';
      
      // ... some code
    }
  }
});
<div id="app">
  <button id="dice1" v-bind:class="class1">Dice 1</button>
  <button id="dice2" v-bind:class="class2">Dice 2</button>
</div>

This rollDice function does exactly the same thing as yours—it adds a new class to the buttons—but with a data-driven approach instead of trying to manipulate the DOM. In this case, the two buttons represent the data in class1 and class2, instead of having them decoupled.

The elements in the DOM are only available in the mounted hook. In created, the DOM elements are not rendered yet so you're unable to get them.

You can also use $ref to get the element from the DOM.

  <input type="text" ref="textbox" />

In the script, you can access this element using

  let tBox = this.$ref.textbox;

tBox will hold the input DOM element.

In the snippet below you can see how to get the element by id.

new Vue({
  el: "#app",
  mounted() {
    document.getElementById('header').textContent = 'New Text'
  }
})
<script src="https://cdn.jsdelivr/npm/[email protected]/dist/vue.js"></script>

<div id="app">
  <h1 id="header">Component Text</h1>
</div>

发布评论

评论列表(0)

  1. 暂无评论