Skip to content Skip to sidebar Skip to footer

Get Incorrect Offsetwidth And Offsetheight Values

Here is my angular2 code. Template Component import {Component, OnInit, Input, ViewChild, ElementRef, Renderer} from '@angular/core'; export class SliderComponent implem

Solution 1:

You can detect size changes by using

MutationObserver

Probably the biggest audience for this new api are the people that write JS frameworks, [...] Another use case would be situations where you are using frameworks that manipulate the DOM and need to react to these modifications efficiently ( and without setTimeout hacks! ).

Here is how you can use it to detect changes in elements :

// select the target nodevar target = document.querySelector('#some-id'); // or // create an observer instancevar observer = newMutationObserver(function(mutations) {
    mutations.forEach(function(mutation) {
        console.log(mutation.type);
    });
});

// configuration of the observer:var config = { attributes: true, childList: true, characterData: true }

// pass in the target node, as well as the observer options
observer.observe(target, config);

// later, you can stop observing
observer.disconnect();

For your case, you could use it inside your ngAfterViewInit and refresh your offsets size. You can be more specific and only detect some mutations, and only then extract your offsets.

more info :

doc: https://developer.mozilla.org/en-US/docs/Web/API/MutationObserver

compatibility : https://caniuse.com/#feat=mutationobserver

Demo:

var observer = newMutationObserver(function(mutations) {
    mutations.forEach(function(mutation) {
       console.log(mutation);
       if(mutation.attributeName == 'class') // detect class change/*
          or if(mutation.target.clientWidth == myWidth)
          */showOffset(mutation.target);
          
          observer.disconnect();
    });
});

var config = { attributes: true}
var demoDiv = document.getElementById('demoDiv');
var logs = document.getElementById('logs');

// wait for document state to be completeif (document.readyState === "complete") {
    ngAfterViewInit();
  }
  
document.onreadystatechange = function () {
  if (document.readyState === "complete") {
    ngAfterViewInit();
  }
}

// observe changes that effects demoDiv + add classfunctionngAfterViewInit(){
	observer.observe(demoDiv, config);
  demoDiv.classList.add('slider-horizontal');
}

// show offsetWidth + height. // N.B offset width and height will be bigger than clientWidth because I added a border. If you remove the border you'll see 220px,20pxfunctionshowOffset(element){
  offsetMessage = "offsetWidth:" + demoDiv.offsetWidth + " offsetHeight: " + demoDiv.offsetHeight;
	console.log(offsetMessage);
  logs.innerHTML = offsetMessage;
}
.slider-horizontal {
  border: 2px solid red;
  width: 210px;
  height: 20px;
  background: grey;
}
<divid='demoDiv'> I am a demo div </div><divstyle="margin-top: 20px;"> logs : <spanid='logs'style=" padding: 5px; border: 1px solid black"></span></div>

Solution 2:

You have to schedule calls to 'offsetWidth' after rendering cycle, angular executes draw on the end of microtask queue, so you could try setTimeout(..., 0) or run Promise.resolve().then(...) outside of zonejs. Hope it helps.

Solution 3:

In order to get correct offset values you can use: ngAfterContentChecked with the AfterContentChecked Interface.

This method is called after every change detection run. So, inside this method use a flag (or counter) and setTimeOut:

if (this.counter <= 10) {
      // this print offsetwidth of my elementconsole.log('mm ' + this.container.nativeElement.offsetWidth);


      // setTimeOut allow to run another changedetection // so ngAfterContentChecked will run againsetTimeout(() => { }, 0);

      //you could use a counter or a flag in order to stop getting the right widththis.counter++;
    }

Hope it helps! Feel free to comment

Post a Comment for "Get Incorrect Offsetwidth And Offsetheight Values"