Skip to content

Commit acd21e6

Browse files
committed
test: added unity tests to lib
2 parents ec6baf9 + 9458fa5 commit acd21e6

29 files changed

+22912
-7859
lines changed
Lines changed: 57 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
name: Release
2+
on:
3+
release:
4+
types: [published]
5+
workflow_dispatch:
6+
7+
jobs:
8+
tag:
9+
name: Add/update 'latest' tag
10+
runs-on: ubuntu-latest
11+
12+
steps:
13+
- name: Checkout repository
14+
uses: actions/checkout@v4
15+
16+
- name: Run latest-tag
17+
uses: EndBug/latest-tag@latest
18+
env:
19+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
20+
21+
test:
22+
name: Testing the lib
23+
runs-on: ubuntu-latest
24+
needs: tag
25+
26+
steps:
27+
- name: Checkout repository
28+
uses: actions/checkout@v4
29+
30+
- name: Setup Node 20
31+
uses: actions/setup-node@v4
32+
with:
33+
node-version: 20
34+
35+
- name: Install dependencies
36+
run: npm install
37+
38+
- name: Run the test suite
39+
run: npm run test
40+
41+
publish-npm:
42+
name: Publish on NPM
43+
runs-on: ubuntu-latest
44+
needs: test
45+
46+
steps:
47+
- uses: actions/checkout@v4
48+
49+
- name: Set up Node.js for NPM
50+
uses: actions/setup-node@v4
51+
with:
52+
registry-url: 'https://registry.npmjs.org'
53+
54+
- run: npm install
55+
- run: npm publish
56+
env:
57+
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}

β€ŽREADME.mdβ€Ž

Lines changed: 58 additions & 16 deletions
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77

88
[![Version](https://img.shields.io/npm/v/react-lite-youtube-embed?label=latest%20version)](https://www.npmjs.com/package/react-lite-youtube-embed)      ![Total Downloads](https://img.shields.io/npm/dt/react-lite-youtube-embed?color=%23FF0000&logo=npm)      [![License](https://badgen.net/github/license/ibrahimcesar/react-lite-youtube-embed)](./LICENSE)     ![GitHub issues by-label](https://img.shields.io/github/issues/ibrahimcesar/react-lite-youtube-embed/bug)
99

10-
10+
1111
<p>Developed in πŸ‡§πŸ‡· <span role="img" aria-label="Flag for Brazil">Brazil</p>
1212

1313
<strong>Port of Paul Irish's [Lite YouTube Embed](https://github.com/paulirish/lite-youtube-embed) to a React Component. Provide videos with a supercharged focus on visual performance. The gain is not the same as the web component of the original implementation but saves some requests and gives you more control of the embed visual. An ["Adaptive Loading"](https://www.youtube.com/watch?v=puUPpVrIRkc) way to handle iframes for YouTube.</strong>
@@ -39,11 +39,11 @@ npm install react-lite-youtube-embed -S
3939
import React from "react";
4040
import { render } from "react-dom";
4141
import LiteYouTubeEmbed from 'react-lite-youtube-embed';
42-
import 'react-lite-youtube-embed/dist/LiteYouTubeEmbed.css'
42+
import 'react-lite-youtube-embed/dist/LiteYouTubeEmbed.css';
4343

4444
const App = () => (
4545
<div>
46-
<LiteYouTubeEmbed
46+
<LiteYouTubeEmbed
4747
id="L2vS_050c-M"
4848
title="What’s new in Material Design for the web (Chrome Dev Summit 2019)"
4949
/>
@@ -62,13 +62,14 @@ const App = () => (
6262
<div>
6363
<LiteYouTubeEmbed
6464
id="L2vS_050c-M" // Default none, id of the video or playlist
65-
adNetwork={true} // Default true, to preconnect or not to doubleclick addresses called by YouTube iframe (the adnetwork from Google)
65+
adNetwork={false} // Default false, to preconnect or not to doubleclick addresses called by YouTube iframe (the adnetwork from Google)
6666
params="" // any params you want to pass to the URL, assume we already had '&' and pass your parameters string
67-
playlist={false} // Use true when your ID be from a playlist
67+
playlist={false} // Use true when your ID be from a playlist
6868
playlistCoverId="L2vS_050c-M" // The ids for playlists did not bring the cover in a pattern to render so you'll need pick up a video from the playlist (or in fact, whatever id) and use to render the cover. There's a programmatic way to get the cover from YouTube API v3 but the aim of this component is do not make any another call and reduce requests and bandwidth usage as much as possibe
6969
poster="hqdefault" // Defines the image size to call on first render as poster image. Possible values are "default","mqdefault", "hqdefault", "sddefault" and "maxresdefault". Default value for this prop is "hqdefault". Please be aware that "sddefault" and "maxresdefault", high resolution images are not always avaialble for every video. See: https://stackoverflow.com/questions/2068344/how-do-i-get-a-youtube-video-thumbnail-from-the-youtube-api
7070
title="YouTube Embed" // a11y, always provide a title for iFrames: https://dequeuniversity.com/tips/provide-iframe-titles Help the web be accessible ;)
71-
noCookie={true} //Default false, connect to YouTube via the Privacy-Enhanced Mode using https://www.youtube-nocookie.com
71+
cookie={false} // Default false, don't connect to YouTube via the Privacy-Enhanced Mode using https://www.youtube-nocookie.com
72+
ref={myRef} // Use this ref prop to programmatically access the underlying iframe element. It will only have a value after the user pressed the play button
7273
/>
7374
</div>
7475
);
@@ -92,6 +93,45 @@ const App = () => (
9293
);
9394
```
9495

96+
## πŸ€– Controlling the player
97+
98+
You can programmatically control the YouTube player via [YouTubes IFrame Player API](https://developers.google.com/youtube/iframe_api_reference). However typically YouTube requires you to load an additional script from their servers (`https://www.youtube.com/iframe_api`), which is small but it will load another script. So this is neither performant nor very privacy-friendly. Instead, you can also send messages to the iframe via (`postMessage`)[https://developer.mozilla.org/en-US/docs/Web/API/Window/postMessage] using the ref prop. If you don't want to create the `postMessage()` calls yourself, there is also a little (wrapper library)[https://github.com/mich418/youtube-iframe-ctrl] for controlling the iframe with this method.
99+
100+
> [!WARNING]
101+
> This will only work if you set the `enableJsApi` prop to true. Also, the ref will only be defined, when the iframe has been loaded (which happens after clicking on the poster). So you can't start the player through this method. If you really want the player to always load the iframe right away (which is not good in terms of privacy), you can use the `alwaysLoadIframe` prop to do this.
102+
103+
```jsx
104+
const App = () => (
105+
const ytRef = useRef(null);
106+
const [isPlaying, setIsPlaying] = useState(false);
107+
108+
return (
109+
<div>
110+
<button
111+
onClick={() => {
112+
setIsPlaying((oldState) => !oldState);
113+
ytRef.current?.contentWindow?.postMessage(
114+
`{"event": "command", "func": "${isPlaying ? "pauseVideo" : "playVideo"}"}`,
115+
"*",
116+
);
117+
}}
118+
>
119+
External Play Button
120+
</button>
121+
<YouTubeNew
122+
title="My Video"
123+
id="L2vS_050c-M"
124+
ref={ytRef}
125+
enableJsApi
126+
alwaysLoadIframe
127+
/>
128+
</div>
129+
);
130+
};
131+
);
132+
133+
```
134+
95135
## ⚠️ After version 1.0.0 - BREAKING CHANGES ⚠️
96136
97137
To play nice with new frameworks like [NextJS](https://nextjs.org/), we now don't import the `.css` necessary. Since version `2.0.9` you can pass custom aspect-ratio props, so be aware of any changes needed in the CSS options. Instead use now you have three options:
@@ -121,7 +161,6 @@ Place the necessary CSS in your Global CSS file method of preference
121161
display: block;
122162
position: absolute;
123163
top: 0;
124-
background-image: url();
125164
background-position: top;
126165
background-repeat: repeat-x;
127166
height: 60px;
@@ -148,16 +187,15 @@ Place the necessary CSS in your Global CSS file method of preference
148187

149188
/* play button */
150189
.yt-lite > .lty-playbtn {
151-
width: 70px;
190+
width: 65px;
152191
height: 46px;
153-
background-color: #212121;
154192
z-index: 1;
155193
opacity: 0.8;
156-
border-radius: 14%; /* TODO: Consider replacing this with YT's actual svg. Eh. */
194+
border: none;
195+
background: url("data:image/svg+xml,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%20version%3D%221.1%22%20id%3D%22YouTube_Icon%22%20x%3D%220px%22%20y%3D%220px%22%20viewBox%3D%220%200%201024%20721%22%20enable-background%3D%22new%200%200%201024%20721%22%20xml%3Aspace%3D%22preserve%22%3E%3Cscript%20xmlns%3D%22%22%3E%0A%20%20%20%20try%20%7B%0A%20%20%20%20%20%20Object.defineProperty(navigator%2C%20%22globalPrivacyControl%22%2C%20%7B%0A%20%20%20%20%20%20%20%20value%3A%20false%2C%0A%20%20%20%20%20%20%20%20configurable%3A%20false%2C%0A%20%20%20%20%20%20%20%20writable%3A%20false%0A%20%20%20%20%20%20%7D)%3B%0A%20%20%20%20%20%20document.currentScript.parentElement.removeChild(document.currentScript)%3B%0A%20%20%20%20%7D%20catch(e)%20%7B%7D%3B%0A%20%20%20%20%20%20%3C%2Fscript%3E%0A%3Cpath%20id%3D%22Triangle%22%20fill%3D%22%23FFFFFF%22%20d%3D%22M407%2C493l276-143L407%2C206V493z%22%2F%3E%0A%3Cpath%20id%3D%22The_Sharpness%22%20opacity%3D%220.12%22%20fill%3D%22%23420000%22%20d%3D%22M407%2C206l242%2C161.6l34-17.6L407%2C206z%22%2F%3E%0A%3Cg%20id%3D%22Lozenge%22%3E%0A%09%3Cg%3E%0A%09%09%0A%09%09%09%3ClinearGradient%20id%3D%22SVGID_1_%22%20gradientUnits%3D%22userSpaceOnUse%22%20x1%3D%22512.5%22%20y1%3D%22719.7%22%20x2%3D%22512.5%22%20y2%3D%221.2%22%20gradientTransform%3D%22matrix(1%200%200%20-1%200%20721)%22%3E%0A%09%09%09%3Cstop%20offset%3D%220%22%20style%3D%22stop-color%3A%23E52D27%22%2F%3E%0A%09%09%09%3Cstop%20offset%3D%221%22%20style%3D%22stop-color%3A%23BF171D%22%2F%3E%0A%09%09%3C%2FlinearGradient%3E%0A%09%09%3Cpath%20fill%3D%22url(%23SVGID_1_)%22%20d%3D%22M1013%2C156.3c0%2C0-10-70.4-40.6-101.4C933.6%2C14.2%2C890%2C14%2C870.1%2C11.6C727.1%2C1.3%2C512.7%2C1.3%2C512.7%2C1.3%20%20%20%20h-0.4c0%2C0-214.4%2C0-357.4%2C10.3C135%2C14%2C91.4%2C14.2%2C52.6%2C54.9C22%2C85.9%2C12%2C156.3%2C12%2C156.3S1.8%2C238.9%2C1.8%2C321.6v77.5%20%20%20%20C1.8%2C481.8%2C12%2C564.4%2C12%2C564.4s10%2C70.4%2C40.6%2C101.4c38.9%2C40.7%2C89.9%2C39.4%2C112.6%2C43.7c81.7%2C7.8%2C347.3%2C10.3%2C347.3%2C10.3%20%20%20%20s214.6-0.3%2C357.6-10.7c20-2.4%2C63.5-2.6%2C102.3-43.3c30.6-31%2C40.6-101.4%2C40.6-101.4s10.2-82.7%2C10.2-165.3v-77.5%20%20%20%20C1023.2%2C238.9%2C1013%2C156.3%2C1013%2C156.3z%20M407%2C493V206l276%2C144L407%2C493z%22%2F%3E%0A%09%3C%2Fg%3E%0A%3C%2Fg%3E%0A%3C%2Fsvg%3E");
157196
transition: all 0.2s cubic-bezier(0, 0, 0.2, 1);
158197
}
159198
.yt-lite:hover > .lty-playbtn {
160-
background-color: #f00;
161199
opacity: 1;
162200
}
163201
/* play button triangle */
@@ -223,13 +261,13 @@ Not work on every framework but you can import the css directly, check what work
223261
<summary>Show me the code!</summary>
224262
225263
```ts
226-
import 'react-lite-youtube-embed/dist/LiteYouTubeEmbed.css';
264+
import 'react-lite-youtube-embed/dist/index.css';
227265
```
228266
229267
or in a *.css/scss etc:
230268
231269
```css
232-
@import "~react-lite-youtube-embed/dist/LiteYouTubeEmbed.css";
270+
@import "~react-lite-youtube-embed/dist/index.css";
233271
```
234272
235273
</details>
@@ -247,7 +285,9 @@ The most minimalist implementation requires two props: `id` from the YouTube you
247285
| announce | string | Default: `Watch`. This will added to the button announce to the final user as in `Clickable Watch, ${title}, button` , customize to match your own language #a11y #i18n |
248286
| aspectHeight | number | Default: `9`. Use this optional prop if you want a custom aspect-ratio. Please be aware of aspect height and width relation and also any custom CSS you are using. |
249287
| aspectWidth | number | Default: `16`. Use this optional prop if you want a custom aspect-ratio. Please be aware of aspect height and width relation and also any custom CSS you are using. |
250-
| cookie | boolean | Default: `false` Connect to YouTube via the Privacy-Enhanced Mode using [https://www.youtube-nocookie.com](https://www.youtube-nocookie.com). You should opt-in to allow cookies|
288+
| cookie | boolean | Default: `false`. Connect to YouTube via the Privacy-Enhanced Mode using [https://www.youtube-nocookie.com](https://www.youtube-nocookie.com). You should opt-in to allow cookies|
289+
| enableJsApi | boolean | Default: `false`. If this is enabled, you can send messages to the iframe (e.g. access via the `ref` prop) to control the player programmatically. |
290+
| alwaysLoadIframe | boolean | Default: `false`. If this is enabled, the original YouTube iframe will always be loaded right away (this is bad for privacy). |
251291
| iframeClass | string | Pass the string class for the own iFrame |
252292
| muted | boolean | If the video has sound or not. Required autoplay `true` to work |
253293
| noCookie | boolean | `Deprecated` Default `false` _use option **cookie** to opt-in_|
@@ -261,6 +301,8 @@ The most minimalist implementation requires two props: `id` from the YouTube you
261301
| thumbnail | string | Pass an optional image url to override the default poster and set a custom poster image |
262302
| webp | boolean | Default `false`. When set, uses the WebP format for poster images |
263303
| wrapperClass | string | Pass the string class that wraps the iFrame |
304+
| containerElement | string | Default `article`. Pass the element to be used for the container. |
305+
| style | object | Pass the style object to be used for the container, overriding any root styles. |
264306
265307
## πŸ™‡β€β™‚οΈ Thanks
266308
@@ -274,11 +316,11 @@ The most minimalist implementation requires two props: `id` from the YouTube you
274316
275317
### 🈺 TODO
276318
277-
- [ ] Add tests
319+
- [x] Add tests
278320
279321
## MIT License
280322
281-
Copyright (c) 2020 β€” 2022 [Ibrahim Cesar](https://ibrahimcesar.cloud)
323+
Copyright (c) 2021 β€” 2023 [Ibrahim Cesar](https://ibrahimcesar.cloud)
282324
283325
Permission is hereby granted, free of charge, to any person obtaining a copy
284326
of this software and associated documentation files (the "Software"), to deal

β€Žbabel.config.cjsβ€Ž

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
module.exports = {
2+
presets: [
3+
'@babel/preset-env',
4+
'@babel/preset-react',
5+
'@babel/preset-typescript'
6+
],
7+
plugins: [
8+
'@babel/plugin-proposal-class-properties'
9+
]
10+
};

β€Ždemo/.gitignoreβ€Ž

Lines changed: 0 additions & 53 deletions
This file was deleted.

β€Ždemo/.vscode/settings.jsonβ€Ž

Lines changed: 0 additions & 11 deletions
This file was deleted.

β€Ždemo/next.config.jsβ€Ž

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
/**
2+
* @type {import('next').NextConfig}
3+
*/
4+
const nextConfig = {
5+
output: 'export',
6+
}
7+
module.exports = nextConfig

0 commit comments

Comments
Β (0)