Анимированные числа вводятся в документ iframe - PullRequest
0 голосов
/ 06 февраля 2019

У меня есть приложение, которое используется для создания электронных писем и отчетов о кликах по ссылкам в электронном письме.Я загружаю письмо в iframe и вставляю статистику кликов для каждой ссылки в iframe.Если возможно, я бы хотел анимировать каждое число от нуля до нужного числа при загрузке.

Кто-нибудь знает, как я могу это сделать?Вот если JSFiddle .

<div id="mailing-preview" :class="{'large': size == 'large'}">
    <div class="iframe-container">
    <iframe :ref="iframeRef" src="about:blank" id="mailing-content-preview" sandbox="allow-same-origin" scrolling="no" width="100%" height="100%" @load="iframeLoaded" ></iframe>
</div>

new Vue({
  el: "#mailing-preview",
  data() {
    return {
      iframeRef: "preview-iframe"
    };
  },
  methods: {
    updatePreview(newHtml) {
      const contentWindow = this.$refs[this.iframeRef].contentWindow;
      contentWindow.document.open("text/html", "replace");
      contentWindow.document.write(newHtml);
      contentWindow.document.close();
    },
   iframeLoaded() {
     this.addStasticBalloons();

   if (this.dynamicHeight) {
    const iframe = this.$refs[this.iframeRef];
    const contentWindow = iframe.contentWindow;
    const iframeHeight = contentWindow.document.body.scrollHeight;
    if (iframeHeight > 0) {
      iframe.setAttribute("style", "height:100%;");
      iframe.style.height = iframeHeight + "px";
    }
  }
},
addStasticBalloons() {
    const iframe = this.$refs[this.iframeRef];
    var markedAnchors = {};

    this.mailing.clickThroughRates.forEach(stat => {
      var anchors = iframe.contentDocument.querySelectorAll("a[href='" + stat.url + "']");
      var currentAnchorIndex = 0;
      var percentage = (stat.clickThroughRate * 100).toFixed(1);

      if (markedAnchors[stat.url] !== undefined) {
        markedAnchors[stat.url]++;
        currentAnchorIndex = markedAnchors[stat.url];
      } else {
        markedAnchors[stat.url] = 0;
      }

      if (anchors[currentAnchorIndex] === undefined) {
        return;
      }

      var bubble = document.createElement("span");
      bubble.style.position = "absolute";
      bubble.style.top = "-38px";
      bubble.style.left = "calc(100% + 18px)";
      bubble.style.background = "#6AC049";
      bubble.style.padding = "10px";
      bubble.style.zIndex = "1";
      bubble.style.fontSize = "16px";
      bubble.style.fontWeight = "bold";
      bubble.style.color = "white";
      bubble.style.borderRadius = "10px";
      bubble.style.boxShadow = "0px 0px 4px 0px #333";
      bubble.style.whiteSpace = "nowrap";

      var line = document.createElement("span");
      line.style.background = "#6AC049";
      line.style.width = "24px";
      line.style.height = "2px";
      line.style.display = "block";
      line.style.position = "absolute";
      line.style.left = "-1.25em";
      line.style.bottom = "-0.25em";
      line.style.transform = "rotate(-35deg)";

      var bubbleContents = document.createTextNode(percentage + "%");

      bubble.appendChild(bubbleContents);
      bubble.appendChild(line);

      anchors[currentAnchorIndex].appendChild(bubble);
      anchors[currentAnchorIndex].style.position = "relative";
      anchors[currentAnchorIndex].style.overflow = "visible";
      anchors[currentAnchorIndex].style.display = "inline-block";
    });
   }
 },
 mounted() {
   this.updatePreview(this.mailing.html);
 },
 watch: {
   mailing(newMailing) {
     this.updatePreview(newMailing.html);
   }
 },
 props: {
   size: {
     type: String,
     default: "small"
   },
   dynamicHeight: {
     type: Boolean,
     default: false
 },
  mailing: {
   type: Object,
   default: function() {
     return {id:1,name:"My Favorite Mailing",html:"",clickThroughRates:[]}
   }
  }
 }
})
...