Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow running web runtime in background windows #793

Open
wants to merge 6 commits into
base: main
Choose a base branch
from

Conversation

majaha
Copy link
Contributor

@majaha majaha commented Dec 20, 2024

This patch makes update timing more regular in various environments, including when the browser window is in the background. This allows audio and netplay to continue if the user e.g. minimises the browser window. This is achieved by switching to a setTimeout() based timing solution if the observed requestAnimationFrame rate is not a multiple of 60.

Also included are a few other relevant fixes and improvements.

This commit reworks the frame timing of the runtime ensuring a regular
update on every frame on 60 fps monitors, while still supporting other
framerates, both higher and lower. It achieves this by measuring the
framerate continuously, and performing updates in a
requestAnimationFrame() callback when the framerate is close to 60, but
switching to a setTimeout() timing scheme otherwise.

This keeps animation smooth on a 60 fps screen, but keeps updates both
full-speed and at a regular 60Hz on e.g. a 30 fps screen, important for
keeping tick-based audio smooth. This also has the effect of allowing
you to run the runtime in a minimised or background browser window. This
really improves the audio experience, allowing you to run games or music
carts in the background. This also improves the netplay experience,
as the game won't lag or halt any time a player clicks out of the
browser window or changes tab.
The web runtime doesn't run on NodeJS, but was using node types. This
was causing issues because some node types override the DOM types,
e.g. `setTimeout()`.

Adding `"types": []` to tsconfig.json filters out the types in
`node_modules/@types` and prevents them from being used, including
`node_modules/@types/node`, which is installed because it is a
dependency of our dependencies.
Setting these security headers grants us access to higher precision time. This
improves our ability to pace frame timings and make accurate performance
measurements. I hope to take advantage of this to improve the devtools in the future.
Previously favicon.ico was being requested from wasm4.org when
developing locally. This won't work if the user is offline or the
website goes down, etc. It also doesn't work with secure CORS headers,
as it is served from another domain.
Copy link
Owner

@aduros aduros left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is really clever, thanks!

I think we should only keep the game running in the background when netplay is active as a special case. By default we should continue to respect requestAnimationFrame's preference to pause when the browser is in the background, to avoid hogging the system unnecessarily. What do you think?

Please also remove this bit from the docs now that this is resolved:

<b><i>The game seems to be paused or incredibly slow?</i></b>
<p>Make sure you're not playing in a background tab! Most browsers heavily throttle pages in a background tab. To test netplay, open the game in two separate, visible windows.</p>

majaha added 2 commits January 7, 2025 10:51
Also fixes up the code that pauses audio, which was buggy and some of
which was never called.

Pausing the audio system also helps to hint to browsers that it's okay
to throttle the update loop when the menu is open.
@majaha
Copy link
Contributor Author

majaha commented Jan 7, 2025

I think we should only keep the game running in the background when netplay is active as a special case. By default we should continue to respect requestAnimationFrame's preference to pause when the browser is in the background, to avoid hogging the system unnecessarily. What do you think?

I disagree, I think we should rather be respecting the user's wishes. If the user wants to continue to listen to the music in the background, they should be able to do so. It's better that they have the choice to pause the game manually if they would like to.

Halting updates like that also interacts poorly with the audio system, especially tick-based audio. Long notes keep playing, and then are missing when you tab back to the game, it sounds pretty clunky and bad. (Also, on Firefox rAF callbacks continue at a slower rate in the background, meaning the game isn't really paused. You can put a game in the background, and then over 15 minutes later it will randomly play note. This is half-true, but Firefox seems to turn this behaviour off when there's an active audio context. It's still worth considering though.)

Allowing the game to continue running also makes development and hot-swapping nicer. You can imagine someone composing music or editing sound effects with the runtime running in the background.

I've also included a patch I had lying around that pauses the audio properly when the menu is open. This stops that strange note-holdover effect I mentioned above happening when you open the menu, but it also hints to the browser that it's okay to throttle timers while the menu is open.
We could potentially stop the game from updating at all while the menu is open, but that would require some non-trivial refactoring (making all input event-based so the menu still works. Or adding a rAF only mode that's used when the menu is open would be a hackier way of doing it. It's probably more hassle than it's worth.)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants