changeset 1909:e62f1d2ead9e

merge in branch dev-pdf-generation
author Bernhard Reiter <bernhard@intevation.de>
date Fri, 18 Jan 2019 17:16:55 +0100
parents 32c56e6c089a (current diff) 0322bce42065 (diff)
children 146369b1c376
files
diffstat 11 files changed, 541 insertions(+), 112 deletions(-) [+]
line wrap: on
line diff
--- a/client/package.json	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/package.json	Fri Jan 18 17:16:55 2019 +0100
@@ -32,6 +32,7 @@
     "debounce": "^1.2.0",
     "font-awesome": "^4.7.0",
     "glob-all": "^3.1.0",
+    "jspdf": "^1.5.3",
     "locale2": "^2.2.0",
     "ol": "^5.3.0",
     "path": "^0.12.7",
--- a/client/src/components/Pdftool.vue	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/components/Pdftool.vue	Fri Jan 18 17:16:55 2019 +0100
@@ -7,8 +7,8 @@
   >
     <div style="width: 20rem">
       <h6 class="mb-0 py-2 px-3 border-bottom d-flex align-items-center">
-        <font-awesome-icon icon="file-pdf" class="mr-2"></font-awesome-icon
-        ><translate>Generate PDF</translate>
+        <font-awesome-icon icon="file-pdf" class="mr-2"></font-awesome-icon>
+        <translate>Generate PDF</translate>
         <font-awesome-icon
           icon="times"
           class="ml-auto text-muted"
@@ -16,15 +16,21 @@
         ></font-awesome-icon>
       </h6>
       <div class="p-3">
-        <b><translate>Chose format:</translate></b>
+        <b><translate>Choose format:</translate></b>
         <select v-model="form.format" class="form-control d-block w-100">
           <option value="landscape"><translate>landscape</translate></option>
           <option value="portrait"><translate>portrait</translate></option>
         </select>
+        <select v-model="form.resolution" class="form-control d-block w-100">
+          <option value="80">80 dpi</option>
+          <option value="120">120 dpi</option>
+          <option value="200">200 dpi</option>
+        </select>
         <select v-model="form.paperSize" class="form-control d-block w-100">
           <option value="a3"><translate>ISO A3</translate></option>
           <option value="a4"><translate>ISO A4</translate></option>
         </select>
+        <!--
         <small class="d-block my-2">
           <input
             type="radio"
@@ -33,19 +39,20 @@
             v-model="form.downloadType"
             selected
           />
-          <label for="pdfexport-downloadtype-download" class="ml-1 mr-2"
-            ><translate>Download</translate></label
-          >
+          <label for="pdfexport-downloadtype-download" class="ml-1 mr-2">
+            <translate>Download</translate>
+          </label>
           <input
             type="radio"
             id="pdfexport-downloadtype-open"
             value="open"
             v-model="form.downloadType"
           />
-          <label for="pdfexport-downloadtype-open" class="ml-1"
-            ><translate>Open in new window</translate></label
-          >
+          <label for="pdfexport-downloadtype-open" class="ml-1">
+            <translate>Open in new window</translate>
+          </label>
         </small>
+        -->
         <button
           @click="download"
           type="button"
@@ -65,15 +72,19 @@
  * SPDX-License-Identifier: AGPL-3.0-or-later
  * License-Filename: LICENSES/AGPL-3.0.txt
  *
- * Copyright (C) 2018 by via donau
+ * Copyright (C) 2018, 2019 by via donau
  *   – Österreichische Wasserstraßen-Gesellschaft mbH
  * Software engineering by Intevation GmbH
  *
  * Author(s):
  * * Markus Kottländer <markus.kottlaender@intevation.de>
  * * Bernhard E. Reiter <bernhard@intevation.de>
+ * * Fadi Abbud <fadi.abbud@intevation.de>
  */
 import { mapState } from "vuex";
+import jsPDF from "jspdf";
+import { getPointResolution } from "ol/proj.js";
+import locale2 from "locale2";
 
 var paperSizes = {
   // in millimeter, landscape [width, height]
@@ -88,31 +99,141 @@
       form: {
         format: "landscape",
         paperSize: "a4",
-        downloadType: "download"
+        downloadType: "download",
+        resolution: "120"
       }
     };
   },
   computed: {
-    ...mapState("application", ["showPdfTool"]),
-    ...mapState("bottlenecks", ["selectedSurvey"])
+    ...mapState("application", ["showPdfTool", "secondaryLogo"]),
+    ...mapState("bottlenecks", ["selectedSurvey"]),
+    ...mapState("map", ["openLayersMap"]),
+    ...mapState("user", ["user"])
   },
   methods: {
-    isLandscape() {
-      return this.form.format !== "portrait";
-    },
     download() {
-      /* eslint-disable no-unused-vars */
-      const width = this.isLandscape()
-        ? paperSizes[this.form.paperSize][0]
-        : paperSizes[this.form.paperSize][1];
-      const height = this.isLandscape()
-        ? paperSizes[this.form.paperSize][1]
-        : paperSizes[this.form.paperSize][0];
+      // FUTURE: disable button while working on it
+      console.log(
+        "will generate pdf with",
+        this.form.paperSize,
+        this.form.format,
+        this.form.resolution
+      );
+      var width, height;
+
+      if (this.form.format !== "portrait") {
+        // landscape, default
+        width = paperSizes[this.form.paperSize][0];
+        height = paperSizes[this.form.paperSize][1];
+      } else {
+        // switch width and height
+        width = paperSizes[this.form.paperSize][1];
+        height = paperSizes[this.form.paperSize][0];
+      }
+
+      // FUTURE: consider margins
+
+      // dots per mm = dots per inch / (25.4 mm/inch)
+      var pixelsPerMapMillimeter = this.form.resolution / 25.4;
+      var mapSizeForPrint = [
+        // in pixel
+        Math.round(width * pixelsPerMapMillimeter),
+        Math.round(height * pixelsPerMapMillimeter)
+      ];
+
+      // generate PDF and open it
+      // our units are milimeters; width 0 x height 0 is left upper corner
+
+      // Step 1 prepare and save current map extend
+      // Then add callback "rendercomplete" for Step 3
+      //    which will generate the pdf and resets the map view
+      // Step 2 which starts rendering a map with the necessary image size
+
+      var map = this.openLayersMap;
+      var mapSize = map.getSize(); // size in pixels of the map in the DOM
+      // Calculate the extent for the current view state and the passed size.
+      // The size is the pixel dimensions of the box into which the calculated
+      // extent should fit.
+      var mapExtent = map.getView().calculateExtent(mapSize);
+
+      var pdf = new jsPDF(this.form.format, "mm", this.form.paperSize);
+      var scalebarSize =
+        this.form.format === "portrait" && this.form.paperSize === "a4"
+          ? 10
+          : 15;
+      var northarrowSize = 3;
+      var self = this;
+
+      // set a callback for after the next complete rendering of the map
+      map.once("rendercomplete", function(event) {
+        let canvas = event.context.canvas;
+
+        // because we are using Web Mercator, a pixel represents
+        // a differently sizes spot depending on the place of the map.
+        // So we use a value calculated from the center of the current view.
+        let view = map.getView();
+        let proj = view.getProjection();
+        let metersPerPixel = // average meters (reality) per pixel (map)
+          getPointResolution(proj, view.getResolution(), view.getCenter()) *
+          proj.getMetersPerUnit();
+        // DEBUG console.log("metersPerPixel = ", metersPerPixel);
+
+        let scaleNominator = Math.round(
+          // the x in 1:x map scale
+          1000 * pixelsPerMapMillimeter * metersPerPixel
+        );
+        console.log("scaleNominator = ", scaleNominator);
+
+        var data = canvas.toDataURL("image/jpeg");
+        pdf.addImage(data, "JPEG", 0, 0, width, height);
+        self.addRoundedBox(
+          pdf,
+          width - scalebarSize * 5.2,
+          height - scalebarSize,
+          scalebarSize * 5,
+          scalebarSize
+        );
+        self.addScalebar(
+          pdf,
+          width - scalebarSize * 5,
+          height - scalebarSize / 2,
+          scalebarSize * 4,
+          scaleNominator
+        );
+        /*
+        self.addText(
+          pdf,
+          width - scalebarSize * 5,
+          height - scalebarSize * 0.6,
+          10,
+          "black",
+          50,
+          "Scale 1:" + scaleNominator
+        );
+        */
+        //self.addText(pdf, 150, 20, 10, "black", 70, "some text");
+        self.addNorthArrow(pdf, 15, 9, northarrowSize);
+        self.addPageInfo(pdf);
+        self.addAboutBox(pdf, width, height);
+
+        pdf.save("map.pdf");
+        // reset to original size
+        map.setSize(mapSize);
+        map.getView().fit(mapExtent, { size: mapSize });
+
+        // FUTURE: re-enable button when done
+      });
+
+      // trigger rendering
+      map.setSize(mapSizeForPrint);
+      map.getView().fit(mapExtent, { size: mapSizeForPrint });
 
       // TODO: replace this src with an API reponse after actually generating PDFs
-      let src = !this.isLandscape()
-        ? "/img/PrintTemplate-Var2-Landscape.pdf"
-        : "/img/PrintTemplate-Var2-Portrait.pdf";
+      /*
+      let src =
+        this.form.format === "landscape"
+          ? "/img/PrintTemplate-Var2-Landscape.pdf"
+          : "/img/PrintTemplate-Var2-Portrait.pdf";
 
       let a = document.createElement("a");
       a.href = src;
@@ -124,6 +245,180 @@
       document.body.appendChild(a);
       a.click();
       document.body.removeChild(a);
+      */
+    },
+    addRoundedBox(doc, x, y, w, h) {
+      // draws a rounded background box at (x,y) width x height
+      // using jsPDF units
+      doc.setDrawColor(255, 255, 255);
+      doc.setFillColor(255, 255, 255);
+      doc.roundedRect(x, y, w, h, 3, 3, "FD");
+    },
+    addScalebar(doc, x, y, maxWidth, scaleNominator) {
+      // maxWidth in mm
+      // scaleNominator is the x in 1:x of the map scale
+
+      // reduce width until we'll find a nice number for printing
+      // strategy:
+      //           1. check which unit prefix we shall use to get [10:10000[
+      //           2. using a mapping for the leading digit to get [1:10[
+      //           3. select a smaller number which is nicely dividable
+      //           4. scale up again to get length in paper mm and to be shown
+
+      // from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/log10#Polyfill
+      let log10 =
+        Math.log10 || // more precise, but unsupported by IE
+        function(x) {
+          return Math.log(x) * Math.LOG10E;
+        };
+
+      let maxLength = maxWidth * scaleNominator;
+
+      let unit = "mm";
+      let unitConversionFactor = 0;
+      if (maxLength > 10e7) {
+        // >10 km
+        unit = "km";
+        unitConversionFactor = 10e6;
+      } else if (maxLength > 10e4) {
+        // >10m
+        unit = "m";
+        unitConversionFactor = 10e3;
+      }
+
+      maxLength /= unitConversionFactor;
+
+      // DEBUG console.log(maxLength, unit);
+      let unroundedLength = maxLength;
+      let numberOfDigits = Math.floor(log10(unroundedLength));
+      let factor = Math.pow(10, numberOfDigits);
+      let mapped = unroundedLength / factor;
+      // DEBUG console.log(mapped);
+
+      var length = Math.floor(maxLength); // just to have an upper limit
+
+      // manually only use numbers that are very nice to devide by 4
+      // note that this is taken into account for rounding later
+      if (mapped > 8) {
+        length = 8 * factor;
+      } else if (mapped > 4) {
+        length = 4 * factor;
+      } else if (mapped > 2) {
+        length = 2 * factor;
+      } else {
+        length = factor;
+      }
+
+      let size = (length * unitConversionFactor) / scaleNominator / 4;
+      // DEBUG console.log(length, size);
+
+      doc.setDrawColor(0, 0, 0);
+      doc.setFillColor(0, 0, 0);
+      doc.rect(x, y, size, 1, "FD");
+      doc.setFillColor(255, 255, 255);
+      doc.setDrawColor(0, 0, 0);
+      doc.rect(x + size, y, size, 1, "FD");
+      doc.setFillColor(0, 0, 0);
+      doc.setDrawColor(0, 0, 0);
+      doc.rect(x + size * 2, y, size * 2, 1, "FD");
+      doc.setFontSize(5);
+      doc.text(x, y + 3, "0");
+      // /4 and could give 2.5. We still round, because of floating point arith
+      doc.text(
+        x + size,
+        y + 3,
+        (Math.round((length * 10) / 4) / 10).toString()
+      );
+      doc.text(x + size * 2, y + 3, Math.round(length / 2).toString());
+      doc.text(x + size * 4, y + 3, Math.round(length).toString() + " " + unit);
+    },
+
+    addNorthArrow(doc, x1, y1, size) {
+      var y2 = y1 + size * 3;
+      var x3 = x1 - size * 2;
+      var y3 = y1 + size * 5;
+      var x4 = x1 + size * 2;
+      //white triangle
+      doc.setFillColor(255, 255, 255);
+      doc.setDrawColor(255, 255, 255);
+      doc.triangle(x3 - 0.8, y3 + 1.2, x1, y1 - 1.2, x1, y2 + 0.6, "F");
+      doc.triangle(x1, y1 - 1.2, x1, y2 + 0.6, x4 + 0.8, y3 + 1.2, "F");
+      //north arrow
+      doc.setDrawColor(0, 0, 0);
+      doc.setFillColor(255, 255, 255);
+      doc.triangle(x3, y3, x1, y1, x1, y2, "FD");
+      doc.setFillColor(0, 0, 0);
+      doc.triangle(x1, y1, x1, y2, x4, y3, "FD");
+      doc.setFontSize(size * 3.1);
+      doc.setTextColor(255, 255, 255);
+      doc.setFontStyle("bold");
+      doc.text(size < 3 ? x1 - 0.5 : x1 - 1.3, y3 + 1, "N");
+      doc.setFontSize(size * 3);
+      doc.setTextColor(0, 0, 0);
+      doc.setFontStyle("normal");
+      doc.text(size < 3 ? x1 - 0.5 : x1 - 1.3, y3 + 1, "N");
+    },
+    // add some text at specific coordinates and determine how many wrolds in single line
+    addText(doc, postitionX, positionY, size, color, lineWidth, text) {
+      // split the incoming string to an array, each element is a string of words in a single line
+      var textLines = doc.splitTextToSize(text, lineWidth);
+      // get the longest line to fit the white backround to it
+      var longestString = "";
+      textLines.forEach(function(element) {
+        if (element.length > longestString.length) longestString = element;
+      });
+      var indexOfMaxString = textLines.indexOf(longestString);
+      // white background (rectangular) around the text
+      doc.setFillColor(255, 255, 255);
+      doc.setDrawColor(255, 255, 255);
+      doc.rect(
+        postitionX - doc.getStringUnitWidth(textLines[indexOfMaxString]) / size,
+        size > 10 ? positionY - size / 1.8 : positionY - size / 2.4,
+        doc.getStringUnitWidth(textLines[indexOfMaxString]) * (size / 2.6),
+        textLines.length * (size / 2),
+        "FD"
+      );
+      //rounded rectangular
+      /* doc.roundedRect(
+        postitionX - doc.getStringUnitWidth(textLines[indexOfMaxString]) / size,
+        size > 10 ? positionY - size / 1.8 : positionY - size / 2.6,
+        doc.getStringUnitWidth(textLines[indexOfMaxString]) * (size / 2.6),
+        textLines.length * (size / 2),
+        3,
+        3,
+        "FD"
+      ); */
+      doc.setTextColor(color);
+      doc.setFontSize(size);
+      doc.text(postitionX, positionY, textLines);
+    },
+    addPageInfo(doc) {
+      this.addRoundedBox(doc, 0, 0, 110, 8);
+      let str =
+        this.$gettext("Date of publication:") +
+        " " +
+        new Date().toLocaleString(locale2) +
+        " " +
+        this.$gettext("– printed by:") +
+        " " +
+        this.user;
+      this.addText(doc, 5, 5, 9, "black", 100, str);
+    },
+    addAboutBox(doc, docWidth, docHeight) {
+      let top = docHeight - 20;
+      this.addRoundedBox(doc, 0, top, 120, 20);
+      console.log("this.secondaryLogo", this.secondaryLogo);
+      let imageData =
+        "";
+      doc.addImage(imageData, "PNG", 5, docHeight - 19, 110, 10);
+      /*
+      if ("") {
+        doc.addImage(this.secondaryLogo, "PNG", 5, docHeight - 19, 50);
+      };
+      */
+      let str =
+        "Dislaimer: Lorem ipsum dolor sit amet, consectetur adipisici elit, sed eiusmod tempor incidunt ut labore et dolore magna aliqua.";
+      this.addText(doc, 5, docHeight - 6, 8, "black", 115, str);
     }
   }
 };
--- a/client/src/locale/bg_BG/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/bg_BG/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -18,7 +18,11 @@
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 3.4-dev\n"
 
-#: src/components/importschedule/Importscheduledetail.vue:565
+#: src/components/Pdftool.vue:402
+msgid "– printed by:"
+msgstr ""
+
+#: src/components/importschedule/Importscheduledetail.vue:452
 msgid "15 minutes"
 msgstr ""
 
@@ -91,7 +95,7 @@
 #: src/components/ImportSoundingresults.vue:16
 #, fuzzy
 msgid "Bottleneck"
-msgstr "Критични участъци"
+msgstr ""
 
 #: src/components/Systemconfiguration.vue:19
 msgid "Bottleneck Areas fill-color"
@@ -149,12 +153,7 @@
 #: src/components/usermanagement/Userdetail.vue:33
 #, fuzzy
 msgid "Country"
-msgstr "Държава"
-
-#: src/components/ImportStretches.vue:166
-#, fuzzy
-msgid "Countrycode"
-msgstr "Държава"
+msgstr ""
 
 #: src/components/importschedule/Importscheduledetail.vue:495
 msgid "Cronstring"
@@ -198,11 +197,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
@@ -410,7 +405,7 @@
 #: src/components/layers/Layers.vue:10
 #, fuzzy
 msgid "Layers"
-msgstr "Слоеве"
+msgstr ""
 
 #: src/components/Login.vue:58
 msgid "Login"
@@ -488,7 +483,7 @@
 #: src/components/Bottlenecks.vue:9 src/components/staging/Staging.vue:11
 #, fuzzy
 msgid "Name"
-msgstr "име"
+msgstr ""
 
 #: src/components/ImportStretches.vue:110
 msgid "National Object name"
@@ -576,7 +571,7 @@
 #: src/components/usermanagement/Userdetail.vue:331
 #, fuzzy
 msgid "Please choose a country"
-msgstr "Моля, изберете държава"
+msgstr ""
 
 #: src/components/usermanagement/Userdetail.vue:336
 msgid "Please choose a role"
@@ -632,10 +627,6 @@
 #: src/components/ImportStretches.vue:156
 #, fuzzy
 msgid "Please enter a source"
-msgstr "Моля, изберете държава"
-
-#: src/components/importschedule/Importscheduledetail.vue:297
-msgid "Please enter a source orgranization"
 msgstr ""
 
 #: src/components/ImportStretches.vue:61
@@ -892,7 +883,7 @@
 #: src/components/staging/Staging.vue:12
 #, fuzzy
 msgid "Type"
-msgstr "Тип"
+msgstr ""
 
 #: src/components/ImportWaterwayProfiles.vue:89
 msgid "under construction"
--- a/client/src/locale/de_AT/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/de_AT/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -196,11 +196,7 @@
 msgid "Depthreference"
 msgstr "Tiefenreferenz"
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr "Herunterladen"
 
@@ -265,7 +261,7 @@
 msgid "Fairwaydimension"
 msgstr "Waterway-Admin"
 
-#: src/components/importschedule/Importscheduledetail.vue:158
+#: src/components/importschedule/Importscheduledetail.vue:155
 msgid "Featuretype"
 msgstr ""
 
--- a/client/src/locale/en_GB/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/en_GB/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -194,11 +194,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
--- a/client/src/locale/hr_HR/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/hr_HR/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -18,7 +18,11 @@
 "Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
 "X-Generator: Weblate 3.4-dev\n"
 
-#: src/components/importschedule/Importscheduledetail.vue:565
+#: src/components/Pdftool.vue:402
+msgid "– printed by:"
+msgstr ""
+
+#: src/components/importschedule/Importscheduledetail.vue:452
 msgid "15 minutes"
 msgstr ""
 
@@ -91,7 +95,7 @@
 #: src/components/ImportSoundingresults.vue:16
 #, fuzzy
 msgid "Bottleneck"
-msgstr "Kritični sektor"
+msgstr ""
 
 #: src/components/Systemconfiguration.vue:19
 msgid "Bottleneck Areas fill-color"
@@ -197,11 +201,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
@@ -220,7 +220,7 @@
 #: src/components/importschedule/Importscheduledetail.vue:53
 #, fuzzy
 msgid "Email Notification"
-msgstr "E-mail Obavijest"
+msgstr ""
 
 #: src/components/ImportStretches.vue:66
 msgid "End rhm"
@@ -630,10 +630,6 @@
 #: src/components/ImportStretches.vue:156
 #, fuzzy
 msgid "Please enter a source"
-msgstr "Odaberite zemlju"
-
-#: src/components/importschedule/Importscheduledetail.vue:297
-msgid "Please enter a source orgranization"
 msgstr ""
 
 #: src/components/ImportStretches.vue:61
--- a/client/src/locale/hu_HU/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/hu_HU/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -194,11 +194,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
--- a/client/src/locale/ro_RO/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/ro_RO/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -18,7 +18,11 @@
 "Plural-Forms: nplurals=3; plural=n==1 ? 0 : (n==0 || (n%100 > 0 && n%100 < 20)) ? 1 : 2;\n"
 "X-Generator: Weblate 3.4-dev\n"
 
-#: src/components/importschedule/Importscheduledetail.vue:565
+#: src/components/Pdftool.vue:402
+msgid "– printed by:"
+msgstr ""
+
+#: src/components/importschedule/Importscheduledetail.vue:452
 msgid "15 minutes"
 msgstr ""
 
@@ -196,11 +200,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
@@ -219,7 +219,7 @@
 #: src/components/importschedule/Importscheduledetail.vue:53
 #, fuzzy
 msgid "Email Notification"
-msgstr "E-mail Notificare"
+msgstr ""
 
 #: src/components/ImportStretches.vue:66
 msgid "End rhm"
@@ -628,10 +628,6 @@
 #: src/components/ImportStretches.vue:156
 #, fuzzy
 msgid "Please enter a source"
-msgstr "Selectați o țară"
-
-#: src/components/importschedule/Importscheduledetail.vue:297
-msgid "Please enter a source orgranization"
 msgstr ""
 
 #: src/components/ImportStretches.vue:61
--- a/client/src/locale/sk_SK/LC_MESSAGES/app.po	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/locale/sk_SK/LC_MESSAGES/app.po	Fri Jan 18 17:16:55 2019 +0100
@@ -13,7 +13,11 @@
 "Plural-Forms: nplurals=2; plural=(n != 1);\n"
 "X-Generator: Weblate 3.4-dev\n"
 
-#: src/components/importschedule/Importscheduledetail.vue:565
+#: src/components/Pdftool.vue:402
+msgid "– printed by:"
+msgstr ""
+
+#: src/components/importschedule/Importscheduledetail.vue:452
 msgid "15 minutes"
 msgstr ""
 
@@ -191,11 +195,7 @@
 msgid "Depthreference"
 msgstr ""
 
-#: src/components/importschedule/Importscheduledetail.vue:46
-msgid "Distance Marks Virtual"
-msgstr ""
-
-#: src/components/Pdftool.vue:36
+#: src/components/Pdftool.vue:10
 msgid "Download"
 msgstr ""
 
@@ -214,7 +214,7 @@
 #: src/components/importschedule/Importscheduledetail.vue:53
 #, fuzzy
 msgid "Email Notification"
-msgstr "Email Oznámenia"
+msgstr ""
 
 #: src/components/ImportStretches.vue:66
 msgid "End rhm"
@@ -437,7 +437,7 @@
 #: src/components/Sidebar.vue:15
 #, fuzzy
 msgid "Map"
-msgstr "Mapa"
+msgstr ""
 
 #: src/components/importschedule/Importscheduledetail.vue:584
 msgid "March"
@@ -625,10 +625,6 @@
 #: src/components/ImportStretches.vue:156
 #, fuzzy
 msgid "Please enter a source"
-msgstr "Vyberte krajinu"
-
-#: src/components/importschedule/Importscheduledetail.vue:297
-msgid "Please enter a source orgranization"
 msgstr ""
 
 #: src/components/ImportStretches.vue:61
--- a/client/src/store/map.js	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/src/store/map.js	Fri Jan 18 17:16:55 2019 +0100
@@ -60,6 +60,7 @@
           source: new TileWMS({
             preload: 1,
             url: "https://service.d4d-portal.info/wms/",
+            crossOrigin: "anonymous",
             params: { LAYERS: "d4d", VERSION: "1.1.1", TILED: true }
           })
         }),
--- a/client/yarn.lock	Fri Jan 18 17:13:12 2019 +0100
+++ b/client/yarn.lock	Fri Jan 18 17:16:55 2019 +0100
@@ -1157,6 +1157,11 @@
   resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.1.tgz#5c85d662f76fa1d34575766c5dcd6615abcd30d8"
   integrity sha512-FZdkNBDqBRHKQ2MEbSC17xnPFOhZxeJ2YGSfr2BKf3sujG49Qe3bB+rGCwQfIaA7WHnGeGkSijX4FuBCdrzW/g==
 
+abab@^1.0.0:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/abab/-/abab-1.0.4.tgz#5faad9c2c07f60dd76770f71cf025b62a63cfd4e"
+  integrity sha1-X6rZwsB/YN12dw9xzwJbYqY8/U4=
+
 abab@^2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/abab/-/abab-2.0.0.tgz#aba0ab4c5eee2d4c79d3487d85450fb2376ebb0f"
@@ -1196,6 +1201,13 @@
   dependencies:
     acorn "^5.0.0"
 
+acorn-globals@^1.0.4:
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/acorn-globals/-/acorn-globals-1.0.9.tgz#55bb5e98691507b74579d0513413217c380c54cf"
+  integrity sha1-VbtemGkVB7dFedBRNBMhfDgMVM8=
+  dependencies:
+    acorn "^2.1.0"
+
 acorn-globals@^3.0.0:
   version "3.1.0"
   resolved "https://registry.yarnpkg.com/acorn-globals/-/acorn-globals-3.1.0.tgz#fd8270f71fbb4996b004fa880ee5d46573a731bf"
@@ -1273,6 +1285,11 @@
   resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-6.1.1.tgz#d363b66f5fac5f018ff9c3a1e7b6f8e310cc3913"
   integrity sha512-OtUw6JUTgxA2QoqqmrmQ7F2NYqiBPi/L2jqHyFtllhOUvXYQXf0Z1CYUinIfyT4bTCGmrA7gX9FvHA81uzCoVw==
 
+acorn@^2.1.0, acorn@^2.4.0:
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/acorn/-/acorn-2.7.0.tgz#ab6e7d9d886aaca8b085bc3312b79a198433f0e7"
+  integrity sha1-q259nYhqrKiwhbwzEreaGYQz8Oc=
+
 acorn@^3.0.4, acorn@^3.1.0:
   version "3.3.0"
   resolved "https://registry.yarnpkg.com/acorn/-/acorn-3.3.0.tgz#45e37fb39e8da3f25baee3ff5369e2bb5f22017a"
@@ -1331,7 +1348,7 @@
     fast-json-stable-stringify "^2.0.0"
     json-schema-traverse "^0.3.0"
 
-ajv@^6.1.0, ajv@^6.5.5:
+ajv@^6.1.0:
   version "6.6.1"
   resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.6.1.tgz#6360f5ed0d80f232cc2b294c362d5dc2e538dd61"
   integrity sha512-ZoJjft5B+EJBjUyu9C9Hc0OZyPZSSlOF+plzouTrg6UlA8f+e/n8NIgBFG/9tppJtpPWfthHakK7juJdNDODww==
@@ -1341,6 +1358,16 @@
     json-schema-traverse "^0.4.1"
     uri-js "^4.2.2"
 
+ajv@^6.5.5:
+  version "6.7.0"
+  resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.7.0.tgz#e3ce7bb372d6577bb1839f1dfdfcbf5ad2948d96"
+  integrity sha512-RZXPviBTtfmtka9n9sy1N5M5b82CbxWIR6HIis4s3WQTXDJamc/0gpCWNGz6EWdWp4DOfjzJfhz/AS9zVPjjWg==
+  dependencies:
+    fast-deep-equal "^2.0.1"
+    fast-json-stable-stringify "^2.0.0"
+    json-schema-traverse "^0.4.1"
+    uri-js "^4.2.2"
+
 align-text@^0.1.1, align-text@^0.1.3:
   version "0.1.4"
   resolved "https://registry.yarnpkg.com/align-text/-/align-text-0.1.4.tgz#0cd90a561093f35d0a99256c22b7069433fad117"
@@ -1876,6 +1903,11 @@
   resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767"
   integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c=
 
+base64-arraybuffer@^0.1.5:
+  version "0.1.5"
+  resolved "https://registry.yarnpkg.com/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz#73926771923b5a19747ad666aa5cd4bf9c6e9ce8"
+  integrity sha1-c5JncZI7Whl0etZmqlzUv5xunOg=
+
 base64-js@^1.0.2:
   version "1.3.0"
   resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.3.0.tgz#cab1e6118f051095e58b5281aea8c1cd22bfc0e3"
@@ -2314,6 +2346,16 @@
   resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30000913.tgz#560311ecf242eaf12159b720e64b11ebd759b5e4"
   integrity sha512-PP7Ypc35XY1mNduHqweGNOp0qfNUCmaQauGOYDByvirlFjrzRyY72pBRx7jnBidOB8zclg00DAzsy2H475BouQ==
 
+canvg@1.5.3:
+  version "1.5.3"
+  resolved "https://registry.yarnpkg.com/canvg/-/canvg-1.5.3.tgz#aad17915f33368bf8eb80b25d129e3ae922ddc5f"
+  integrity sha512-7Gn2IuQzvUQWPIuZuFHrzsTM0gkPz2RRT9OcbdmA03jeKk8kltrD8gqUzNX15ghY/4PV5bbe5lmD6yDLDY6Ybg==
+  dependencies:
+    jsdom "^8.1.0"
+    rgbcolor "^1.0.1"
+    stackblur-canvas "^1.4.1"
+    xmldom "^0.1.22"
+
 capture-exit@^1.2.0:
   version "1.2.0"
   resolved "https://registry.yarnpkg.com/capture-exit/-/capture-exit-1.2.0.tgz#1c5fcc489fd0ab00d4f1ac7ae1072e3173fbab6f"
@@ -2945,6 +2987,13 @@
     postcss "^7.0.1"
     timsort "^0.3.0"
 
+css-line-break@1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/css-line-break/-/css-line-break-1.0.1.tgz#19f2063a33e95fb2831b86446c0b80c188af450a"
+  integrity sha1-GfIGOjPpX7KDG4ZEbAuAwYivRQo=
+  dependencies:
+    base64-arraybuffer "^0.1.5"
+
 css-loader@^1.0.1:
   version "1.0.1"
   resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-1.0.1.tgz#6885bb5233b35ec47b006057da01cc640b6b79fe"
@@ -3123,11 +3172,18 @@
   dependencies:
     css-tree "1.0.0-alpha.29"
 
-cssom@0.3.x, "cssom@>= 0.3.2 < 0.4.0":
+cssom@0.3.x, "cssom@>= 0.3.0 < 0.4.0", "cssom@>= 0.3.2 < 0.4.0":
   version "0.3.4"
   resolved "https://registry.yarnpkg.com/cssom/-/cssom-0.3.4.tgz#8cd52e8a3acfd68d3aed38ee0a640177d2f9d797"
   integrity sha512-+7prCSORpXNeR4/fUP3rL+TzqtiFfhMvTd7uEqMdgPvLPt4+uzFUeufx5RHjGTACCargg/DiEt/moMQmvnfkog==
 
+"cssstyle@>= 0.2.34 < 0.3.0":
+  version "0.2.37"
+  resolved "https://registry.yarnpkg.com/cssstyle/-/cssstyle-0.2.37.tgz#541097234cb2513c83ceed3acddc27ff27987d54"
+  integrity sha1-VBCXI0yyUTyDzu06zdwn/yeYfVQ=
+  dependencies:
+    cssom "0.3.x"
+
 cssstyle@^1.0.0:
   version "1.1.1"
   resolved "https://registry.yarnpkg.com/cssstyle/-/cssstyle-1.1.1.tgz#18b038a9c44d65f7a8e428a653b9f6fe42faf5fb"
@@ -3966,7 +4022,7 @@
   resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
   integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=
 
-escodegen@1.x.x, escodegen@^1.9.1:
+escodegen@1.x.x, escodegen@^1.6.1, escodegen@^1.9.1:
   version "1.11.0"
   resolved "https://registry.yarnpkg.com/escodegen/-/escodegen-1.11.0.tgz#b27a9389481d5bfd5bec76f7bb1eb3f8f4556589"
   integrity sha512-IeMV45ReixHS53K/OmfKAIztN/igDHzTJUhZM3k1jMhIZWjk45SMwAtBsEXiJp3vSPmTcu6CXn7mDvFHRN66fw==
@@ -4474,6 +4530,10 @@
     loader-utils "^1.0.2"
     schema-utils "^1.0.0"
 
+"file-saver@github:eligrey/FileSaver.js#1.3.8":
+  version "1.3.8"
+  resolved "https://codeload.github.com/eligrey/FileSaver.js/tar.gz/e865e37af9f9947ddcced76b549e27dc45c1cb2e"
+
 file-uri-to-path@1:
   version "1.0.0"
   resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd"
@@ -5229,6 +5289,13 @@
     toposort "^1.0.0"
     util.promisify "1.0.0"
 
+html2canvas@1.0.0-alpha.12:
+  version "1.0.0-alpha.12"
+  resolved "https://registry.yarnpkg.com/html2canvas/-/html2canvas-1.0.0-alpha.12.tgz#3b1992e3c9b3f56063c35fd620494f37eba88513"
+  integrity sha1-OxmS48mz9WBjw1/WIElPN+uohRM=
+  dependencies:
+    css-line-break "1.0.1"
+
 htmlparser2@^3.9.1:
   version "3.10.0"
   resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.10.0.tgz#5f5e422dcf6119c0d983ed36260ce9ded0bee464"
@@ -5322,7 +5389,7 @@
     debug "2"
     extend "3"
 
-iconv-lite@0.4, iconv-lite@0.4.24, iconv-lite@^0.4.17, iconv-lite@^0.4.4:
+iconv-lite@0.4, iconv-lite@0.4.24, iconv-lite@^0.4.13, iconv-lite@^0.4.17, iconv-lite@^0.4.4:
   version "0.4.24"
   resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
   integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
@@ -6413,6 +6480,29 @@
     ws "^5.2.0"
     xml-name-validator "^3.0.0"
 
+jsdom@^8.1.0:
+  version "8.5.0"
+  resolved "https://registry.yarnpkg.com/jsdom/-/jsdom-8.5.0.tgz#d4d8f5dbf2768635b62a62823b947cf7071ebc98"
+  integrity sha1-1Nj12/J2hjW2KmKCO5R89wcevJg=
+  dependencies:
+    abab "^1.0.0"
+    acorn "^2.4.0"
+    acorn-globals "^1.0.4"
+    array-equal "^1.0.0"
+    cssom ">= 0.3.0 < 0.4.0"
+    cssstyle ">= 0.2.34 < 0.3.0"
+    escodegen "^1.6.1"
+    iconv-lite "^0.4.13"
+    nwmatcher ">= 1.3.7 < 2.0.0"
+    parse5 "^1.5.1"
+    request "^2.55.0"
+    sax "^1.1.4"
+    symbol-tree ">= 3.1.0 < 4.0.0"
+    tough-cookie "^2.2.0"
+    webidl-conversions "^3.0.1"
+    whatwg-url "^2.0.1"
+    xml-name-validator ">= 2.0.1 < 3.0.0"
+
 jsesc@^1.3.0:
   version "1.3.0"
   resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-1.3.0.tgz#46c3fec8c1892b12b0833db9bc7622176dbab34b"
@@ -6487,6 +6577,18 @@
   resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.0.tgz#2c74b6ee41d93ca51b7b5aaee8f503631d252a73"
   integrity sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=
 
+jspdf@^1.5.3:
+  version "1.5.3"
+  resolved "https://registry.yarnpkg.com/jspdf/-/jspdf-1.5.3.tgz#5a12c011479defabef5735de55c913060ed219f2"
+  integrity sha512-J9X76xnncMw+wIqb15HeWfPMqPwYxSpPY8yWPJ7rAZN/ZDzFkjCSZObryCyUe8zbrVRNiuCnIeQteCzMn7GnWw==
+  dependencies:
+    canvg "1.5.3"
+    file-saver eligrey/FileSaver.js#1.3.8
+    html2canvas "1.0.0-alpha.12"
+    omggif "1.0.7"
+    promise-polyfill "8.1.0"
+    stackblur-canvas "2.2.0"
+
 jsprim@^1.2.2:
   version "1.4.1"
   resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.1.tgz#313e66bc1e5cc06e438bc1b7499c2e5c56acb6a2"
@@ -7623,6 +7725,11 @@
   resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d"
   integrity sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=
 
+"nwmatcher@>= 1.3.7 < 2.0.0":
+  version "1.4.4"
+  resolved "https://registry.yarnpkg.com/nwmatcher/-/nwmatcher-1.4.4.tgz#2285631f34a95f0d0395cd900c96ed39b58f346e"
+  integrity sha512-3iuY4N5dhgMpCUrOVnuAdGrgxVqV2cJpM+XNccjR2DKOB1RUP0aA+wGXEiNziG/UKboFyGBIoKOaNlJxx8bciQ==
+
 nwsapi@^2.0.7:
   version "2.0.9"
   resolved "https://registry.yarnpkg.com/nwsapi/-/nwsapi-2.0.9.tgz#77ac0cdfdcad52b6a1151a84e73254edc33ed016"
@@ -7721,6 +7828,11 @@
     pixelworks "1.1.0"
     rbush "2.0.2"
 
+omggif@1.0.7:
+  version "1.0.7"
+  resolved "https://registry.yarnpkg.com/omggif/-/omggif-1.0.7.tgz#59d2eecb0263de84635b3feb887c0c9973f1e49d"
+  integrity sha1-WdLuywJj3oRjWz/riHwMmXPx5J0=
+
 on-finished@~2.3.0:
   version "2.3.0"
   resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947"
@@ -8004,6 +8116,11 @@
   resolved "https://registry.yarnpkg.com/parse5/-/parse5-4.0.0.tgz#6d78656e3da8d78b4ec0b906f7c08ef1dfe3f608"
   integrity sha512-VrZ7eOd3T1Fk4XWNXMgiGBK/z0MG48BWG2uQNU4I72fkQuKUTZpl+u9k+CxEG0twMVzSmXEEz12z5Fnw1jIQFA==
 
+parse5@^1.5.1:
+  version "1.5.1"
+  resolved "https://registry.yarnpkg.com/parse5/-/parse5-1.5.1.tgz#9b7f3b0de32be78dc2401b17573ccaf0f6f59d94"
+  integrity sha1-m387DeMr543CQBsXVzzK8Pb1nZQ=
+
 parse5@^3.0.1:
   version "3.0.3"
   resolved "https://registry.yarnpkg.com/parse5/-/parse5-3.0.3.tgz#042f792ffdd36851551cf4e9e066b3874ab45b5c"
@@ -8629,6 +8746,11 @@
   resolved "https://registry.yarnpkg.com/promise-inflight/-/promise-inflight-1.0.1.tgz#98472870bf228132fcbdd868129bad12c3c029e3"
   integrity sha1-mEcocL8igTL8vdhoEputEsPAKeM=
 
+promise-polyfill@8.1.0:
+  version "8.1.0"
+  resolved "https://registry.yarnpkg.com/promise-polyfill/-/promise-polyfill-8.1.0.tgz#30059da54d1358ce905ac581f287e184aedf995d"
+  integrity sha512-OzSf6gcCUQ01byV4BgwyUCswlaQQ6gzXc23aLQWhicvfX9kfsUiUhgt3CCQej8jDnl8/PhGF31JdHX2/MzF3WA==
+
 promise@^7.0.1:
   version "7.3.1"
   resolved "https://registry.yarnpkg.com/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf"
@@ -8687,9 +8809,9 @@
   integrity sha1-8FKijacOYYkX7wqKw0wa5aaChrM=
 
 psl@^1.1.24, psl@^1.1.28:
-  version "1.1.29"
-  resolved "https://registry.yarnpkg.com/psl/-/psl-1.1.29.tgz#60f580d360170bb722a797cc704411e6da850c67"
-  integrity sha512-AeUmQ0oLN02flVHXWh9sSJF7mcdFq0ppid/JkErufc3hGIV/AMa8Fo9VgDo/cT2jFdOWoFvHp90qqBH54W+gjQ==
+  version "1.1.31"
+  resolved "https://registry.yarnpkg.com/psl/-/psl-1.1.31.tgz#e9aa86d0101b5b105cbe93ac6b784cd547276184"
+  integrity sha512-/6pt4+C+T+wZUieKR620OpzN/LlnNKuWjy1iFLQ/UG35JqHlR/89MP1d96dUfkf6Dne3TuLQzOYEYshJ+Hx8mw==
 
 public-encrypt@^4.0.0:
   version "4.0.3"
@@ -9197,7 +9319,7 @@
     stealthy-require "^1.1.0"
     tough-cookie ">=2.3.3"
 
-request@^2.87.0, request@^2.88.0:
+request@^2.55.0, request@^2.87.0, request@^2.88.0:
   version "2.88.0"
   resolved "https://registry.yarnpkg.com/request/-/request-2.88.0.tgz#9c2fca4f7d35b592efe57c7f0a55e81052124fef"
   integrity sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==
@@ -9315,6 +9437,11 @@
   resolved "https://registry.yarnpkg.com/rgba-regex/-/rgba-regex-1.0.0.tgz#43374e2e2ca0968b0ef1523460b7d730ff22eeb3"
   integrity sha1-QzdOLiyglosO8VI0YLfXMP8i7rM=
 
+rgbcolor@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/rgbcolor/-/rgbcolor-1.0.1.tgz#d6505ecdb304a6595da26fa4b43307306775945d"
+  integrity sha1-1lBezbMEplldom+ktDMHMGd1lF0=
+
 right-align@^0.1.1:
   version "0.1.3"
   resolved "https://registry.yarnpkg.com/right-align/-/right-align-0.1.3.tgz#61339b722fe6a3515689210d24e14c96148613ef"
@@ -9435,7 +9562,7 @@
     pify "^3.0.0"
     semver "^5.5.0"
 
-sax@^1.2.4, sax@~1.2.4:
+sax@^1.1.4, sax@^1.2.4, sax@~1.2.4:
   version "1.2.4"
   resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9"
   integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==
@@ -9861,9 +9988,9 @@
   integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=
 
 sshpk@^1.7.0:
-  version "1.15.2"
-  resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.15.2.tgz#c946d6bd9b1a39d0e8635763f5242d6ed6dcb629"
-  integrity sha512-Ra/OXQtuh0/enyl4ETZAfTaeksa6BXks5ZcjpSUNrjBr0DvrJKX+1fsKDPpT9TBXgHAFsa4510aNVgI8g/+SzA==
+  version "1.16.0"
+  resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.16.0.tgz#1d4963a2fbffe58050aa9084ca20be81741c07de"
+  integrity sha512-Zhev35/y7hRMcID/upReIvRse+I9SVhyVre/KTJSJQWMz3C3+G+HpO7m1wK/yckEtujKZ7dS4hkVxAnmHaIGVQ==
   dependencies:
     asn1 "~0.2.3"
     assert-plus "^1.0.0"
@@ -9899,6 +10026,16 @@
   resolved "https://registry.yarnpkg.com/stack-utils/-/stack-utils-1.0.2.tgz#33eba3897788558bebfc2db059dc158ec36cebb8"
   integrity sha512-MTX+MeG5U994cazkjd/9KNAapsHnibjMLnfXodlkXw76JEea0UiNzrqidzo1emMwk7w5Qhc9jd4Bn9TBb1MFwA==
 
+stackblur-canvas@2.2.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/stackblur-canvas/-/stackblur-canvas-2.2.0.tgz#cacc5924a0744b3e183eb2e6c1d8559c1a17c26e"
+  integrity sha512-5Gf8dtlf8k6NbLzuly2NkGrkS/Ahh+I5VUjO7TnFizdJtgpfpLLEdQlLe9umbcnZlitU84kfYjXE67xlSXfhfQ==
+
+stackblur-canvas@^1.4.1:
+  version "1.4.1"
+  resolved "https://registry.yarnpkg.com/stackblur-canvas/-/stackblur-canvas-1.4.1.tgz#849aa6f94b272ff26f6471fa4130ed1f7e47955b"
+  integrity sha1-hJqm+UsnL/JvZHH6QTDtH35HlVs=
+
 stackframe@^1.0.4:
   version "1.0.4"
   resolved "https://registry.yarnpkg.com/stackframe/-/stackframe-1.0.4.tgz#357b24a992f9427cba6b545d96a14ed2cbca187b"
@@ -10138,7 +10275,7 @@
     unquote "~1.1.1"
     util.promisify "~1.0.0"
 
-symbol-tree@^3.2.2:
+"symbol-tree@>= 3.1.0 < 4.0.0", symbol-tree@^3.2.2:
   version "3.2.2"
   resolved "https://registry.yarnpkg.com/symbol-tree/-/symbol-tree-3.2.2.tgz#ae27db38f660a7ae2e1c3b7d1bc290819b8519e6"
   integrity sha1-rifbOPZgp64uHDt9G8KQgZuFGeY=
@@ -10357,7 +10494,7 @@
   resolved "https://registry.yarnpkg.com/toposort/-/toposort-1.0.7.tgz#2e68442d9f64ec720b8cc89e6443ac6caa950029"
   integrity sha1-LmhELZ9k7HILjMieZEOsbKqVACk=
 
-tough-cookie@>=2.3.3, tough-cookie@^2.3.4:
+tough-cookie@>=2.3.3, tough-cookie@^2.2.0, tough-cookie@^2.3.4:
   version "2.5.0"
   resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.5.0.tgz#cd9fb2a0aa1d5a12b473bd9fb96fa3dcff65ade2"
   integrity sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==
@@ -10380,6 +10517,11 @@
   dependencies:
     punycode "^2.1.0"
 
+tr46@~0.0.3:
+  version "0.0.3"
+  resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a"
+  integrity sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=
+
 tree-kill@^1.1.0:
   version "1.2.1"
   resolved "https://registry.yarnpkg.com/tree-kill/-/tree-kill-1.2.1.tgz#5398f374e2f292b9dcc7b2e71e30a5c3bb6c743a"
@@ -10883,6 +11025,11 @@
   dependencies:
     defaults "^1.0.3"
 
+webidl-conversions@^3.0.0, webidl-conversions@^3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"
+  integrity sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=
+
 webidl-conversions@^4.0.2:
   version "4.0.2"
   resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-4.0.2.tgz#a855980b1f0b6b359ba1d5d9fb39ae941faa63ad"
@@ -11052,6 +11199,14 @@
   resolved "https://registry.yarnpkg.com/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz#3d4b1e0312d2079879f826aff18dbeeca5960fbf"
   integrity sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==
 
+whatwg-url@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-2.0.1.tgz#5396b2043f020ee6f704d9c45ea8519e724de659"
+  integrity sha1-U5ayBD8CDub3BNnEXqhRnnJN5lk=
+  dependencies:
+    tr46 "~0.0.3"
+    webidl-conversions "^3.0.0"
+
 whatwg-url@^6.4.1:
   version "6.5.0"
   resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-6.5.0.tgz#f2df02bff176fd65070df74ad5ccbb5a199965a8"
@@ -11172,11 +11327,21 @@
   dependencies:
     async-limiter "~1.0.0"
 
+"xml-name-validator@>= 2.0.1 < 3.0.0":
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-2.0.1.tgz#4d8b8f1eccd3419aa362061becef515e1e559635"
+  integrity sha1-TYuPHszTQZqjYgYb7O9RXh5VljU=
+
 xml-name-validator@^3.0.0:
   version "3.0.0"
   resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-3.0.0.tgz#6ae73e06de4d8c6e47f9fb181f78d648ad457c6a"
   integrity sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==
 
+xmldom@^0.1.22:
+  version "0.1.27"
+  resolved "https://registry.yarnpkg.com/xmldom/-/xmldom-0.1.27.tgz#d501f97b3bdb403af8ef9ecc20573187aadac0e9"
+  integrity sha1-1QH5ezvbQDr4757MIFcxh6rawOk=
+
 xregexp@2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/xregexp/-/xregexp-2.0.0.tgz#52a63e56ca0b84a7f3a5f3d61872f126ad7a5943"