Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
css-api-fetch
Advanced tools
Make API Requests in CSS and store the response data on :root --vars without JS.
Make remote API Requests in CSS (Cascading Style Sheets) and store the response data in --vars
on :root
without JavaScript.
Curious how this works? Read about it here!
There are 3 css files you can use with different trade-offs.
api-fetch.css
- contains both of the following
api-fetch-root.css
:root
and can be used anywhere.api-fetch-compat.css
:root
, it can only be used within the element.url()
css-api-fetch
requires specific html in additon to the CSS.
$ npm install css-api-fetch
Then include /node_modules/css-api-fetch/api-fetch.css
From html:
<link rel="stylesheet" type="text/css" href="https://unpkg.com/css-api-fetch@3/api-fetch.css">
or directly from your CSS:
@import url(https://unpkg.com/css-api-fetch@3/api-fetch.css);
See ./html-templates-compat.md
for instructions on the compat
version setup and usage.
For the root
version, add a single tag anywhere on the page per api you want to use:
<div class="api-1-fetch"></div>
You can save responses to :root
from up to 4 different API requests at the same time.
api-1-fetch
api-2-fetch
api-3-fetch
api-4-fetch
Use separate html elements for each of these, do not nest anything inside.
If you wish to see debug info, add this tag anywhere on the page:
<div class="api-debug-on"></div>
Both versions, root
and compat
, can be used at the same time.
You can specify any of the API endpoints anywhere in your CSS so long as the var is visible to the corresponding api-fetch-X
tag:
body {
--api-1-fetch: url(https://css-api.propjockey.io/os-country.php);
--api-2-fetch: url(https://picsum.photos/512/256);
--api-3-fetch: url(https://picsum.photos/100/222);
--api-4-fetch: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="0px" height="99999px"></svg>');
}
You may want to only fetch this API data in specific app state conditions.
You can easily use container style queries to conditionally use an endpoint:
@container style(--amazing-computation: 0) {
.api-2-fetch { --api-2-fetch: none; }
}
@container style(--amazing-computation: 1) {
.api-2-fetch { --api-2-fetch: url(...) }
}
...
Added in v 3.1.0 - no functionality changes unless you set these.
In the root
version, you can configure the maximum response size by setting a variable on :root
:
:root { --api-1-max-w: <integer>; }
// maximum width returned from api 1
:root { --api-4-max-h: <integer>; }
// maximum height returned from api 4
You can be less specific and specify both height and width at once:
:root { --api-2-max: <integer>; }
// maximum height and width returned from api 2
You can be even less specific and specify a maximum height OR width for all api results:
:root { --api-max-w: <integer>; }
// maximum width returned from all apis
:root { --api-max-h: <integer>; }
// maximum height returned from all apis
You can be as non-specific as possible and specify the maximum for height AND width of all api responses:
:root { --api-max: <integer>; }
// maximum height and width returned from all APIs.
Higher specificity overrides lower specificity.
The default is the lowest specificity, and you can set it to whatever you want:
:root { --api-max: 99999; }
css-api-fetch
expects an image response with the response data encoded into the height and width.
Unless overwritten within the :root
version, the maximum width in both versions is 99999px
which is more than 16 bits of data.
Unless overwritten within the :root
version, the maximum height in both versions is also 99999px
which is more than 32 bits of data total.
Here, is php generating an svg that encodes the 32 bit request IP Address:
<?php
header('Content-type: image/svg+xml');
$adr = $_SERVER['REMOTE_ADDR'] ?: '255.255.255.255';
$hex = str_pad(implode(array_map('dechex', explode('.', $adr, 4))), 8, '0', STR_PAD_LEFT);
$width = hexdec(preg_replace('/(^0{0,3})|(.{4}$)/', '', $hex) ?: '0');
$height = hexdec(preg_replace('/^.{4}0{0,3}/', '', $hex) ?: '0');
echo '<svg xmlns="http://www.w3.org/2000/svg" width="' . $width . 'px" height="' . $height . 'px"></svg>';
?>
16 bits in width, 16 bits in height
Here is a live example of this in action:
Once a request is complete, response data from the width and height of the image will be returned and set on root as an integer.
@property --api-1-w { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-1-h { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-2-w { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-2-h { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-3-w { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-3-h { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-4-w { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-4-h { syntax: "<integer>"; inherits: true; initial-value: 0; }
You can use calc()
and other techniques to do any decoding necessary. Please reach out if you have a specific goal, there's not much that can't be done yet.
For example, css-bin-bits can help you convert 16 bit decimal numbers between decimal and binary and perform bitwise operations on the values without JS.
Additionally, there is a ready bit available for all 4 api ids that will be set to 1
when it has any non-0 data:
@property --api-1-ready { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-2-ready { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-3-ready { syntax: "<integer>"; inherits: true; initial-value: 0; }
@property --api-4-ready { syntax: "<integer>"; inherits: true; initial-value: 0; }
Kizu for suggesting a different way to lift data to root and for providing a firefox precision fix in the compat version.
T. Afif for this article demonstrating the concept of using view timelines as a better way to measure and pass around element sizes instead of using my own tan(atan2()) approach to do it.
Bramus for writing many articles on scroll and view timelines and building amazing tools so I could learn what I needed to make it uniquely and accurately work for this.
Please do reach out if you need help with any of this, have feature requests, want to share what you've created, or wish to learn more.
PropJockey.io | CodePen | DEV Blog | GitHub | Mastodon |
---|---|---|---|---|
FAQs
Make API Requests in CSS and store the response data on :root --vars without JS.
The npm package css-api-fetch receives a total of 312 weekly downloads. As such, css-api-fetch popularity was classified as not popular.
We found that css-api-fetch demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 0 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.