diff --git a/package.json b/package.json index f5f3c2b..d453f9d 100644 --- a/package.json +++ b/package.json @@ -23,7 +23,7 @@ "postcss": "8.4.21", "postcss-cli": "10.1.0", "sass-lint": "1.13.1", - "vanilla-framework": "3.13.0", + "vanilla-framework": "4.16.0", "watch-cli": "0.2.3" }, "devDependencies": { diff --git a/static/sass/main.scss b/static/sass/main.scss index 8b8b8f4..13e3e2d 100644 --- a/static/sass/main.scss +++ b/static/sass/main.scss @@ -10,18 +10,13 @@ } } -.asset-thumbnail { +.p-asset-card--thumbnail { max-height: 100px; min-height: 100px; } -.asset-additional-edit-row { - display: flex; - flex-direction: row; - justify-content: space-between; - align-items: center; - flex-wrap: wrap; - & > * { - margin: 0 !important; - } -} +.p-asset-card { + padding:0.75rem; + background:#f7f7f7; + margin-top:2rem; +} \ No newline at end of file diff --git a/templates/_asset-card-image.html b/templates/_asset-card-image.html new file mode 100644 index 0000000..b7a78c9 --- /dev/null +++ b/templates/_asset-card-image.html @@ -0,0 +1,51 @@ +
+
+ {% if asset.data.image %} + + + + {% endif %} +
+
+ {% if asset.name %} +

{{ asset.name }}

+ {% else %} +

{{ asset.file_path.split('.')[0] }}

+ {% endif %} +

File type: .{{ asset.file_path.split('.', 1)[1] }}

+

Image size: {% if asset.data.width and asset.data.height %}{{asset.data.width}} x {{asset.data.height}}{% endif %}

+

+ Tags: + {% for tag in asset.tags %} + {{ tag.name }}{% if not loop.last %},{% endif %} + {% endfor %} +

+

+ Date created: {{ asset.created.strftime('%d %B %Y') }} +

+
+
+

+ Full asset details › +

+
+
+

+ + + Edit asset + +

+
+
diff --git a/templates/_asset-card.html b/templates/_asset-card.html new file mode 100644 index 0000000..453df9c --- /dev/null +++ b/templates/_asset-card.html @@ -0,0 +1,48 @@ +
+
+ {% if asset.asset_type == "whitepaper" %} + + {% else %} + + {% endif %} +
+
+ {% if asset.name %} +

{{ asset.name }}

+ {% else %} +

{{ asset.file_path.split('.')[0] }}

+ {% endif %} +

File type: .{{ asset.file_path.split('.', 1)[1] }}

+

Asset type: {{ asset.asset_type }}

+

+ Tags: + {% for tag in asset.tags %} + {{ tag.name }}{% if not loop.last %},{% endif %} + {% endfor %} +

+

+ Date created: {{ asset.created.strftime('%d %B %Y') }} +

+
+
+

+ Full asset details › +

+
+
+

+ + + Edit asset + +

+
+
diff --git a/templates/_asset-list.html b/templates/_asset-list.html index 0f5974d..f561c60 100644 --- a/templates/_asset-list.html +++ b/templates/_asset-list.html @@ -1,3 +1,7 @@ {% for asset in assets %} - {% include "_asset.html" %} + {% if asset.data.image %} + {% include "_asset-card-image.html" %} + {% else %} + {% include "_asset-card.html" %} + {% endif %} {% endfor %} \ No newline at end of file diff --git a/templates/_asset.html b/templates/_asset.html deleted file mode 100644 index 4b1e1fa..0000000 --- a/templates/_asset.html +++ /dev/null @@ -1,57 +0,0 @@ -
- {% if asset.data.image %} - - - -
- {% endif %} -
-

{{ asset.file_path }}

-

- - Tags: {% for tag in asset.tags %} - - {{tag.name}} - - {% endfor %} - -

-

- Created: {{ asset.created.strftime('%d/%m/%Y, %H:%M') }} -

-

- Copy: - - {% if asset.data.image %} - - {% endif %} -

-

- - - Edit - - {% if asset.deprecated %} - - Deprecated - - {% endif %} -

-
-
diff --git a/templates/created.html b/templates/created.html index 07c2b22..0ec2442 100644 --- a/templates/created.html +++ b/templates/created.html @@ -13,14 +13,14 @@

Failed to upload {{failed|length}} assets

{% endif %} {% if assets %} -
+

{{assets|length}} assets created

{% include "_asset-list.html" %}
{% endif %} {% if existing %} -
+
{% set assets = existing %}

{{existing|length}} existing assets

{% include "_asset-list.html" %} diff --git a/templates/details.html b/templates/details.html new file mode 100644 index 0000000..35742ac --- /dev/null +++ b/templates/details.html @@ -0,0 +1,58 @@ +{% extends "_layout.html" %} + +{% block title %}Asset details{% endblock %} + +{% block content %} +
+
+
+

Asset details

+ {% if asset.data.image %} + + + + {% endif %} +

File path:{{ file_path }}

+

Name: {{ asset.name }}

+

File type: .{{ asset.file_path.split('.', 1)[1] }}

+

Asset type: {{ asset.asset_type }}

+

Image size: {% if asset.data.width and asset.data.height %}{{asset.data.width}} x {{asset.data.height}}{% else %}Unknown{% endif %}

+

+ Products: + {% for product in asset.products %} + {{ product.name }}{% if not loop.last %},{% endif %} + {% endfor %} +

+

+ Tags: + {% for tag in asset.tags %} + {{ tag.name }}{% if not loop.last %},{% endif %} + {% endfor %} +

+

Created: {{ asset.created.strftime('%d %B %Y') }}

+

Updated: {{ asset.updated.strftime('%d %B %Y') }}

+

Google drive link: {{ asset.google_drive_link }}

+

Author email: {{ asset.author_email }}

+

Language: {{ asset.language }}

+

Salesforce campaign ID: {{ asset.salesforce_campaign_id }}

+

Deprecated: {{ asset.deprecated }}

+

+ + +

+
+
+
+{% endblock %} \ No newline at end of file diff --git a/templates/index.html b/templates/index.html index 9db02a0..d353c41 100644 --- a/templates/index.html +++ b/templates/index.html @@ -34,7 +34,7 @@
-
+
{% if assets %} {% include "_asset-list.html" %} {% elif request.values.get("q", None) %} diff --git a/templates/update.html b/templates/update.html index 9c7561a..fe7c1ce 100644 --- a/templates/update.html +++ b/templates/update.html @@ -24,7 +24,7 @@

{{ asset.file_path or request.args.get("file-path")}}

{% if asset.data.image %} {% endif %} diff --git a/webapp/routes.py b/webapp/routes.py index ca84ed4..cf2db57 100644 --- a/webapp/routes.py +++ b/webapp/routes.py @@ -81,7 +81,7 @@ def home(): ) -@ui_blueprint.route("/create", methods=["POST"]) +@ui_blueprint.route("/create", methods=["GET", "POST"]) @login_required def create(): created_assets = [] @@ -195,6 +195,16 @@ def update(): return flask.render_template("update.html", asset=asset) +@ui_blueprint.route("/details", methods=["GET"]) +@login_required +def details(): + file_path = request.args.get("file-path") + + asset = asset_service.find_asset(file_path) + if not asset: + flask.flash("Asset not found", "negative") + + return flask.render_template("details.html", asset=asset) # API Routes # === diff --git a/webapp/services.py b/webapp/services.py index d490921..3fb2ee4 100644 --- a/webapp/services.py +++ b/webapp/services.py @@ -1,11 +1,13 @@ # System import imghdr -from base64 import b64decode +from base64 import b64decode, b64encode from datetime import datetime, timezone +from io import BytesIO # Packages from wand.image import Image from typing import List +from PIL import Image as PillowImage # Local from webapp.database import db_session @@ -98,7 +100,11 @@ def create_asset( friendly_name = clean_unicode(friendly_name) url_path = clean_unicode(url_path) - encoded_file_content = (b64decode(file_content),) + # First we ensure it is b64 encoded + encoded_file_content = b64encode(file_content) + # Then we can decode it + decoded_file_content = (b64decode(encoded_file_content)) + if imghdr.what(None, h=file_content) is not None or is_svg( file_content ): @@ -107,30 +113,33 @@ def create_asset( # As it's not an image, there is no need for optimization data["optimized"] = False + if data.get("image"): + try: + # Use Pillow to open the image and get dimensions + with PillowImage.open(BytesIO(decoded_file_content)) as img: + data["width"] = img.width + data["height"] = img.height + except Exception as e: + print(f"Error opening image with Pillow: {e}") + data["width"] = None + data["height"] = None + # Try to optimize the asset if it's an image if data.get("image") and optimize: try: - image = ImageProcessor(encoded_file_content) + image = ImageProcessor(decoded_file_content) image.optimize(allow_svg_errors=True) - encoded_file_content = image.data + decoded_file_content = image.data data["optimized"] = True except Exception: # If optimisation failed, just don't bother optimising data["optimized"] = False + if not url_path: url_path = file_manager.generate_asset_path( file_content, friendly_name ) - if data.get("image"): - try: - with Image(blob=encoded_file_content) as image_info: - data["width"] = image_info.width - data["height"] = image_info.height - except Exception: - # Just don't worry if image reading fails - pass - asset = ( db_session.query(Asset) .filter(Asset.file_path == url_path) diff --git a/yarn.lock b/yarn.lock index 5c49fa0..40027e7 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2,16 +2,6 @@ # yarn lockfile v1 -"@canonical/cookie-policy@3.4.0": - version "3.4.0" - resolved "https://registry.yarnpkg.com/@canonical/cookie-policy/-/cookie-policy-3.4.0.tgz#0d6708da340df5867fd2cc9dbd95538c46f20cf8" - integrity sha512-cdVqxQmGu+j+Q86UobihWWVFzGzHlekFeMFxlbRpm+yqxEOUCrLkA9/t/RsMfLNDToP2ECPgsMbS20aPlA2tIg== - -"@canonical/latest-news@1.4.1": - version "1.4.1" - resolved "https://registry.yarnpkg.com/@canonical/latest-news/-/latest-news-1.4.1.tgz#dcdd445ac2268a54cf60f2f8c725b6bdeb285d71" - integrity sha512-lwrikCj0Y11X8Ln8D+elp6Ri3mYjMjsAOJtadNEFzhBrgmg8TVGYJjOdwy8TvRaxy7fHnvVajIKAYWX44Tfj6w== - "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -115,23 +105,6 @@ argparse@^1.0.7: dependencies: sprintf-js "~1.0.2" -array-union@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/array-union/-/array-union-3.0.1.tgz#da52630d327f8b88cfbfb57728e2af5cd9b6b975" - integrity sha512-1OvF9IbWwaeiM9VhzYXVQacMibxpXOMYVNIvMtKRyX9SImBXpKcFr8XvFDeEslCyuH/t6KRt7HEO94AlP8Iatw== - -autoprefixer@10.4.13: - version "10.4.13" - resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-10.4.13.tgz#b5136b59930209a321e9fa3dca2e7c4d223e83a8" - integrity sha512-49vKpMqcZYsJjwotvt4+h/BCjJVnhGwcLpDt5xkcaOG3eLrG/HUYLagrihYsQ+qrIBgIzX1Rw7a6L8I/ZA1Atg== - dependencies: - browserslist "^4.21.4" - caniuse-lite "^1.0.30001426" - fraction.js "^4.2.0" - normalize-range "^0.1.2" - picocolors "^1.0.0" - postcss-value-parser "^4.2.0" - autoprefixer@10.4.14: version "10.4.14" resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-10.4.14.tgz#e28d49902f8e759dd25b153264e862df2705f79d" @@ -169,7 +142,7 @@ braces@^3.0.1, braces@~3.0.2: dependencies: fill-range "^7.0.1" -browserslist@^4.21.4, browserslist@^4.21.5: +browserslist@^4.21.5: version "4.21.5" resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.21.5.tgz#75c5dae60063ee641f977e00edd3cfb2fb7af6a7" integrity sha512-tUkiguQGW7S3IhB7N+c2MV/HZPSCPAAiYBZXLsBhFB/PCy6ZKKsZrmBayHV9fdGV/ARIfJ14NkxKzRDjvp7L6w== @@ -196,7 +169,7 @@ callsites@^0.2.0: resolved "https://registry.yarnpkg.com/callsites/-/callsites-0.2.0.tgz#afab96262910a7f33c19a5775825c69f34e350ca" integrity sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo= -caniuse-lite@^1.0.30001426, caniuse-lite@^1.0.30001449, caniuse-lite@^1.0.30001464: +caniuse-lite@^1.0.30001449, caniuse-lite@^1.0.30001464: version "1.0.30001477" resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001477.tgz#a2ffb2276258233034bbb869d4558b02658a511e" integrity sha512-lZim4iUHhGcy5p+Ri/G7m84hJwncj+Kz7S5aD4hoQfslKZJgt0tHc/hafVbqHC5bbhHb+mrW2JOUHkI5KH7toQ== @@ -583,7 +556,7 @@ ext@^1.1.2: dependencies: type "^2.5.0" -fast-glob@^3.2.11, fast-glob@^3.2.7: +fast-glob@^3.2.11: version "3.2.12" resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.2.12.tgz#7f39ec99c2e6ab030337142da9e0c18f37afae80" integrity sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w== @@ -651,15 +624,6 @@ front-matter@2.1.2: dependencies: js-yaml "^3.4.6" -fs-extra@^10.0.0: - version "10.1.0" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-10.1.0.tgz#02873cfbc4084dde127eaa5f9905eef2325d1abf" - integrity sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ== - dependencies: - graceful-fs "^4.2.0" - jsonfile "^6.0.1" - universalify "^2.0.0" - fs-extra@^11.0.0: version "11.1.1" resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-11.1.1.tgz#da69f7c39f3b002378b0954bb6ae7efdc0876e2d" @@ -764,18 +728,6 @@ globals@^9.2.0: resolved "https://registry.yarnpkg.com/globals/-/globals-9.18.0.tgz#aa3896b3e69b487f17e31ed2143d69a8e30c2d8a" integrity sha512-S0nG3CLEQiY/ILxqtztTWH/3iRRdyBLw6KMDxnKMchrtbj2OFmehVh0WUCfW3DUrIgx/qFrJPICrq4Z4sTR9UQ== -globby@^12.0.0: - version "12.2.0" - resolved "https://registry.yarnpkg.com/globby/-/globby-12.2.0.tgz#2ab8046b4fba4ff6eede835b29f678f90e3d3c22" - integrity sha512-wiSuFQLZ+urS9x2gGPl1H5drc5twabmm4m2gTR27XDFyjUHJUNsS8o/2aKyIF6IoBaR630atdher0XJ5g6OMmA== - dependencies: - array-union "^3.0.1" - dir-glob "^3.0.1" - fast-glob "^3.2.7" - ignore "^5.1.9" - merge2 "^1.4.1" - slash "^4.0.0" - globby@^13.0.0: version "13.1.3" resolved "https://registry.yarnpkg.com/globby/-/globby-13.1.3.tgz#f62baf5720bcb2c1330c8d4ef222ee12318563ff" @@ -849,7 +801,7 @@ ignore@^3.1.2: resolved "https://registry.yarnpkg.com/ignore/-/ignore-3.3.10.tgz#0a97fb876986e8081c631160f8f9f389157f0043" integrity sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug== -ignore@^5.1.9, ignore@^5.2.0: +ignore@^5.2.0: version "5.2.4" resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.4.tgz#a291c0c6178ff1b960befe47fcdec301674a6324" integrity sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ== @@ -859,20 +811,6 @@ immutable@^4.0.0: resolved "https://registry.yarnpkg.com/immutable/-/immutable-4.3.0.tgz#eb1738f14ffb39fd068b1dbe1296117484dd34be" integrity sha512-0AOCmOip+xgJwEVTQj1EfiDDOkPmuyllDuTuEX+DDXUgapLAsBIfkg3sxCYyCEA8mQqZrrxPUGjcOQ2JS3WLkg== -import-cwd@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/import-cwd/-/import-cwd-3.0.0.tgz#20845547718015126ea9b3676b7592fb8bd4cf92" - integrity sha512-4pnzH16plW+hgvRECbDWpQl3cqtvSofHWh44met7ESfZ8UZOWWddm8hEyDTqREJ9RbYHY8gi8DqmaelApoOGMg== - dependencies: - import-from "^3.0.0" - -import-from@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/import-from/-/import-from-3.0.0.tgz#055cfec38cd5a27d8057ca51376d7d3bf0891966" - integrity sha512-CiuXOFFSzkU5x/CR0+z7T91Iht4CXgfCxVOFRhh2Zyhg5wOpWvvDLQUsWl+gcN+QscYBjez8hDCt85O7RLDttQ== - dependencies: - resolve-from "^5.0.0" - imurmurhash@^0.1.4: version "0.1.4" resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" @@ -1046,11 +984,6 @@ levn@^0.3.0, levn@~0.3.0: prelude-ls "~1.1.2" type-check "~0.3.2" -lilconfig@^2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-2.0.3.tgz#68f3005e921dafbd2a2afb48379986aa6d2579fd" - integrity sha512-EHKqr/+ZvdKCifpNrJCKxBTgk5XupZA3y/aCPY9mxfgBzmgh93Mt/WqjjQ38oMxXuvDokaKiM3lAgvSH2sjtHg== - lilconfig@^2.0.5: version "2.1.0" resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-2.1.0.tgz#78e23ac89ebb7e1bfbf25b18043de756548e7f52" @@ -1288,33 +1221,6 @@ postcss-cli@10.1.0: slash "^5.0.0" yargs "^17.0.0" -postcss-cli@9.1.0: - version "9.1.0" - resolved "https://registry.yarnpkg.com/postcss-cli/-/postcss-cli-9.1.0.tgz#1a86404cbe848e370127b4bdf5cd2be83bc45ebe" - integrity sha512-zvDN2ADbWfza42sAnj+O2uUWyL0eRL1V+6giM2vi4SqTR3gTYy8XzcpfwccayF2szcUif0HMmXiEaDv9iEhcpw== - dependencies: - chokidar "^3.3.0" - dependency-graph "^0.11.0" - fs-extra "^10.0.0" - get-stdin "^9.0.0" - globby "^12.0.0" - picocolors "^1.0.0" - postcss-load-config "^3.0.0" - postcss-reporter "^7.0.0" - pretty-hrtime "^1.0.3" - read-cache "^1.0.0" - slash "^4.0.0" - yargs "^17.0.0" - -postcss-load-config@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/postcss-load-config/-/postcss-load-config-3.1.0.tgz#d39c47091c4aec37f50272373a6a648ef5e97829" - integrity sha512-ipM8Ds01ZUophjDTQYSVP70slFSYg3T0/zyfII5vzhN6V57YSxMgG5syXuwi5VtS8wSf3iL30v0uBdoIVx4Q0g== - dependencies: - import-cwd "^3.0.0" - lilconfig "^2.0.3" - yaml "^1.10.2" - postcss-load-config@^4.0.0: version "4.0.1" resolved "https://registry.yarnpkg.com/postcss-load-config/-/postcss-load-config-4.0.1.tgz#152383f481c2758274404e4962743191d73875bd" @@ -1335,11 +1241,6 @@ postcss-reporter@^7.0.0: lodash.sortby "^4.7.0" picocolors "^1.0.0" -postcss-scss@4.0.6: - version "4.0.6" - resolved "https://registry.yarnpkg.com/postcss-scss/-/postcss-scss-4.0.6.tgz#5d62a574b950a6ae12f2aa89b60d63d9e4432bfd" - integrity sha512-rLDPhJY4z/i4nVFZ27j9GqLxj1pwxE80eAzUNRMXtcpipFYIeowerzBgG3yJhMtObGEXidtIgbUpQ3eLDsf5OQ== - postcss-value-parser@^4.2.0: version "4.2.0" resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz#723c09920836ba6d3e5af019f92bc0971c02e514" @@ -1433,11 +1334,6 @@ resolve-from@^1.0.0: resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-1.0.1.tgz#26cbfe935d1aeeeabb29bc3fe5aeb01e93d44226" integrity sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY= -resolve-from@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-5.0.0.tgz#c35225843df8f776df21c57557bc087e9dfdfc69" - integrity sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw== - restore-cursor@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-1.0.1.tgz#34661f46886327fed2991479152252df92daa541" @@ -1509,15 +1405,6 @@ sass-lint@1.13.1: path-is-absolute "^1.0.0" util "^0.10.3" -sass@1.58.3: - version "1.58.3" - resolved "https://registry.yarnpkg.com/sass/-/sass-1.58.3.tgz#2348cc052061ba4f00243a208b09c40e031f270d" - integrity sha512-Q7RaEtYf6BflYrQ+buPudKR26/lH+10EmO9bBqbmPh/KeLqv8bjpTNqxe71ocONqXq+jYiCbpPUmQMS+JJPk4A== - dependencies: - chokidar ">=3.0.0 <4.0.0" - immutable "^4.0.0" - source-map-js ">=0.6.2 <2.0.0" - sass@1.61.0: version "1.61.0" resolved "https://registry.yarnpkg.com/sass/-/sass-1.61.0.tgz#d1f6761bb833887b8fdab32a24e052c40531d02b" @@ -1760,19 +1647,10 @@ util@^0.10.3: dependencies: inherits "2.0.3" -vanilla-framework@3.13.0: - version "3.13.0" - resolved "https://registry.yarnpkg.com/vanilla-framework/-/vanilla-framework-3.13.0.tgz#264f6f33e6926e6d1019e7659660f36fa20cd4b4" - integrity sha512-L6Z+8q4nbUTl12yCDJ/TqdWRugFeCaPMxUV7o0qrI1/bmuOLGpDaVPhGOcBPIO+uzIxANICVABXvk0qvzAClxA== - dependencies: - "@canonical/cookie-policy" "3.4.0" - "@canonical/latest-news" "1.4.1" - autoprefixer "10.4.13" - postcss "8.4.21" - postcss-cli "9.1.0" - postcss-scss "4.0.6" - sass "1.58.3" - yaml "1.10.2" +vanilla-framework@4.16.0: + version "4.16.0" + resolved "https://registry.yarnpkg.com/vanilla-framework/-/vanilla-framework-4.16.0.tgz#54e7a51e073de043d45a7bac37ffaa4f4351ac4a" + integrity sha512-LrxZLiNOm0cTpG++1X4MMy2efg8Xhc08JUAwNAybeSQ5FaaBGiAodbV1Fx3QvJxlaPFQqsjOdT6uZDwuOD7YJg== verbalize@^0.1.2: version "0.1.2" @@ -1828,11 +1706,6 @@ y18n@^5.0.5: resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== -yaml@1.10.2, yaml@^1.10.2: - version "1.10.2" - resolved "https://registry.yarnpkg.com/yaml/-/yaml-1.10.2.tgz#2301c5ffbf12b467de8da2333a459e29e7920e4b" - integrity sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg== - yaml@^2.1.1: version "2.2.1" resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.2.1.tgz#3014bf0482dcd15147aa8e56109ce8632cd60ce4"