Skip to content

Commit

Permalink
Fixes for Chrome v38 (#456)
Browse files Browse the repository at this point in the history
Fixes for handling base64 src strings on Chrome v38, rework Image
texture fallback & detection.

This PR includes:
* Using a 1x1 PNG pixel to determine which `createImageBitmap` signature
is supported on the browser
* Fallback the image worker and main thread `ImageTexture` handling
according to the detected signature
* Support for base64 `src` on Chrome v38
* Support for `createImageBitmap(blob)` without a config object for
Chrome v50 (linked to #454)
* Support for ES5 legacy output of the `examples` directory (will be
published on https://lightning-js.github.io/renderer/ automatically)
* Upgrade `vite` and `vitest` to 2.x devDependencies, pin to TS 5.6.x
for now (as 5.7.x has issues with other dependencies)
* Upgrade Playwright 1.39 to 1.49 for automated tests
* Refactor Visual Regression test pixel matching and snapshot handling
by using [pixelmatch](https://www.npmjs.com/package/pixelmatch) and
[pngjs](https://www.npmjs.com/package/pngjs) over a homebrew detection +
upng (which is stale).
* SDF font initialization waits for image support detection before
uploading the SDF texture atlas.
* Added documentation on the fallback process and references to chrome
versions in `BROWSERS.md`
  • Loading branch information
wouterlucas authored Nov 27, 2024
2 parents 89ae392 + 095cc08 commit 2b3a866
Show file tree
Hide file tree
Showing 22 changed files with 3,757 additions and 1,674 deletions.
14 changes: 0 additions & 14 deletions .eslintignore

This file was deleted.

55 changes: 0 additions & 55 deletions .eslintrc.cjs

This file was deleted.

100 changes: 100 additions & 0 deletions BROWSERS.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
# Browser Support

LightningJS is designed with legacy devices in mind, ensuring compatibility with a wide variety of browsers. From **Chrome v38**, released in late 2014, to the latest modern browsers, LightningJS provides a reliable and efficient rendering experience.

## WebGL Compatibility

LightningJS relies on **WebGL 1.x** (based on OpenGL ES 2.0) or newer for its rendering capabilities. If WebGL 1.x is supported in the browser, LightningJS will run without issues. Here are some key points about our WebGL implementation:

- **Independent Rendering**: LightningJS is a contained renderer that uses WebGL, giving us full pixel-for-pixel control over the output.
- **No CSS Dependency**: Unlike traditional DOM/CSS rendering, LightningJS avoids reliance on CSS features, extensions, or browser-specific CSS implementations.
- **Consistency**: In our experience, WebGL support is consistent across browsers. Once LightningJS is confirmed to work in a browser, it will deliver uniform output.

## Supported Browsers and Quirks

Below is a detailed breakdown of confirmed browsers that work with LightningJS, along with the quirks or features specific to their versions:

| **Browser** | **Version** | **Quirks / Features** |
| ------------- | ----------- | ----------------------------------------------------------------------------------------------------------------------------------------------- |
| **Chrome** | v38 | Bottom line for support; introduces Promises. Lacks `createImageBitmap`, iterable `UInt8Array/UInt8ClampedArray`. `ImageData` is mostly broken. |
| | v48 | Adds support for `Blob:` and `Data:` in the Fetch API. |
| | v49 | Introduces Proxies, improving reactive property handling in Blits. Not used in the Renderer. |
| | v50 | Adds `createImageBitmap` (without options object). Enables Image Worker in a web worker. |
| | v52 | Adds `config` for `createImageBitmap`, including support for alpha channel. |
| | v54 | Enables resizing in `createImageBitmap`. |
| | v63 | Adds support for dynamic imports. |
| | v71 | Introduces `globalThis`. |
| **WPEWebKit** | 2017 | Adds support for Image Worker and `createImageBitmap`. |
| | 2.22 | Full support, including Lightning Native mode (disables DOM/CSS compositing except for `<canvas>`, `<video>`, `<audio>` tags). |
| | 2.28 | Full support. |
| | 2.36 | Adds offscreen canvas and shared array buffers with await support for multithreading. |
| | 2.46 | (Upcoming) Adds FTL support for 32-bit devices and offscreen rendering support for Lightning Native. |

For createImageBitmap the Lightning 3 renderer will use a 1x1 PNG Pixel to validate whether the createImageBitmap API is available and which version can be used.
If the createImageBitmap API is not available it will gracefully fallback to `new Image()` with a performance decrease.

Lightning 3 prefers to run on **WPEWebKit** with `Lightning Native` (also known as nonCompositedWebGL) enabled for maximum performance!
For more information please see (https://wpewebkit.org/)[https://wpewebkit.org/]

## Running Lightning in older Browsers

To run LightningJS in older browsers, you can use **Vite's legacy plugin** and specific polyfills.

### Installation

Install the required dependencies:

```bash
pnpm i -D @vitejs/plugin-legacy whatwg-fetch
```

### Configuration

Add the following to your \`vite.config.js\`:

```javascript
import legacy from '@vitejs/plugin-legacy';

export default {
plugins: [
legacy({
targets: ['chrome>=38'],
modernPolyfills: true,
additionalLegacyPolyfills: ['whatwg-fetch'],
}),
],
};
```

### Adjusting Targets

Modify the \`chrome>=38\` target to match the browser version you need to support. If the target version is **Chrome v71** or higher, or **WPEWebKit 2.22** or newer, the legacy plugin is not required. You can adjust the target in your build configuration:

```javascript
build: {
target: prodTarget,
},
esbuild: {
target: devTarget,
},
```

Define the respective targets as follows:

```javascript
const devTarget = 'es2020';
const prodTarget = 'es2019';
```

For newer browsers, you can specify a higher target to reduce the use of polyfills and leverage native browser APIs for better performance.

---

## Performance Considerations

When evaluating performance, it is essential to understand the context of your development and target devices:

- **64-bit vs. 32-bit**: Your development browser on a PC/Mac runs in 64-bit mode, utilizing higher JIT tiers than the SmartTV/STB devices targeted by LightningJS.
- **Browser Age**: Development browsers are often much newer than the browsers found on Smart TVs or set-top boxes.
- **Targeted Output**: Align your build output with the legacy plugin/targets in Vite to match the browser version for deployment. Minimizing polyfills and leveraging native browser APIs will improve performance.
- **Device Testing**: Always test on target devices early and frequently. An **RPI3b+ with WPEWebKit** or similar ([WPE](https://github.com/webplatformforembedded/buildroot)) can be a great tool for replicating target environments.
2 changes: 1 addition & 1 deletion Dockerfile
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
# Use Playwright's base image
FROM mcr.microsoft.com/playwright:v1.39.0-jammy
FROM mcr.microsoft.com/playwright:v1.49.0-jammy

# Set the working directory
WORKDIR /work
Expand Down
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,8 @@ The Lightning 3 Renderer's goal is to work with the following browser versions a

Any JavaScript language features or browser APIs that cannot be automatically transpiled or polyfilled by industry standard transpilers (such as Babel) to target these versions must be carefully considered before use.

For a more detailed and comprehensive list of browsers and their features please see [browsers](./BROWSERS.md)

## Example Tests

The Example Tests sub-project define a set of tests for various Renderer
Expand All @@ -59,6 +61,11 @@ The Example Tests can be launched with:
pnpm start
```

A hosted version can be found at:
(https://https://lightning-js.github.io/renderer/)[https://lightning-js.github.io/renderer/]

This supports modern browsers as well as Chrome 38 and above through a legacy build.

See [examples/README.md](./examples/README.md) for more info.

## Visual Regression Tests
Expand Down
50 changes: 50 additions & 0 deletions eslint.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import eslintPluginTs from '@typescript-eslint/eslint-plugin';
import tsParser from '@typescript-eslint/parser';

export default [
{
files: ['**/*.ts'],
languageOptions: {
parser: tsParser,
parserOptions: {
ecmaVersion: 2022,
sourceType: 'module',
project: [
'./tsconfig.json',
'./tsconfig.vitest.json',
'./tsconfig.cfg.json',
'./examples/tsconfig.json',
'./visual-regression/tsconfig.json',
'./performance/tsconfig.json',
],
},
},
plugins: {
'@typescript-eslint': eslintPluginTs,
},
rules: {
...eslintPluginTs.configs.recommended.rules,
'@typescript-eslint/no-unsafe-argument': 'warn',
'@typescript-eslint/no-unsafe-assignment': 'warn',
'@typescript-eslint/no-unsafe-return': 'warn',
'@typescript-eslint/no-unsafe-call': 'warn',
'@typescript-eslint/no-non-null-assertion': 'warn',
'@typescript-eslint/no-unsafe-member-access': 'warn',
'@typescript-eslint/ban-ts-comment': 'warn',
'@typescript-eslint/no-explicit-any': 'warn',
'@typescript-eslint/no-unused-vars': 'warn',
'@typescript-eslint/no-empty-object-type': 'warn',
'@typescript-eslint/no-unused-expressions': 'warn',
},
},
{
ignores: [
'**/dist/**',
'node_modules',
'**config**.ts',
'**/docs/**',
'dist-vitest',
'dist-cfg',
],
},
];
10 changes: 10 additions & 0 deletions examples/common/setupMathRandom.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
export async function setupMathRandom() {
const mt19937 = await import('@stdlib/random-base-mt19937');

const factory = mt19937.factory || mt19937.default.factory;
const rand = factory({ seed: 1234 });
Math.random = function () {
return rand() / rand.MAX;
};
console.log('Math.random overridden with mt19937');
}
22 changes: 8 additions & 14 deletions examples/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,14 +34,15 @@ import {

import { Inspector } from '@lightningjs/renderer/inspector';
import { assertTruthy } from '@lightningjs/renderer/utils';
import * as mt19937 from '@stdlib/random-base-mt19937';

import type {
ExampleSettings,
SnapshotOptions,
} from './common/ExampleSettings.js';
import { StatTracker } from './common/StatTracker.js';
import { installFonts } from './common/installFonts.js';
import { MemMonitor } from './common/MemMonitor.js';
import { setupMathRandom } from './common/setupMathRandom.js';

interface TestModule {
default: (settings: ExampleSettings) => Promise<void>;
Expand Down Expand Up @@ -141,9 +142,8 @@ async function runTest(
throw new Error(`Test "${test}" not found`);
}

// eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment
const module = await testModule();
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call

const customSettings: Partial<RendererMainSettings> =
typeof module.customSettings === 'function'
? module.customSettings(urlParams)
Expand Down Expand Up @@ -216,7 +216,6 @@ async function runTest(
memMonitor,
};

// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call
await module.default(exampleSettings);
}

Expand Down Expand Up @@ -374,20 +373,15 @@ async function runAutomation(
continue;
}
assertTruthy(testModule);
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment

// Setup Math.random to use a seeded random number generator for consistent
// results in automation mode.
await setupMathRandom();

const { automation, customSettings } = await testModule();
console.log(`Attempting to run automation for ${testName}...`);
if (automation) {
console.log(`Running automation for ${testName}...`);
// Override Math.random() as stable random number generator
// - Each test gets the same sequence of random numbers
// - This only is in effect when tests are run in automation mode
// eslint-disable-next-line @typescript-eslint/unbound-method
const factory = mt19937.factory || mt19937.default.factory;
const rand = factory({ seed: 1234 });
Math.random = function () {
return rand() / rand.MAX;
};
if (customSettings) {
console.error('customSettings not supported for automation');
} else {
Expand Down
4 changes: 3 additions & 1 deletion examples/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,9 @@
"@stdlib/random-base-mt19937": "^0.2.1"
},
"devDependencies": {
"vite": "^5.0.0"
"@vitejs/plugin-legacy": "^5.4.2",
"vite": "^5.4.8",
"whatwg-fetch": "^3.6.2"
},
"engines": {
"npm": "please-use-pnpm"
Expand Down
41 changes: 41 additions & 0 deletions examples/tests/texture-base64.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
/*
* If not stated otherwise in this file or this component's LICENSE file the
* following copyright and licenses apply:
*
* Copyright 2023 Comcast Cable Communications Management, LLC.
*
* Licensed under the Apache License, Version 2.0 (the License);
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

import type { ExampleSettings } from '../common/ExampleSettings.js';

export async function automation(settings: ExampleSettings) {
await test(settings);
await settings.snapshot();
}

/**
* Load a texture from a base64 encoded string.
* @param settings
*/
export default async function test(settings: ExampleSettings) {
const { renderer, testRoot } = settings;
const node = renderer.createNode({
x: 400,
y: 200,
width: 228,
height: 228,
parent: testRoot,
src: '',
});
}
Loading

0 comments on commit 2b3a866

Please sign in to comment.