Merge pull request #1629 from thecodingmachine/translator
Translator: i18n
This commit is contained in:
commit
2312d2e129
5
.github/workflows/continuous_integration.yml
vendored
5
.github/workflows/continuous_integration.yml
vendored
@ -10,7 +10,6 @@ on:
|
|||||||
pull_request:
|
pull_request:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
continuous-integration-front:
|
continuous-integration-front:
|
||||||
name: "Continuous Integration Front"
|
name: "Continuous Integration Front"
|
||||||
|
|
||||||
@ -46,6 +45,10 @@ jobs:
|
|||||||
run: ./templater.sh
|
run: ./templater.sh
|
||||||
working-directory: "front"
|
working-directory: "front"
|
||||||
|
|
||||||
|
- name: "Generate i18n files"
|
||||||
|
run: yarn run typesafe-i18n
|
||||||
|
working-directory: "front"
|
||||||
|
|
||||||
- name: "Build"
|
- name: "Build"
|
||||||
run: yarn run build
|
run: yarn run build
|
||||||
env:
|
env:
|
||||||
|
4
.github/workflows/push-to-npm.yml
vendored
4
.github/workflows/push-to-npm.yml
vendored
@ -43,6 +43,10 @@ jobs:
|
|||||||
run: ./templater.sh
|
run: ./templater.sh
|
||||||
working-directory: "front"
|
working-directory: "front"
|
||||||
|
|
||||||
|
- name: "Generate i18n files"
|
||||||
|
run: yarn run typesafe-i18n
|
||||||
|
working-directory: "front"
|
||||||
|
|
||||||
- name: "Build"
|
- name: "Build"
|
||||||
run: yarn run build-typings
|
run: yarn run build-typings
|
||||||
env:
|
env:
|
||||||
|
@ -107,3 +107,7 @@ $ LIVE_RELOAD=0 docker-compose up -d
|
|||||||
# Wait 2-3 minutes for the environment to start, then:
|
# Wait 2-3 minutes for the environment to start, then:
|
||||||
$ PROJECT_DIR=$(pwd) docker-compose -f docker-compose.testcafe.yml up
|
$ PROJECT_DIR=$(pwd) docker-compose -f docker-compose.testcafe.yml up
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### A bad wording or a missing language
|
||||||
|
|
||||||
|
If you notice a translation error or missing language you can help us by following the [how to translate](docs/dev/how-to-translate.md) documentation.
|
||||||
|
76
docs/dev/how-to-translate.md
Normal file
76
docs/dev/how-to-translate.md
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
# How to translate WorkAdventure
|
||||||
|
|
||||||
|
We use the [typesafe-i18n](https://github.com/ivanhofer/typesafe-i18n) package to handle the translation.
|
||||||
|
|
||||||
|
## Add a new language
|
||||||
|
|
||||||
|
It is very easy to add a new language!
|
||||||
|
|
||||||
|
First, in the `front/src/i18n` folder create a new folder with the language code as name (the language code according to [RFC 5646](https://datatracker.ietf.org/doc/html/rfc5646)).
|
||||||
|
|
||||||
|
In the previously created folder, add a file named index.ts with the following content containing your language information (french from France in this example):
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const fr_FR: Translation = {
|
||||||
|
...en_US,
|
||||||
|
language: "Français",
|
||||||
|
country: "France",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default fr_FR;
|
||||||
|
```
|
||||||
|
|
||||||
|
## Add a new key
|
||||||
|
|
||||||
|
### Add a simple key
|
||||||
|
|
||||||
|
The keys are searched by a path through the properties of the sub-objects and it is therefore advisable to write your translation as a JavaScript object.
|
||||||
|
|
||||||
|
Please use kamelcase to name your keys!
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
```ts
|
||||||
|
{
|
||||||
|
messages: {
|
||||||
|
coffeMachine: {
|
||||||
|
start: "Coffe machine has been started!";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In the code you can translate using `$LL`:
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
|
console.log($LL.messages.coffeMachine.start());
|
||||||
|
```
|
||||||
|
|
||||||
|
### Add a key with parameters
|
||||||
|
|
||||||
|
You can also use parameters to make the translation dynamic.
|
||||||
|
Use the tag { [parameter name] } to apply your parameters in the translations
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
```ts
|
||||||
|
{
|
||||||
|
messages: {
|
||||||
|
coffeMachine: {
|
||||||
|
playerStart: "{ playerName } started the coffee machine!";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In the code you can use it like this:
|
||||||
|
|
||||||
|
```ts
|
||||||
|
$LL.messages.coffeMachine.playerStart.start({
|
||||||
|
playerName: "John",
|
||||||
|
});
|
||||||
|
```
|
@ -36,6 +36,23 @@ WA.onInit().then(() => {
|
|||||||
})
|
})
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Get the player language
|
||||||
|
|
||||||
|
```
|
||||||
|
WA.player.language: string;
|
||||||
|
```
|
||||||
|
|
||||||
|
The current language of player is available from the `WA.player.language` property.
|
||||||
|
|
||||||
|
{.alert.alert-info}
|
||||||
|
You need to wait for the end of the initialization before accessing `WA.player.language`
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
WA.onInit().then(() => {
|
||||||
|
console.log('Player language: ', WA.player.language);
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
### Get the tags of the player
|
### Get the tags of the player
|
||||||
|
|
||||||
```
|
```
|
||||||
|
@ -8,7 +8,7 @@ module.exports = {
|
|||||||
"extends": [
|
"extends": [
|
||||||
"eslint:recommended",
|
"eslint:recommended",
|
||||||
"plugin:@typescript-eslint/eslint-recommended",
|
"plugin:@typescript-eslint/eslint-recommended",
|
||||||
"plugin:@typescript-eslint/recommended-requiring-type-checking"
|
"plugin:@typescript-eslint/recommended-requiring-type-checking",
|
||||||
],
|
],
|
||||||
"globals": {
|
"globals": {
|
||||||
"Atomics": "readonly",
|
"Atomics": "readonly",
|
||||||
@ -23,7 +23,7 @@ module.exports = {
|
|||||||
},
|
},
|
||||||
"plugins": [
|
"plugins": [
|
||||||
"@typescript-eslint",
|
"@typescript-eslint",
|
||||||
"svelte3"
|
"svelte3",
|
||||||
],
|
],
|
||||||
"overrides": [
|
"overrides": [
|
||||||
{
|
{
|
||||||
@ -33,6 +33,7 @@ module.exports = {
|
|||||||
],
|
],
|
||||||
"rules": {
|
"rules": {
|
||||||
"no-unused-vars": "off",
|
"no-unused-vars": "off",
|
||||||
|
"eol-last": ["error", "always"],
|
||||||
"@typescript-eslint/no-explicit-any": "error",
|
"@typescript-eslint/no-explicit-any": "error",
|
||||||
"no-throw-literal": "error",
|
"no-throw-literal": "error",
|
||||||
// TODO: remove those ignored rules and write a stronger code!
|
// TODO: remove those ignored rules and write a stronger code!
|
||||||
|
@ -1,2 +1,5 @@
|
|||||||
src/Messages/generated
|
src/Messages/generated
|
||||||
src/Messages/JsonMessages
|
src/Messages/JsonMessages
|
||||||
|
src/i18n/i18n-svelte.ts
|
||||||
|
src/i18n/i18n-types.ts
|
||||||
|
src/i18n/i18n-util.ts
|
||||||
|
5
front/.typesafe-i18n.json
Normal file
5
front/.typesafe-i18n.json
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://unpkg.com/typesafe-i18n@2.59.0/schema/typesafe-i18n.json",
|
||||||
|
"baseLocale": "en-US",
|
||||||
|
"adapter": "svelte"
|
||||||
|
}
|
1
front/dist/resources/translations/.gitignore
vendored
Normal file
1
front/dist/resources/translations/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
*.json
|
@ -64,10 +64,12 @@
|
|||||||
"socket.io-client": "^2.3.0",
|
"socket.io-client": "^2.3.0",
|
||||||
"standardized-audio-context": "^25.2.4",
|
"standardized-audio-context": "^25.2.4",
|
||||||
"ts-proto": "^1.96.0",
|
"ts-proto": "^1.96.0",
|
||||||
"uuidv4": "^6.2.10"
|
"typesafe-i18n": "^2.59.0",
|
||||||
|
"uuidv4": "^6.2.10",
|
||||||
|
"zod": "^3.11.6"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "run-p templater serve svelte-check-watch",
|
"start": "run-p templater serve svelte-check-watch typesafe-i18n",
|
||||||
"templater": "cross-env ./templater.sh",
|
"templater": "cross-env ./templater.sh",
|
||||||
"serve": "cross-env TS_NODE_PROJECT=\"tsconfig-for-webpack.json\" webpack serve --open",
|
"serve": "cross-env TS_NODE_PROJECT=\"tsconfig-for-webpack.json\" webpack serve --open",
|
||||||
"build": "cross-env TS_NODE_PROJECT=\"tsconfig-for-webpack.json\" NODE_ENV=production webpack",
|
"build": "cross-env TS_NODE_PROJECT=\"tsconfig-for-webpack.json\" NODE_ENV=production webpack",
|
||||||
@ -79,7 +81,8 @@
|
|||||||
"svelte-check-watch": "svelte-check --fail-on-warnings --fail-on-hints --compiler-warnings \"a11y-no-onchange:ignore,a11y-autofocus:ignore,a11y-media-has-caption:ignore\" --watch",
|
"svelte-check-watch": "svelte-check --fail-on-warnings --fail-on-hints --compiler-warnings \"a11y-no-onchange:ignore,a11y-autofocus:ignore,a11y-media-has-caption:ignore\" --watch",
|
||||||
"svelte-check": "svelte-check --fail-on-warnings --fail-on-hints --compiler-warnings \"a11y-no-onchange:ignore,a11y-autofocus:ignore,a11y-media-has-caption:ignore\"",
|
"svelte-check": "svelte-check --fail-on-warnings --fail-on-hints --compiler-warnings \"a11y-no-onchange:ignore,a11y-autofocus:ignore,a11y-media-has-caption:ignore\"",
|
||||||
"pretty": "yarn prettier --write 'src/**/*.{ts,svelte}'",
|
"pretty": "yarn prettier --write 'src/**/*.{ts,svelte}'",
|
||||||
"pretty-check": "yarn prettier --check 'src/**/*.{ts,svelte}'"
|
"pretty-check": "yarn prettier --check 'src/**/*.{ts,svelte}'",
|
||||||
|
"typesafe-i18n": "typesafe-i18n --no-watch"
|
||||||
},
|
},
|
||||||
"lint-staged": {
|
"lint-staged": {
|
||||||
"*.svelte": [
|
"*.svelte": [
|
||||||
|
@ -5,6 +5,7 @@ export const isGameStateEvent = new tg.IsInterface()
|
|||||||
roomId: tg.isString,
|
roomId: tg.isString,
|
||||||
mapUrl: tg.isString,
|
mapUrl: tg.isString,
|
||||||
nickname: tg.isString,
|
nickname: tg.isString,
|
||||||
|
language: tg.isUnion(tg.isString, tg.isUndefined),
|
||||||
uuid: tg.isUnion(tg.isString, tg.isUndefined),
|
uuid: tg.isUnion(tg.isString, tg.isUndefined),
|
||||||
startLayerName: tg.isUnion(tg.isString, tg.isNull),
|
startLayerName: tg.isUnion(tg.isString, tg.isNull),
|
||||||
tags: tg.isArray(tg.isString),
|
tags: tg.isArray(tg.isString),
|
||||||
|
@ -13,6 +13,12 @@ export const setPlayerName = (name: string) => {
|
|||||||
playerName = name;
|
playerName = name;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let playerLanguage: string | undefined;
|
||||||
|
|
||||||
|
export const setPlayerLanguage = (language: string | undefined) => {
|
||||||
|
playerLanguage = language;
|
||||||
|
};
|
||||||
|
|
||||||
let tags: string[] | undefined;
|
let tags: string[] | undefined;
|
||||||
|
|
||||||
export const setTags = (_tags: string[]) => {
|
export const setTags = (_tags: string[]) => {
|
||||||
@ -61,6 +67,15 @@ export class WorkadventurePlayerCommands extends IframeApiContribution<Workadven
|
|||||||
return playerName;
|
return playerName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get language(): string {
|
||||||
|
if (playerLanguage === undefined) {
|
||||||
|
throw new Error(
|
||||||
|
"Player language not initialized yet. You should call WA.player.language within a WA.onInit callback."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return playerLanguage;
|
||||||
|
}
|
||||||
|
|
||||||
get tags(): string[] {
|
get tags(): string[] {
|
||||||
if (tags === undefined) {
|
if (tags === undefined) {
|
||||||
throw new Error("Tags not initialized yet. You should call WA.player.tags within a WA.onInit callback.");
|
throw new Error("Tags not initialized yet. You should call WA.player.tags within a WA.onInit callback.");
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
import { get } from "svelte/store";
|
import { get } from "svelte/store";
|
||||||
import type { Unsubscriber } from "svelte/store";
|
import type { Unsubscriber } from "svelte/store";
|
||||||
import { onDestroy, onMount } from "svelte";
|
import { onDestroy, onMount } from "svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let HTMLAudioPlayer: HTMLAudioElement;
|
let HTMLAudioPlayer: HTMLAudioElement;
|
||||||
let audioPlayerVolumeIcon: HTMLElement;
|
let audioPlayerVolumeIcon: HTMLElement;
|
||||||
@ -144,7 +145,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="audio-manager-reduce-conversation">
|
<div class="audio-manager-reduce-conversation">
|
||||||
<label>
|
<label>
|
||||||
reduce in conversations
|
{$LL.audio.manager.reduce()}
|
||||||
<input type="checkbox" bind:checked={decreaseWhileTalking} on:change={setDecrease} />
|
<input type="checkbox" bind:checked={decreaseWhileTalking} on:change={setDecrease} />
|
||||||
</label>
|
</label>
|
||||||
<section class="audio-manager-file">
|
<section class="audio-manager-file">
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
import ChatElement from "./ChatElement.svelte";
|
import ChatElement from "./ChatElement.svelte";
|
||||||
import { afterUpdate, beforeUpdate, onMount } from "svelte";
|
import { afterUpdate, beforeUpdate, onMount } from "svelte";
|
||||||
import { HtmlUtils } from "../../WebRtc/HtmlUtils";
|
import { HtmlUtils } from "../../WebRtc/HtmlUtils";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let listDom: HTMLElement;
|
let listDom: HTMLElement;
|
||||||
let chatWindowElement: HTMLElement;
|
let chatWindowElement: HTMLElement;
|
||||||
@ -45,7 +46,7 @@
|
|||||||
<p class="close-icon" on:click={closeChat}>×</p>
|
<p class="close-icon" on:click={closeChat}>×</p>
|
||||||
<section class="messagesList" bind:this={listDom}>
|
<section class="messagesList" bind:this={listDom}>
|
||||||
<ul>
|
<ul>
|
||||||
<li><p class="system-text">Here is your chat history:</p></li>
|
<li><p class="system-text">{$LL.chat.intro()}</p></li>
|
||||||
{#each $chatMessagesStore as message, i}
|
{#each $chatMessagesStore as message, i}
|
||||||
<li><ChatElement {message} line={i} /></li>
|
<li><ChatElement {message} line={i} /></li>
|
||||||
{/each}
|
{/each}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
import { chatMessagesStore, chatInputFocusStore } from "../../Stores/ChatStore";
|
import { chatMessagesStore, chatInputFocusStore } from "../../Stores/ChatStore";
|
||||||
|
|
||||||
export const handleForm = {
|
export const handleForm = {
|
||||||
@ -27,7 +28,7 @@
|
|||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
bind:value={newMessageText}
|
bind:value={newMessageText}
|
||||||
placeholder="Enter your message..."
|
placeholder={$LL.chat.enter()}
|
||||||
on:focus={onFocus}
|
on:focus={onFocus}
|
||||||
on:blur={onBlur}
|
on:blur={onBlur}
|
||||||
bind:this={inputElement}
|
bind:this={inputElement}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
import type { PlayerInterface } from "../../Phaser/Game/PlayerInterface";
|
import type { PlayerInterface } from "../../Phaser/Game/PlayerInterface";
|
||||||
import { requestVisitCardsStore } from "../../Stores/GameStore";
|
import { requestVisitCardsStore } from "../../Stores/GameStore";
|
||||||
|
|
||||||
@ -12,8 +13,12 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<ul class="selectMenu" style="border-top: {player.color || 'whitesmoke'} 5px solid">
|
<ul class="selectMenu" style="border-top: {player.color || 'whitesmoke'} 5px solid">
|
||||||
<li><button class="text-btn" disabled={!player.visitCardUrl} on:click={openVisitCard}>Visit card</button></li>
|
<li>
|
||||||
<li><button class="text-btn" disabled>Add friend</button></li>
|
<button class="text-btn" disabled={!player.visitCardUrl} on:click={openVisitCard}
|
||||||
|
>{$LL.chat.menu.visitCard()}</button
|
||||||
|
>
|
||||||
|
</li>
|
||||||
|
<li><button class="text-btn" disabled>{$LL.chat.menu.addFriend}</button></li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import type { Game } from "../../Phaser/Game/Game";
|
import type { Game } from "../../Phaser/Game/Game";
|
||||||
import { CustomizeScene, CustomizeSceneName } from "../../Phaser/Login/CustomizeScene";
|
import { CustomizeScene, CustomizeSceneName } from "../../Phaser/Login/CustomizeScene";
|
||||||
import { activeRowStore } from "../../Stores/CustomCharacterStore";
|
import { activeRowStore } from "../../Stores/CustomCharacterStore";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let game: Game;
|
export let game: Game;
|
||||||
|
|
||||||
@ -34,7 +35,7 @@
|
|||||||
|
|
||||||
<form class="customCharacterScene">
|
<form class="customCharacterScene">
|
||||||
<section class="text-center">
|
<section class="text-center">
|
||||||
<h2>Customize your WOKA</h2>
|
<h2>{$LL.woka.customWoka.title()}</h2>
|
||||||
</section>
|
</section>
|
||||||
<section class="action action-move">
|
<section class="action action-move">
|
||||||
<button
|
<button
|
||||||
@ -53,26 +54,29 @@
|
|||||||
<section class="action">
|
<section class="action">
|
||||||
{#if $activeRowStore === 0}
|
{#if $activeRowStore === 0}
|
||||||
<button type="submit" class="customCharacterSceneFormBack nes-btn" on:click|preventDefault={previousScene}
|
<button type="submit" class="customCharacterSceneFormBack nes-btn" on:click|preventDefault={previousScene}
|
||||||
>Return</button
|
>{$LL.woka.customWoka.navigation.return()}</button
|
||||||
>
|
>
|
||||||
{/if}
|
{/if}
|
||||||
{#if $activeRowStore !== 0}
|
{#if $activeRowStore !== 0}
|
||||||
<button type="submit" class="customCharacterSceneFormBack nes-btn" on:click|preventDefault={selectUp}
|
<button type="submit" class="customCharacterSceneFormBack nes-btn" on:click|preventDefault={selectUp}
|
||||||
>Back <img src="resources/objects/arrow_up_black.png" alt="" /></button
|
>{$LL.woka.customWoka.navigation.back()}
|
||||||
|
<img src="resources/objects/arrow_up_black.png" alt="" /></button
|
||||||
>
|
>
|
||||||
{/if}
|
{/if}
|
||||||
{#if $activeRowStore === 5}
|
{#if $activeRowStore === 5}
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
class="customCharacterSceneFormSubmit nes-btn is-primary"
|
class="customCharacterSceneFormSubmit nes-btn is-primary"
|
||||||
on:click|preventDefault={finish}>Finish</button
|
on:click|preventDefault={finish}>{$LL.woka.customWoka.navigation.finish()}</button
|
||||||
>
|
>
|
||||||
{/if}
|
{/if}
|
||||||
{#if $activeRowStore !== 5}
|
{#if $activeRowStore !== 5}
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
class="customCharacterSceneFormSubmit nes-btn is-primary"
|
class="customCharacterSceneFormSubmit nes-btn is-primary"
|
||||||
on:click|preventDefault={selectDown}>Next <img src="resources/objects/arrow_down.png" alt="" /></button
|
on:click|preventDefault={selectDown}
|
||||||
|
>{$LL.woka.customWoka.navigation.next()}
|
||||||
|
<img src="resources/objects/arrow_down.png" alt="" /></button
|
||||||
>
|
>
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
import cinemaCloseImg from "../images/cinema-close.svg";
|
import cinemaCloseImg from "../images/cinema-close.svg";
|
||||||
import cinemaImg from "../images/cinema.svg";
|
import cinemaImg from "../images/cinema.svg";
|
||||||
import microphoneImg from "../images/microphone.svg";
|
import microphoneImg from "../images/microphone.svg";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let game: Game;
|
export let game: Game;
|
||||||
let selectedCamera: string | undefined = undefined;
|
let selectedCamera: string | undefined = undefined;
|
||||||
@ -76,7 +77,7 @@
|
|||||||
|
|
||||||
<form class="enableCameraScene" on:submit|preventDefault={submit}>
|
<form class="enableCameraScene" on:submit|preventDefault={submit}>
|
||||||
<section class="text-center">
|
<section class="text-center">
|
||||||
<h2>Turn on your camera and microphone</h2>
|
<h2>{$LL.camera.enable.title()}</h2>
|
||||||
</section>
|
</section>
|
||||||
{#if $localStreamStore.type === "success" && $localStreamStore.stream}
|
{#if $localStreamStore.type === "success" && $localStreamStore.stream}
|
||||||
<video class="myCamVideoSetup" use:srcObject={$localStreamStore.stream} autoplay muted playsinline />
|
<video class="myCamVideoSetup" use:srcObject={$localStreamStore.stream} autoplay muted playsinline />
|
||||||
@ -121,7 +122,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
<section class="action">
|
<section class="action">
|
||||||
<button type="submit" class="nes-btn is-primary letsgo">Let's go!</button>
|
<button type="submit" class="nes-btn is-primary letsgo">{$LL.camera.enable.start()}</button>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
@ -4,13 +4,14 @@ vim: ft=typescript
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { gameManager } from "../../Phaser/Game/GameManager";
|
import { gameManager } from "../../Phaser/Game/GameManager";
|
||||||
import followImg from "../images/follow.svg";
|
import followImg from "../images/follow.svg";
|
||||||
|
|
||||||
import { followStateStore, followRoleStore, followUsersStore } from "../../Stores/FollowStore";
|
import { followStateStore, followRoleStore, followUsersStore } from "../../Stores/FollowStore";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
const gameScene = gameManager.getCurrentGameScene();
|
const gameScene = gameManager.getCurrentGameScene();
|
||||||
|
|
||||||
function name(userId: number): string | undefined {
|
function name(userId: number): string {
|
||||||
return gameScene.MapPlayersByKey.get(userId)?.PlayerValue;
|
const user = gameScene.MapPlayersByKey.get(userId);
|
||||||
|
return user ? user.PlayerValue : "";
|
||||||
}
|
}
|
||||||
|
|
||||||
function sendFollowRequest() {
|
function sendFollowRequest() {
|
||||||
@ -42,11 +43,15 @@ vim: ft=typescript
|
|||||||
{#if $followStateStore === "requesting" && $followRoleStore === "follower"}
|
{#if $followStateStore === "requesting" && $followRoleStore === "follower"}
|
||||||
<div class="interact-menu nes-container is-rounded">
|
<div class="interact-menu nes-container is-rounded">
|
||||||
<section class="interact-menu-title">
|
<section class="interact-menu-title">
|
||||||
<h2>Do you want to follow {name($followUsersStore[0])}?</h2>
|
<h2>{$LL.follow.interactMenu.title.follow({ leader: name($followUsersStore[0]) })}</h2>
|
||||||
</section>
|
</section>
|
||||||
<section class="interact-menu-action">
|
<section class="interact-menu-action">
|
||||||
<button type="button" class="nes-btn is-success" on:click|preventDefault={acceptFollowRequest}>Yes</button>
|
<button type="button" class="nes-btn is-success" on:click|preventDefault={acceptFollowRequest}
|
||||||
<button type="button" class="nes-btn is-error" on:click|preventDefault={reset}>No</button>
|
>{$LL.follow.interactMenu.yes()}</button
|
||||||
|
>
|
||||||
|
<button type="button" class="nes-btn is-error" on:click|preventDefault={reset}
|
||||||
|
>{$LL.follow.interactMenu.no()}</button
|
||||||
|
>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
@ -54,20 +59,24 @@ vim: ft=typescript
|
|||||||
{#if $followStateStore === "ending"}
|
{#if $followStateStore === "ending"}
|
||||||
<div class="interact-menu nes-container is-rounded">
|
<div class="interact-menu nes-container is-rounded">
|
||||||
<section class="interact-menu-title">
|
<section class="interact-menu-title">
|
||||||
<h2>Interaction</h2>
|
<h2>{$LL.follow.interactMenu.title.interact()}</h2>
|
||||||
</section>
|
</section>
|
||||||
{#if $followRoleStore === "follower"}
|
{#if $followRoleStore === "follower"}
|
||||||
<section class="interact-menu-question">
|
<section class="interact-menu-question">
|
||||||
<p>Do you want to stop following {name($followUsersStore[0])}?</p>
|
<p>{$LL.follow.interactMenu.stop.follower({ leader: name($followUsersStore[0]) })}</p>
|
||||||
</section>
|
</section>
|
||||||
{:else if $followRoleStore === "leader"}
|
{:else if $followRoleStore === "leader"}
|
||||||
<section class="interact-menu-question">
|
<section class="interact-menu-question">
|
||||||
<p>Do you want to stop leading the way?</p>
|
<p>{$LL.follow.interactMenu.stop.leader()}</p>
|
||||||
</section>
|
</section>
|
||||||
{/if}
|
{/if}
|
||||||
<section class="interact-menu-action">
|
<section class="interact-menu-action">
|
||||||
<button type="button" class="nes-btn is-success" on:click|preventDefault={reset}>Yes</button>
|
<button type="button" class="nes-btn is-success" on:click|preventDefault={reset}
|
||||||
<button type="button" class="nes-btn is-error" on:click|preventDefault={abortEnding}>No</button>
|
>{$LL.follow.interactMenu.yes()}</button
|
||||||
|
>
|
||||||
|
<button type="button" class="nes-btn is-error" on:click|preventDefault={abortEnding}
|
||||||
|
>{$LL.follow.interactMenu.no()}</button
|
||||||
|
>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
@ -76,18 +85,24 @@ vim: ft=typescript
|
|||||||
<div class="interact-status nes-container is-rounded">
|
<div class="interact-status nes-container is-rounded">
|
||||||
<section class="interact-status">
|
<section class="interact-status">
|
||||||
{#if $followRoleStore === "follower"}
|
{#if $followRoleStore === "follower"}
|
||||||
<p>Following {name($followUsersStore[0])}</p>
|
<p>{$LL.follow.interactStatus.following({ leader: name($followUsersStore[0]) })}</p>
|
||||||
{:else if $followUsersStore.length === 0}
|
{:else if $followUsersStore.length === 0}
|
||||||
<p>Waiting for followers' confirmation</p>
|
<p>{$LL.follow.interactStatus.waitingFollowers()}</p>
|
||||||
{:else if $followUsersStore.length === 1}
|
{:else if $followUsersStore.length === 1}
|
||||||
<p>{name($followUsersStore[0])} is following you</p>
|
<p>{$LL.follow.interactStatus.followed.one({ follower: name($followUsersStore[0]) })}</p>
|
||||||
{:else if $followUsersStore.length === 2}
|
{:else if $followUsersStore.length === 2}
|
||||||
<p>{name($followUsersStore[0])} and {name($followUsersStore[1])} are following you</p>
|
<p>
|
||||||
|
{$LL.follow.interactStatus.followed.two({
|
||||||
|
firstFollower: name($followUsersStore[0]),
|
||||||
|
secondFollower: name($followUsersStore[1]),
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
{:else}
|
{:else}
|
||||||
<p>
|
<p>
|
||||||
{$followUsersStore.slice(0, -1).map(name).join(", ")} and {name(
|
{$LL.follow.interactStatus.followed.many({
|
||||||
$followUsersStore[$followUsersStore.length - 1]
|
followers: $followUsersStore.slice(0, -1).map(name).join(", "),
|
||||||
)} are following you
|
lastFollower: name($followUsersStore[$followUsersStore.length - 1]),
|
||||||
|
})}
|
||||||
</p>
|
</p>
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
import firefoxImg from "./images/help-setting-camera-permission-firefox.png";
|
import firefoxImg from "./images/help-setting-camera-permission-firefox.png";
|
||||||
import chromeImg from "./images/help-setting-camera-permission-chrome.png";
|
import chromeImg from "./images/help-setting-camera-permission-chrome.png";
|
||||||
import { getNavigatorType, isAndroid as isAndroidFct, NavigatorType } from "../../WebRtc/DeviceUtils";
|
import { getNavigatorType, isAndroid as isAndroidFct, NavigatorType } from "../../WebRtc/DeviceUtils";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let isAndroid = isAndroidFct();
|
let isAndroid = isAndroidFct();
|
||||||
let isFirefox = getNavigatorType() === NavigatorType.firefox;
|
let isFirefox = getNavigatorType() === NavigatorType.firefox;
|
||||||
@ -24,14 +25,13 @@
|
|||||||
transition:fly={{ y: -900, duration: 500 }}
|
transition:fly={{ y: -900, duration: 500 }}
|
||||||
>
|
>
|
||||||
<section>
|
<section>
|
||||||
<h2>Camera / Microphone access needed</h2>
|
<h2>{$LL.camera.help.title()}</h2>
|
||||||
<p class="err">Permission denied</p>
|
<p class="err">{$LL.camera.help.permissionDenied()}</p>
|
||||||
<p>You must allow camera and microphone access in your browser.</p>
|
<p>{$LL.camera.help.content()}</p>
|
||||||
<p>
|
<p>
|
||||||
{#if isFirefox}
|
{#if isFirefox}
|
||||||
<p class="err">
|
<p class="err">
|
||||||
Please click the "Remember this decision" checkbox, if you don't want Firefox to keep asking you the
|
{$LL.camera.help.firefoxContent()}
|
||||||
authorization.
|
|
||||||
</p>
|
</p>
|
||||||
<img src={firefoxImg} alt="" />
|
<img src={firefoxImg} alt="" />
|
||||||
{:else if isChrome && !isAndroid}
|
{:else if isChrome && !isAndroid}
|
||||||
@ -40,9 +40,11 @@
|
|||||||
</p>
|
</p>
|
||||||
</section>
|
</section>
|
||||||
<section>
|
<section>
|
||||||
<button class="helpCameraSettingsFormRefresh nes-btn" on:click|preventDefault={refresh}>Refresh</button>
|
<button class="helpCameraSettingsFormRefresh nes-btn" on:click|preventDefault={refresh}
|
||||||
|
>{$LL.camera.help.refresh()}</button
|
||||||
|
>
|
||||||
<button type="submit" class="helpCameraSettingsFormContinue nes-btn is-primary" on:click|preventDefault={close}
|
<button type="submit" class="helpCameraSettingsFormContinue nes-btn is-primary" on:click|preventDefault={close}
|
||||||
>Continue without webcam</button
|
>{$LL.camera.help.continue()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
import { DISPLAY_TERMS_OF_USE, MAX_USERNAME_LENGTH } from "../../Enum/EnvironmentVariable";
|
import { DISPLAY_TERMS_OF_USE, MAX_USERNAME_LENGTH } from "../../Enum/EnvironmentVariable";
|
||||||
import logoImg from "../images/logo.png";
|
import logoImg from "../images/logo.png";
|
||||||
import { gameManager } from "../../Phaser/Game/GameManager";
|
import { gameManager } from "../../Phaser/Game/GameManager";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let game: Game;
|
export let game: Game;
|
||||||
|
|
||||||
@ -27,7 +28,7 @@
|
|||||||
<img src={logoImg} alt="WorkAdventure logo" />
|
<img src={logoImg} alt="WorkAdventure logo" />
|
||||||
</section>
|
</section>
|
||||||
<section class="text-center">
|
<section class="text-center">
|
||||||
<h2>Enter your name</h2>
|
<h2>{$LL.login.input.name.placeholder()}</h2>
|
||||||
</section>
|
</section>
|
||||||
<!-- svelte-ignore a11y-autofocus -->
|
<!-- svelte-ignore a11y-autofocus -->
|
||||||
<input
|
<input
|
||||||
@ -44,22 +45,20 @@
|
|||||||
/>
|
/>
|
||||||
<section class="error-section">
|
<section class="error-section">
|
||||||
{#if name.trim() === "" && startValidating}
|
{#if name.trim() === "" && startValidating}
|
||||||
<p class="err">The name is empty</p>
|
<p class="err">{$LL.login.input.name.empty()}</p>
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
{#if DISPLAY_TERMS_OF_USE}
|
{#if DISPLAY_TERMS_OF_USE}
|
||||||
<section class="terms-and-conditions">
|
<section class="terms-and-conditions">
|
||||||
|
<a style="display: none;" href="traduction">Need for traduction</a>
|
||||||
<p>
|
<p>
|
||||||
By continuing, you are agreeing our <a href="https://workadventu.re/terms-of-use" target="_blank"
|
{$LL.login.terms()}
|
||||||
>terms of use</a
|
|
||||||
>, <a href="https://workadventu.re/privacy-policy" target="_blank">privacy policy</a> and
|
|
||||||
<a href="https://workadventu.re/cookie-policy" target="_blank">cookie policy</a>.
|
|
||||||
</p>
|
</p>
|
||||||
</section>
|
</section>
|
||||||
{/if}
|
{/if}
|
||||||
<section class="action">
|
<section class="action">
|
||||||
<button type="submit" class="nes-btn is-primary loginSceneFormSubmit">Continue</button>
|
<button type="submit" class="nes-btn is-primary loginSceneFormSubmit">{$LL.login.continue()}</button>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { gameManager } from "../../Phaser/Game/GameManager";
|
import { gameManager } from "../../Phaser/Game/GameManager";
|
||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let gameScene = gameManager.getCurrentGameScene();
|
let gameScene = gameManager.getCurrentGameScene();
|
||||||
|
|
||||||
@ -11,7 +12,7 @@
|
|||||||
let mapName: string = "";
|
let mapName: string = "";
|
||||||
let mapLink: string = "";
|
let mapLink: string = "";
|
||||||
let mapDescription: string = "";
|
let mapDescription: string = "";
|
||||||
let mapCopyright: string = "The map creator did not declare a copyright for the map.";
|
let mapCopyright: string = $LL.menu.about.copyrights.map.empty();
|
||||||
let tilesetCopyright: string[] = [];
|
let tilesetCopyright: string[] = [];
|
||||||
let audioCopyright: string[] = [];
|
let audioCopyright: string[] = [];
|
||||||
|
|
||||||
@ -62,41 +63,37 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="about-room-main">
|
<div class="about-room-main">
|
||||||
<h2>Information on the map</h2>
|
<h2>{$LL.menu.about.mapInfo()}</h2>
|
||||||
<section class="container-overflow">
|
<section class="container-overflow">
|
||||||
<h3>{mapName}</h3>
|
<h3>{mapName}</h3>
|
||||||
<p class="string-HTML">{mapDescription}</p>
|
<p class="string-HTML">{mapDescription}</p>
|
||||||
{#if mapLink}
|
{#if mapLink}
|
||||||
<p class="string-HTML">> <a href={mapLink} target="_blank">link to this map</a> <</p>
|
<p class="string-HTML">
|
||||||
|
> <a href={mapLink} target="_blank">{$LL.menu.about.mapLink()}</a> <
|
||||||
|
</p>
|
||||||
{/if}
|
{/if}
|
||||||
<h3 class="nes-pointer hoverable" on:click={() => (expandedMapCopyright = !expandedMapCopyright)}>
|
<h3 class="nes-pointer hoverable" on:click={() => (expandedMapCopyright = !expandedMapCopyright)}>
|
||||||
Copyrights of the map
|
{$LL.menu.about.copyrights.map.title()}
|
||||||
</h3>
|
</h3>
|
||||||
<p class="string-HTML" hidden={!expandedMapCopyright}>{mapCopyright}</p>
|
<p class="string-HTML" hidden={!expandedMapCopyright}>{mapCopyright}</p>
|
||||||
<h3 class="nes-pointer hoverable" on:click={() => (expandedTilesetCopyright = !expandedTilesetCopyright)}>
|
<h3 class="nes-pointer hoverable" on:click={() => (expandedTilesetCopyright = !expandedTilesetCopyright)}>
|
||||||
Copyrights of the tilesets
|
{$LL.menu.about.copyrights.tileset.title()}
|
||||||
</h3>
|
</h3>
|
||||||
<section hidden={!expandedTilesetCopyright}>
|
<section hidden={!expandedTilesetCopyright}>
|
||||||
{#each tilesetCopyright as copyright}
|
{#each tilesetCopyright as copyright}
|
||||||
<p class="string-HTML">{copyright}</p>
|
<p class="string-HTML">{copyright}</p>
|
||||||
{:else}
|
{:else}
|
||||||
<p>
|
<p>{$LL.menu.about.copyrights.tileset.empty()}</p>
|
||||||
The map creator did not declare a copyright for the tilesets. This doesn't mean that those tilesets
|
|
||||||
have no license.
|
|
||||||
</p>
|
|
||||||
{/each}
|
{/each}
|
||||||
</section>
|
</section>
|
||||||
<h3 class="nes-pointer hoverable" on:click={() => (expandedAudioCopyright = !expandedAudioCopyright)}>
|
<h3 class="nes-pointer hoverable" on:click={() => (expandedAudioCopyright = !expandedAudioCopyright)}>
|
||||||
Copyrights of audio files
|
{$LL.menu.about.copyrights.audio.title()}
|
||||||
</h3>
|
</h3>
|
||||||
<section hidden={!expandedAudioCopyright}>
|
<section hidden={!expandedAudioCopyright}>
|
||||||
{#each audioCopyright as copyright}
|
{#each audioCopyright as copyright}
|
||||||
<p class="string-HTML">{copyright}</p>
|
<p class="string-HTML">{copyright}</p>
|
||||||
{:else}
|
{:else}
|
||||||
<p>
|
<p>{$LL.menu.about.copyrights.audio.empty()}</p>
|
||||||
The map creator did not declare a copyright for audio files. This doesn't mean that those tilesets
|
|
||||||
have no license.
|
|
||||||
</p>
|
|
||||||
{/each}
|
{/each}
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
import { AdminMessageEventTypes } from "../../Connexion/AdminMessagesService";
|
import { AdminMessageEventTypes } from "../../Connexion/AdminMessagesService";
|
||||||
import uploadFile from "../images/music-file.svg";
|
import uploadFile from "../images/music-file.svg";
|
||||||
import type { PlayGlobalMessageInterface } from "../../Connexion/ConnexionModels";
|
import type { PlayGlobalMessageInterface } from "../../Connexion/ConnexionModels";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
interface EventTargetFiles extends EventTarget {
|
interface EventTargetFiles extends EventTarget {
|
||||||
files: Array<File>;
|
files: Array<File>;
|
||||||
@ -76,7 +77,7 @@
|
|||||||
<img
|
<img
|
||||||
class="nes-pointer"
|
class="nes-pointer"
|
||||||
src={uploadFile}
|
src={uploadFile}
|
||||||
alt="Upload a file"
|
alt={$LL.menu.globalAudio.uploadInfo()}
|
||||||
on:click|preventDefault={() => {
|
on:click|preventDefault={() => {
|
||||||
fileInput.click();
|
fileInput.click();
|
||||||
}}
|
}}
|
||||||
@ -85,7 +86,7 @@
|
|||||||
<p>{fileName} : {fileSize}</p>
|
<p>{fileName} : {fileSize}</p>
|
||||||
{/if}
|
{/if}
|
||||||
{#if errorFile}
|
{#if errorFile}
|
||||||
<p class="err">No file selected. You need to upload a file before sending it.</p>
|
<p class="err">{$LL.menu.globalAudio.error()}</p>
|
||||||
{/if}
|
{/if}
|
||||||
<input
|
<input
|
||||||
type="file"
|
type="file"
|
||||||
|
@ -1,4 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
import { contactPageStore } from "../../Stores/MenuStore";
|
||||||
|
|
||||||
function goToGettingStarted() {
|
function goToGettingStarted() {
|
||||||
const sparkHost = "https://workadventu.re/getting-started";
|
const sparkHost = "https://workadventu.re/getting-started";
|
||||||
window.open(sparkHost, "_blank");
|
window.open(sparkHost, "_blank");
|
||||||
@ -8,25 +11,24 @@
|
|||||||
const sparkHost = "https://workadventu.re/map-building/";
|
const sparkHost = "https://workadventu.re/map-building/";
|
||||||
window.open(sparkHost, "_blank");
|
window.open(sparkHost, "_blank");
|
||||||
}
|
}
|
||||||
|
|
||||||
import { contactPageStore } from "../../Stores/MenuStore";
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="create-map-main">
|
<div class="create-map-main">
|
||||||
<section class="container-overflow">
|
<section class="container-overflow">
|
||||||
<section>
|
<section>
|
||||||
<h3>Getting started</h3>
|
<h3>{$LL.menu.contact.gettingStarted.title()}</h3>
|
||||||
<p>
|
<p>{$LL.menu.contact.gettingStarted.description()}</p>
|
||||||
WorkAdventure allows you to create an online space to communicate spontaneously with others. And it all
|
<button type="button" class="nes-btn is-primary" on:click={goToGettingStarted}
|
||||||
starts with creating your own space. Choose from a large selection of prefabricated maps by our team.
|
>{$LL.menu.contact.gettingStarted.title()}</button
|
||||||
</p>
|
>
|
||||||
<button type="button" class="nes-btn is-primary" on:click={goToGettingStarted}>Getting started</button>
|
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<h3>Create your map</h3>
|
<h3>{$LL.menu.contact.createMap.title()}</h3>
|
||||||
<p>You can also create your own custom map by following the step of the documentation.</p>
|
<p>{$LL.menu.contact.createMap.description()}</p>
|
||||||
<button type="button" class="nes-btn" on:click={goToBuildingMap}>Create your map</button>
|
<button type="button" class="nes-btn" on:click={goToBuildingMap}
|
||||||
|
>{$LL.menu.contact.createMap.title()}</button
|
||||||
|
>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<iframe
|
<iframe
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import TextGlobalMessage from "./TextGlobalMessage.svelte";
|
import TextGlobalMessage from "./TextGlobalMessage.svelte";
|
||||||
import AudioGlobalMessage from "./AudioGlobalMessage.svelte";
|
import AudioGlobalMessage from "./AudioGlobalMessage.svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let handleSendText: { sendTextMessage(broadcast: boolean): void };
|
let handleSendText: { sendTextMessage(broadcast: boolean): void };
|
||||||
let handleSendAudio: { sendAudioMessage(broadcast: boolean): Promise<void> };
|
let handleSendAudio: { sendAudioMessage(broadcast: boolean): Promise<void> };
|
||||||
@ -35,14 +36,14 @@
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nes-btn {inputSendTextActive ? 'is-disabled' : ''}"
|
class="nes-btn {inputSendTextActive ? 'is-disabled' : ''}"
|
||||||
on:click|preventDefault={activateInputText}>Text</button
|
on:click|preventDefault={activateInputText}>{$LL.menu.globalMessage.text()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
<section>
|
<section>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nes-btn {uploadAudioActive ? 'is-disabled' : ''}"
|
class="nes-btn {uploadAudioActive ? 'is-disabled' : ''}"
|
||||||
on:click|preventDefault={activateUploadAudio}>Audio</button
|
on:click|preventDefault={activateUploadAudio}>{$LL.menu.globalMessage.audio()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
@ -57,10 +58,10 @@
|
|||||||
<div class="global-message-footer">
|
<div class="global-message-footer">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" class="nes-checkbox is-dark nes-pointer" bind:checked={broadcastToWorld} />
|
<input type="checkbox" class="nes-checkbox is-dark nes-pointer" bind:checked={broadcastToWorld} />
|
||||||
<span>Broadcast to all rooms of the world</span>
|
<span>{$LL.menu.globalMessage.warning()}</span>
|
||||||
</label>
|
</label>
|
||||||
<section>
|
<section>
|
||||||
<button class="nes-btn is-primary" on:click|preventDefault={send}>Send</button>
|
<button class="nes-btn is-primary" on:click|preventDefault={send}>{$LL.menu.globalMessage.send()}</button>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
function copyLink() {
|
function copyLink() {
|
||||||
const input: HTMLInputElement = document.getElementById("input-share-link") as HTMLInputElement;
|
const input: HTMLInputElement = document.getElementById("input-share-link") as HTMLInputElement;
|
||||||
input.focus();
|
input.focus();
|
||||||
@ -21,14 +23,14 @@
|
|||||||
<div class="guest-main">
|
<div class="guest-main">
|
||||||
<section class="container-overflow">
|
<section class="container-overflow">
|
||||||
<section class="share-url not-mobile">
|
<section class="share-url not-mobile">
|
||||||
<h3>Share the link of the room!</h3>
|
<h3>{$LL.menu.invite.description()}</h3>
|
||||||
<input type="text" readonly id="input-share-link" value={location.toString()} />
|
<input type="text" readonly id="input-share-link" value={location.toString()} />
|
||||||
<button type="button" class="nes-btn is-primary" on:click={copyLink}>Copy</button>
|
<button type="button" class="nes-btn is-primary" on:click={copyLink}>{$LL.menu.invite.copy()}</button>
|
||||||
</section>
|
</section>
|
||||||
<section class="is-mobile">
|
<section class="is-mobile">
|
||||||
<h3>Share the link of the room!</h3>
|
<h3>{$LL.menu.invite.description()}</h3>
|
||||||
<input type="hidden" readonly id="input-share-link" value={location.toString()} />
|
<input type="hidden" readonly id="input-share-link" value={location.toString()} />
|
||||||
<button type="button" class="nes-btn is-primary" on:click={shareLink}>Share</button>
|
<button type="button" class="nes-btn is-primary" on:click={shareLink}>{$LL.menu.invite.share()}</button>
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
|
@ -14,26 +14,27 @@
|
|||||||
SubMenusInterface,
|
SubMenusInterface,
|
||||||
subMenusStore,
|
subMenusStore,
|
||||||
} from "../../Stores/MenuStore";
|
} from "../../Stores/MenuStore";
|
||||||
|
import type { MenuItem } from "../../Stores/MenuStore";
|
||||||
import { onDestroy, onMount } from "svelte";
|
import { onDestroy, onMount } from "svelte";
|
||||||
import { get } from "svelte/store";
|
|
||||||
import type { Unsubscriber } from "svelte/store";
|
import type { Unsubscriber } from "svelte/store";
|
||||||
import { sendMenuClickedEvent } from "../../Api/iframe/Ui/MenuItem";
|
import { sendMenuClickedEvent } from "../../Api/iframe/Ui/MenuItem";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let activeSubMenu: string = SubMenusInterface.profile;
|
let activeSubMenu: MenuItem = $subMenusStore[0];
|
||||||
let activeComponent: typeof ProfileSubMenu | typeof CustomSubMenu = ProfileSubMenu;
|
let activeComponent: typeof ProfileSubMenu | typeof CustomSubMenu = ProfileSubMenu;
|
||||||
let props: { url: string; allowApi: boolean };
|
let props: { url: string; allowApi: boolean };
|
||||||
let unsubscriberSubMenuStore: Unsubscriber;
|
let unsubscriberSubMenuStore: Unsubscriber;
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
unsubscriberSubMenuStore = subMenusStore.subscribe(() => {
|
unsubscriberSubMenuStore = subMenusStore.subscribe(() => {
|
||||||
if (!get(subMenusStore).includes(activeSubMenu)) {
|
if (!$subMenusStore.includes(activeSubMenu)) {
|
||||||
switchMenu(SubMenusInterface.profile);
|
switchMenu($subMenusStore[0]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
checkSubMenuToShow();
|
checkSubMenuToShow();
|
||||||
|
|
||||||
switchMenu(SubMenusInterface.profile);
|
switchMenu($subMenusStore[0]);
|
||||||
});
|
});
|
||||||
|
|
||||||
onDestroy(() => {
|
onDestroy(() => {
|
||||||
@ -42,10 +43,10 @@
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function switchMenu(menu: string) {
|
function switchMenu(menu: MenuItem) {
|
||||||
if (get(subMenusStore).find((subMenu) => subMenu === menu)) {
|
if (menu.type === "translated") {
|
||||||
activeSubMenu = menu;
|
activeSubMenu = menu;
|
||||||
switch (menu) {
|
switch (menu.key) {
|
||||||
case SubMenusInterface.settings:
|
case SubMenusInterface.settings:
|
||||||
activeComponent = SettingsSubMenu;
|
activeComponent = SettingsSubMenu;
|
||||||
break;
|
break;
|
||||||
@ -64,36 +65,46 @@
|
|||||||
case SubMenusInterface.contact:
|
case SubMenusInterface.contact:
|
||||||
activeComponent = ContactSubMenu;
|
activeComponent = ContactSubMenu;
|
||||||
break;
|
break;
|
||||||
default: {
|
|
||||||
const customMenu = customMenuIframe.get(menu);
|
|
||||||
if (customMenu !== undefined) {
|
|
||||||
props = { url: customMenu.url, allowApi: customMenu.allowApi };
|
|
||||||
activeComponent = CustomSubMenu;
|
|
||||||
} else {
|
|
||||||
sendMenuClickedEvent(menu);
|
|
||||||
menuVisiblilityStore.set(false);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else throw new Error("There is no menu called " + menu);
|
} else {
|
||||||
|
const customMenu = customMenuIframe.get(menu.label);
|
||||||
|
if (customMenu !== undefined) {
|
||||||
|
props = { url: customMenu.url, allowApi: customMenu.allowApi };
|
||||||
|
activeComponent = CustomSubMenu;
|
||||||
|
} else {
|
||||||
|
sendMenuClickedEvent(menu.label);
|
||||||
|
menuVisiblilityStore.set(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function closeMenu() {
|
function closeMenu() {
|
||||||
menuVisiblilityStore.set(false);
|
menuVisiblilityStore.set(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
function onKeyDown(e: KeyboardEvent) {
|
function onKeyDown(e: KeyboardEvent) {
|
||||||
if (e.key === "Escape") {
|
if (e.key === "Escape") {
|
||||||
closeMenu();
|
closeMenu();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function translateMenuName(menu: MenuItem) {
|
||||||
|
if (menu.type === "scripting") {
|
||||||
|
return menu.label;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bypass the proxy of typesafe for getting the menu name : https://github.com/ivanhofer/typesafe-i18n/issues/156
|
||||||
|
const getMenuName = $LL.menu.sub[menu.key];
|
||||||
|
|
||||||
|
return getMenuName();
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<svelte:window on:keydown={onKeyDown} />
|
<svelte:window on:keydown={onKeyDown} />
|
||||||
|
|
||||||
<div class="menu-container-main">
|
<div class="menu-container-main">
|
||||||
<div class="menu-nav-sidebar nes-container is-rounded" transition:fly={{ x: -1000, duration: 500 }}>
|
<div class="menu-nav-sidebar nes-container is-rounded" transition:fly={{ x: -1000, duration: 500 }}>
|
||||||
<h2>Menu</h2>
|
<h2>{$LL.menu.title()}</h2>
|
||||||
<nav>
|
<nav>
|
||||||
{#each $subMenusStore as submenu}
|
{#each $subMenusStore as submenu}
|
||||||
<button
|
<button
|
||||||
@ -101,14 +112,14 @@
|
|||||||
class="nes-btn {activeSubMenu === submenu ? 'is-disabled' : ''}"
|
class="nes-btn {activeSubMenu === submenu ? 'is-disabled' : ''}"
|
||||||
on:click|preventDefault={() => switchMenu(submenu)}
|
on:click|preventDefault={() => switchMenu(submenu)}
|
||||||
>
|
>
|
||||||
{submenu}
|
{translateMenuName(submenu)}
|
||||||
</button>
|
</button>
|
||||||
{/each}
|
{/each}
|
||||||
</nav>
|
</nav>
|
||||||
</div>
|
</div>
|
||||||
<div class="menu-submenu-container nes-container is-rounded" transition:fly={{ y: -1000, duration: 500 }}>
|
<div class="menu-submenu-container nes-container is-rounded" transition:fly={{ y: -1000, duration: 500 }}>
|
||||||
<button type="button" class="nes-btn is-error close" on:click={closeMenu}>×</button>
|
<button type="button" class="nes-btn is-error close" on:click={closeMenu}>×</button>
|
||||||
<h2>{activeSubMenu}</h2>
|
<h2>{translateMenuName(activeSubMenu)}</h2>
|
||||||
<svelte:component this={activeComponent} {...props} />
|
<svelte:component this={activeComponent} {...props} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
import { get } from "svelte/store";
|
import { get } from "svelte/store";
|
||||||
import { ADMIN_URL } from "../../Enum/EnvironmentVariable";
|
import { ADMIN_URL } from "../../Enum/EnvironmentVariable";
|
||||||
import { showShareLinkMapModalStore } from "../../Stores/ModalStore";
|
import { showShareLinkMapModalStore } from "../../Stores/ModalStore";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
function showMenu() {
|
function showMenu() {
|
||||||
menuVisiblilityStore.set(!get(menuVisiblilityStore));
|
menuVisiblilityStore.set(!get(menuVisiblilityStore));
|
||||||
@ -29,11 +30,21 @@
|
|||||||
|
|
||||||
<main class="menuIcon">
|
<main class="menuIcon">
|
||||||
{#if $limitMapStore}
|
{#if $limitMapStore}
|
||||||
<img src={logoInvite} alt="open menu" class="nes-pointer" on:click|preventDefault={showInvite} />
|
<img
|
||||||
<img src={logoRegister} alt="open menu" class="nes-pointer" on:click|preventDefault={register} />
|
src={logoInvite}
|
||||||
|
alt={$LL.menu.icon.open.invite()}
|
||||||
|
class="nes-pointer"
|
||||||
|
on:click|preventDefault={showInvite}
|
||||||
|
/>
|
||||||
|
<img
|
||||||
|
src={logoRegister}
|
||||||
|
alt={$LL.menu.icon.open.register()}
|
||||||
|
class="nes-pointer"
|
||||||
|
on:click|preventDefault={register}
|
||||||
|
/>
|
||||||
{:else}
|
{:else}
|
||||||
<img src={logoWA} alt="open menu" class="nes-pointer" on:click|preventDefault={showMenu} />
|
<img src={logoWA} alt={$LL.menu.icon.open.menu()} class="nes-pointer" on:click|preventDefault={showMenu} />
|
||||||
<img src={logoTalk} alt="open menu" class="nes-pointer" on:click|preventDefault={showChat} />
|
<img src={logoTalk} alt={$LL.menu.icon.open.chat()} class="nes-pointer" on:click|preventDefault={showChat} />
|
||||||
{/if}
|
{/if}
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
import btnProfileSubMenuCompanion from "../images/btn-menu-profile-companion.svg";
|
import btnProfileSubMenuCompanion from "../images/btn-menu-profile-companion.svg";
|
||||||
import Woka from "../Woka/Woka.svelte";
|
import Woka from "../Woka/Woka.svelte";
|
||||||
import Companion from "../Companion/Companion.svelte";
|
import Companion from "../Companion/Companion.svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
function disableMenuStores() {
|
function disableMenuStores() {
|
||||||
menuVisiblilityStore.set(false);
|
menuVisiblilityStore.set(false);
|
||||||
@ -62,20 +63,20 @@
|
|||||||
<div class="submenu">
|
<div class="submenu">
|
||||||
<section>
|
<section>
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={openEditNameScene}>
|
<button type="button" class="nes-btn" on:click|preventDefault={openEditNameScene}>
|
||||||
<img src={btnProfileSubMenuIdentity} alt="Edit your name" />
|
<img src={btnProfileSubMenuIdentity} alt={$LL.menu.profile.edit.name()} />
|
||||||
<span class="btn-hover">Edit your name</span>
|
<span class="btn-hover">{$LL.menu.profile.edit.name()}</span>
|
||||||
</button>
|
</button>
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={openEditSkinScene}>
|
<button type="button" class="nes-btn" on:click|preventDefault={openEditSkinScene}>
|
||||||
<Woka userId={-1} placeholderSrc="" width="26px" height="26px" />
|
<Woka userId={-1} placeholderSrc="" width="26px" height="26px" />
|
||||||
<span class="btn-hover">Edit your WOKA</span>
|
<span class="btn-hover">{$LL.menu.profile.edit.woka()}</span>
|
||||||
</button>
|
</button>
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={openEditCompanionScene}>
|
<button type="button" class="nes-btn" on:click|preventDefault={openEditCompanionScene}>
|
||||||
<Companion userId={-1} placeholderSrc={btnProfileSubMenuCompanion} width="26px" height="26px" />
|
<Companion userId={-1} placeholderSrc={btnProfileSubMenuCompanion} width="26px" height="26px" />
|
||||||
<span class="btn-hover">Edit your companion</span>
|
<span class="btn-hover">{$LL.menu.profile.edit.companion()}</span>
|
||||||
</button>
|
</button>
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={openEnableCameraScene}>
|
<button type="button" class="nes-btn" on:click|preventDefault={openEnableCameraScene}>
|
||||||
<img src={btnProfileSubMenuCamera} alt="Edit your camera" />
|
<img src={btnProfileSubMenuCamera} alt={$LL.menu.profile.edit.camera()} />
|
||||||
<span class="btn-hover">Edit your camera</span>
|
<span class="btn-hover">{$LL.menu.profile.edit.camera()}</span>
|
||||||
</button>
|
</button>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
@ -88,11 +89,13 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
<section>
|
<section>
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={logOut}>Log out</button>
|
<button type="button" class="nes-btn" on:click|preventDefault={logOut}
|
||||||
|
>{$LL.menu.profile.logout()}</button
|
||||||
|
>
|
||||||
</section>
|
</section>
|
||||||
{:else}
|
{:else}
|
||||||
<section>
|
<section>
|
||||||
<a type="button" class="nes-btn" href="/login">Sign in</a>
|
<a type="button" class="nes-btn" href="/login">{$LL.menu.profile.login()}</a>
|
||||||
</section>
|
</section>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
@ -4,6 +4,9 @@
|
|||||||
import { HtmlUtils } from "../../WebRtc/HtmlUtils";
|
import { HtmlUtils } from "../../WebRtc/HtmlUtils";
|
||||||
import { isMobile } from "../../Enum/EnvironmentVariable";
|
import { isMobile } from "../../Enum/EnvironmentVariable";
|
||||||
import { menuVisiblilityStore } from "../../Stores/MenuStore";
|
import { menuVisiblilityStore } from "../../Stores/MenuStore";
|
||||||
|
import LL, { locale } from "../../i18n/i18n-svelte";
|
||||||
|
import type { Locales } from "../../i18n/i18n-types";
|
||||||
|
import { displayableLocales, setCurrentLocale } from "../../i18n/locales";
|
||||||
|
|
||||||
let fullscreen: boolean = localUserStore.getFullscreen();
|
let fullscreen: boolean = localUserStore.getFullscreen();
|
||||||
let notification: boolean = localUserStore.getNotification() === "granted";
|
let notification: boolean = localUserStore.getNotification() === "granted";
|
||||||
@ -11,14 +14,17 @@
|
|||||||
let ignoreFollowRequests: boolean = localUserStore.getIgnoreFollowRequests();
|
let ignoreFollowRequests: boolean = localUserStore.getIgnoreFollowRequests();
|
||||||
let valueGame: number = localUserStore.getGameQualityValue();
|
let valueGame: number = localUserStore.getGameQualityValue();
|
||||||
let valueVideo: number = localUserStore.getVideoQualityValue();
|
let valueVideo: number = localUserStore.getVideoQualityValue();
|
||||||
|
let valueLocale: string = $locale;
|
||||||
let previewValueGame = valueGame;
|
let previewValueGame = valueGame;
|
||||||
let previewValueVideo = valueVideo;
|
let previewValueVideo = valueVideo;
|
||||||
|
let previewValueLocale = valueLocale;
|
||||||
|
|
||||||
function saveSetting() {
|
function saveSetting() {
|
||||||
if (valueGame !== previewValueGame) {
|
let change = false;
|
||||||
previewValueGame = valueGame;
|
|
||||||
localUserStore.setGameQualityValue(valueGame);
|
if (valueLocale !== previewValueLocale) {
|
||||||
window.location.reload();
|
previewValueLocale = valueLocale;
|
||||||
|
setCurrentLocale(valueLocale as Locales);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (valueVideo !== previewValueVideo) {
|
if (valueVideo !== previewValueVideo) {
|
||||||
@ -26,6 +32,16 @@
|
|||||||
videoConstraintStore.setFrameRate(valueVideo);
|
videoConstraintStore.setFrameRate(valueVideo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (valueGame !== previewValueGame) {
|
||||||
|
previewValueGame = valueGame;
|
||||||
|
localUserStore.setGameQualityValue(valueGame);
|
||||||
|
change = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change) {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
|
||||||
closeMenu();
|
closeMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -73,34 +89,74 @@
|
|||||||
|
|
||||||
<div class="settings-main" on:submit|preventDefault={saveSetting}>
|
<div class="settings-main" on:submit|preventDefault={saveSetting}>
|
||||||
<section>
|
<section>
|
||||||
<h3>Game quality</h3>
|
<h3>{$LL.menu.settings.gameQuality.title()}</h3>
|
||||||
<div class="nes-select is-dark">
|
<div class="nes-select is-dark">
|
||||||
<select bind:value={valueGame}>
|
<select bind:value={valueGame}>
|
||||||
<option value={120}>{isMobile() ? "High (120 fps)" : "High video quality (120 fps)"}</option>
|
<option value={120}
|
||||||
<option value={60}
|
>{isMobile()
|
||||||
>{isMobile() ? "Medium (60 fps)" : "Medium video quality (60 fps, recommended)"}</option
|
? $LL.menu.settings.gameQuality.short.high()
|
||||||
|
: $LL.menu.settings.gameQuality.long.high()}</option
|
||||||
|
>
|
||||||
|
<option value={60}
|
||||||
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.gameQuality.short.medium()
|
||||||
|
: $LL.menu.settings.gameQuality.long.medium()}</option
|
||||||
|
>
|
||||||
|
<option value={40}
|
||||||
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.gameQuality.short.minimum()
|
||||||
|
: $LL.menu.settings.gameQuality.long.minimum()}</option
|
||||||
|
>
|
||||||
|
<option value={20}
|
||||||
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.gameQuality.short.small()
|
||||||
|
: $LL.menu.settings.gameQuality.long.small()}</option
|
||||||
>
|
>
|
||||||
<option value={40}>{isMobile() ? "Minimum (40 fps)" : "Minimum video quality (40 fps)"}</option>
|
|
||||||
<option value={20}>{isMobile() ? "Small (20 fps)" : "Small video quality (20 fps)"}</option>
|
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
<section>
|
<section>
|
||||||
<h3>Video quality</h3>
|
<h3>{$LL.menu.settings.videoQuality.title()}</h3>
|
||||||
<div class="nes-select is-dark">
|
<div class="nes-select is-dark">
|
||||||
<select bind:value={valueVideo}>
|
<select bind:value={valueVideo}>
|
||||||
<option value={30}>{isMobile() ? "High (30 fps)" : "High video quality (30 fps)"}</option>
|
<option value={30}
|
||||||
<option value={20}
|
>{isMobile()
|
||||||
>{isMobile() ? "Medium (20 fps)" : "Medium video quality (20 fps, recommended)"}</option
|
? $LL.menu.settings.videoQuality.short.high()
|
||||||
|
: $LL.menu.settings.videoQuality.long.high()}</option
|
||||||
>
|
>
|
||||||
<option value={10}>{isMobile() ? "Minimum (10 fps)" : "Minimum video quality (10 fps)"}</option>
|
<option value={20}
|
||||||
<option value={5}>{isMobile() ? "Small (5 fps)" : "Small video quality (5 fps)"}</option>
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.videoQuality.short.medium()
|
||||||
|
: $LL.menu.settings.videoQuality.long.medium()}</option
|
||||||
|
>
|
||||||
|
<option value={10}
|
||||||
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.videoQuality.short.minimum()
|
||||||
|
: $LL.menu.settings.videoQuality.long.minimum()}</option
|
||||||
|
>
|
||||||
|
<option value={5}
|
||||||
|
>{isMobile()
|
||||||
|
? $LL.menu.settings.videoQuality.short.small()
|
||||||
|
: $LL.menu.settings.videoQuality.long.small()}</option
|
||||||
|
>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<h3>{$LL.menu.settings.language.title()}</h3>
|
||||||
|
<div class="nes-select is-dark">
|
||||||
|
<select class="languages-switcher" bind:value={valueLocale}>
|
||||||
|
{#each displayableLocales as locale (locale.id)}
|
||||||
|
<option value={locale.id}>{`${locale.language} (${locale.country})`}</option>
|
||||||
|
{/each}
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
<section class="settings-section-save">
|
<section class="settings-section-save">
|
||||||
<p>(Saving these settings will restart the game)</p>
|
<p>{$LL.menu.settings.save.warning()}</p>
|
||||||
<button type="button" class="nes-btn is-primary" on:click|preventDefault={saveSetting}>Save</button>
|
<button type="button" class="nes-btn is-primary" on:click|preventDefault={saveSetting}
|
||||||
|
>{$LL.menu.settings.save.button()}</button
|
||||||
|
>
|
||||||
</section>
|
</section>
|
||||||
<section class="settings-section-noSaveOption">
|
<section class="settings-section-noSaveOption">
|
||||||
<label>
|
<label>
|
||||||
@ -110,7 +166,7 @@
|
|||||||
bind:checked={fullscreen}
|
bind:checked={fullscreen}
|
||||||
on:change={changeFullscreen}
|
on:change={changeFullscreen}
|
||||||
/>
|
/>
|
||||||
<span>Fullscreen</span>
|
<span>{$LL.menu.settings.fullscreen()}</span>
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input
|
<input
|
||||||
@ -119,7 +175,7 @@
|
|||||||
bind:checked={notification}
|
bind:checked={notification}
|
||||||
on:change={changeNotification}
|
on:change={changeNotification}
|
||||||
/>
|
/>
|
||||||
<span>Notifications</span>
|
<span>{$LL.menu.settings.notifications()}</span>
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input
|
<input
|
||||||
@ -128,7 +184,7 @@
|
|||||||
bind:checked={forceCowebsiteTrigger}
|
bind:checked={forceCowebsiteTrigger}
|
||||||
on:change={changeForceCowebsiteTrigger}
|
on:change={changeForceCowebsiteTrigger}
|
||||||
/>
|
/>
|
||||||
<span>Always ask before opening websites and Jitsi Meet rooms</span>
|
<span>{$LL.menu.settings.cowebsiteTrigger()}</span>
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input
|
<input
|
||||||
@ -137,7 +193,7 @@
|
|||||||
bind:checked={ignoreFollowRequests}
|
bind:checked={ignoreFollowRequests}
|
||||||
on:change={changeIgnoreFollowRequests}
|
on:change={changeIgnoreFollowRequests}
|
||||||
/>
|
/>
|
||||||
<span>Ignore requests to follow other users</span>
|
<span>{$LL.menu.settings.ignoreFollowRequest()}</span>
|
||||||
</label>
|
</label>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
@ -174,6 +230,10 @@
|
|||||||
margin: 0 0 15px;
|
margin: 0 0 15px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.languages-switcher option {
|
||||||
|
text-transform: capitalize;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 800px), only screen and (max-height: 800px) {
|
@media only screen and (max-width: 800px), only screen and (max-height: 800px) {
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
import { AdminMessageEventTypes } from "../../Connexion/AdminMessagesService";
|
import { AdminMessageEventTypes } from "../../Connexion/AdminMessagesService";
|
||||||
import type { Quill } from "quill";
|
import type { Quill } from "quill";
|
||||||
import type { PlayGlobalMessageInterface } from "../../Connexion/ConnexionModels";
|
import type { PlayGlobalMessageInterface } from "../../Connexion/ConnexionModels";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
//toolbar
|
//toolbar
|
||||||
const toolbarOptions = [
|
const toolbarOptions = [
|
||||||
@ -58,7 +59,7 @@
|
|||||||
const { default: Quill } = await import("quill"); // eslint-disable-line @typescript-eslint/no-explicit-any
|
const { default: Quill } = await import("quill"); // eslint-disable-line @typescript-eslint/no-explicit-any
|
||||||
|
|
||||||
quill = new Quill(QUILL_EDITOR, {
|
quill = new Quill(QUILL_EDITOR, {
|
||||||
placeholder: "Enter your message here...",
|
placeholder: $LL.menu.globalMessage.enter(),
|
||||||
theme: "snow",
|
theme: "snow",
|
||||||
modules: {
|
modules: {
|
||||||
toolbar: toolbarOptions,
|
toolbar: toolbarOptions,
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
import SoundMeterWidget from "./SoundMeterWidget.svelte";
|
import SoundMeterWidget from "./SoundMeterWidget.svelte";
|
||||||
import { onDestroy } from "svelte";
|
import { onDestroy } from "svelte";
|
||||||
import { srcObject } from "./Video/utils";
|
import { srcObject } from "./Video/utils";
|
||||||
|
import LL from "../i18n/i18n-svelte";
|
||||||
|
|
||||||
let stream: MediaStream | null;
|
let stream: MediaStream | null;
|
||||||
|
|
||||||
@ -32,5 +33,5 @@
|
|||||||
<SoundMeterWidget {stream} />
|
<SoundMeterWidget {stream} />
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
<div class="is-silent" class:hide={isSilent}>Silent zone</div>
|
<div class="is-silent" class:hide={isSilent}>{$LL.camera.my.silentZone()}</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import { blackListManager } from "../../WebRtc/BlackListManager";
|
import { blackListManager } from "../../WebRtc/BlackListManager";
|
||||||
import { showReportScreenStore, userReportEmpty } from "../../Stores/ShowReportScreenStore";
|
import { showReportScreenStore, userReportEmpty } from "../../Stores/ShowReportScreenStore";
|
||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let userUUID: string | undefined;
|
export let userUUID: string | undefined;
|
||||||
export let userName: string;
|
export let userName: string;
|
||||||
@ -29,10 +30,10 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="block-container">
|
<div class="block-container">
|
||||||
<h3>Block</h3>
|
<h3>{$LL.report.block.title()}</h3>
|
||||||
<p>Block any communication from and to {userName}. This can be reverted.</p>
|
<p>{$LL.report.block.content({ userName })}</p>
|
||||||
<button type="button" class="nes-btn is-error" on:click|preventDefault={blockUser}>
|
<button type="button" class="nes-btn is-error" on:click|preventDefault={blockUser}>
|
||||||
{userIsBlocked ? "Unblock this user" : "Block this user"}
|
{userIsBlocked ? $LL.report.block.unblock() : $LL.report.block.block()}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
import { playersStore } from "../../Stores/PlayersStore";
|
import { playersStore } from "../../Stores/PlayersStore";
|
||||||
import { connectionManager } from "../../Connexion/ConnectionManager";
|
import { connectionManager } from "../../Connexion/ConnectionManager";
|
||||||
import { get } from "svelte/store";
|
import { get } from "svelte/store";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
let blockActive = true;
|
let blockActive = true;
|
||||||
let reportActive = !blockActive;
|
let reportActive = !blockActive;
|
||||||
@ -59,7 +60,7 @@
|
|||||||
|
|
||||||
<div class="report-menu-main nes-container is-rounded">
|
<div class="report-menu-main nes-container is-rounded">
|
||||||
<section class="report-menu-title">
|
<section class="report-menu-title">
|
||||||
<h2>Moderate {userName}</h2>
|
<h2>{$LL.report.moderate.title({ userName })}</h2>
|
||||||
<section class="justify-center">
|
<section class="justify-center">
|
||||||
<button type="button" class="nes-btn" on:click|preventDefault={close}>X</button>
|
<button type="button" class="nes-btn" on:click|preventDefault={close}>X</button>
|
||||||
</section>
|
</section>
|
||||||
@ -69,14 +70,14 @@
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nes-btn {blockActive ? 'is-disabled' : ''}"
|
class="nes-btn {blockActive ? 'is-disabled' : ''}"
|
||||||
on:click|preventDefault={activateBlock}>Block</button
|
on:click|preventDefault={activateBlock}>{$LL.report.moderate.block()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
<section class="justify-center">
|
<section class="justify-center">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nes-btn {reportActive ? 'is-disabled' : ''}"
|
class="nes-btn {reportActive ? 'is-disabled' : ''}"
|
||||||
on:click|preventDefault={activateReport}>Report</button
|
on:click|preventDefault={activateReport}>{$LL.report.moderate.report()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
@ -86,7 +87,7 @@
|
|||||||
{:else if reportActive}
|
{:else if reportActive}
|
||||||
<ReportSubMenu {userUUID} />
|
<ReportSubMenu {userUUID} />
|
||||||
{:else}
|
{:else}
|
||||||
<p>ERROR : There is no action selected.</p>
|
<p>{$LL.report.moderate.noSelect()}</p>
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { showReportScreenStore, userReportEmpty } from "../../Stores/ShowReportScreenStore";
|
import { showReportScreenStore, userReportEmpty } from "../../Stores/ShowReportScreenStore";
|
||||||
import { gameManager } from "../../Phaser/Game/GameManager";
|
import { gameManager } from "../../Phaser/Game/GameManager";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let userUUID: string | undefined;
|
export let userUUID: string | undefined;
|
||||||
let reportMessage: string;
|
let reportMessage: string;
|
||||||
@ -22,18 +23,18 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="report-container-main">
|
<div class="report-container-main">
|
||||||
<h3>Report</h3>
|
<h3>{$LL.report.title()}</h3>
|
||||||
<p>Send a report message to the administrators of this room. They may later ban this user.</p>
|
<p>{$LL.report.content()}</p>
|
||||||
<form>
|
<form>
|
||||||
<section>
|
<section>
|
||||||
<label>
|
<label>
|
||||||
<span>Your message: </span>
|
<span>{$LL.report.message.title()}</span>
|
||||||
<textarea type="text" class="nes-textarea" bind:value={reportMessage} />
|
<textarea type="text" class="nes-textarea" bind:value={reportMessage} />
|
||||||
</label>
|
</label>
|
||||||
<p hidden={hiddenError}>Report message cannot to be empty.</p>
|
<p hidden={hiddenError}>{$LL.report.message.empty()}</p>
|
||||||
</section>
|
</section>
|
||||||
<section>
|
<section>
|
||||||
<button type="submit" class="nes-btn is-error" on:click={submitReport}>Report this user</button>
|
<button type="submit" class="nes-btn is-error" on:click={submitReport}>{$LL.report.submit()}</button>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
<script lang="typescript">
|
<script lang="typescript">
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
import type { Game } from "../../Phaser/Game/Game";
|
import type { Game } from "../../Phaser/Game/Game";
|
||||||
import { SelectCompanionScene, SelectCompanionSceneName } from "../../Phaser/Login/SelectCompanionScene";
|
import { SelectCompanionScene, SelectCompanionSceneName } from "../../Phaser/Login/SelectCompanionScene";
|
||||||
|
|
||||||
@ -25,7 +26,7 @@
|
|||||||
|
|
||||||
<form class="selectCompanionScene">
|
<form class="selectCompanionScene">
|
||||||
<section class="text-center">
|
<section class="text-center">
|
||||||
<h2>Select your companion</h2>
|
<h2>{$LL.companion.select.title()}</h2>
|
||||||
<button class="selectCharacterButton selectCharacterButtonLeft nes-btn" on:click|preventDefault={selectLeft}>
|
<button class="selectCharacterButton selectCharacterButtonLeft nes-btn" on:click|preventDefault={selectLeft}>
|
||||||
<
|
<
|
||||||
</button>
|
</button>
|
||||||
@ -35,12 +36,12 @@
|
|||||||
</section>
|
</section>
|
||||||
<section class="action">
|
<section class="action">
|
||||||
<button href="/" class="selectCompanionSceneFormBack nes-btn" on:click|preventDefault={noCompanion}
|
<button href="/" class="selectCompanionSceneFormBack nes-btn" on:click|preventDefault={noCompanion}
|
||||||
>No companion</button
|
>{$LL.companion.select.any()}</button
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
class="selectCompanionSceneFormSubmit nes-btn is-primary"
|
class="selectCompanionSceneFormSubmit nes-btn is-primary"
|
||||||
on:click|preventDefault={selectCompanion}>Continue</button
|
on:click|preventDefault={selectCompanion}>{$LL.companion.select.continue()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
import type { Message } from "../../Stores/TypeMessageStore/MessageStore";
|
import type { Message } from "../../Stores/TypeMessageStore/MessageStore";
|
||||||
import { banMessageStore } from "../../Stores/TypeMessageStore/BanMessageStore";
|
import { banMessageStore } from "../../Stores/TypeMessageStore/BanMessageStore";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let message: Message;
|
export let message: Message;
|
||||||
|
|
||||||
@ -37,7 +38,8 @@
|
|||||||
out:fade={{ duration: 200 }}
|
out:fade={{ duration: 200 }}
|
||||||
>
|
>
|
||||||
<h2 class="title-ban-message">
|
<h2 class="title-ban-message">
|
||||||
<img src="resources/logos/report.svg" alt="***" /> Important message
|
<img src="resources/logos/report.svg" alt="***" />
|
||||||
|
{$LL.warning.importantMessage()}
|
||||||
<img src="resources/logos/report.svg" alt="***" />
|
<img src="resources/logos/report.svg" alt="***" />
|
||||||
</h2>
|
</h2>
|
||||||
<div class="content-ban-message">
|
<div class="content-ban-message">
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
import megaphoneImg from "./images/megaphone.svg";
|
import megaphoneImg from "./images/megaphone.svg";
|
||||||
import { soundPlayingStore } from "../../Stores/SoundPlayingStore";
|
import { soundPlayingStore } from "../../Stores/SoundPlayingStore";
|
||||||
import { afterUpdate } from "svelte";
|
import { afterUpdate } from "svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let url: string;
|
export let url: string;
|
||||||
let audio: HTMLAudioElement;
|
let audio: HTMLAudioElement;
|
||||||
@ -18,7 +19,7 @@
|
|||||||
|
|
||||||
<div class="audio-playing" transition:fly={{ x: 210, duration: 500 }}>
|
<div class="audio-playing" transition:fly={{ x: 210, duration: 500 }}>
|
||||||
<img src={megaphoneImg} alt="Audio playing" />
|
<img src={megaphoneImg} alt="Audio playing" />
|
||||||
<p>Audio message</p>
|
<p>{$LL.audio.message()}</p>
|
||||||
<audio bind:this={audio} src={url} on:ended={soundEnded}>
|
<audio bind:this={audio} src={url} on:ended={soundEnded}>
|
||||||
<track kind="captions" />
|
<track kind="captions" />
|
||||||
</audio>
|
</audio>
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { errorStore, hasClosableMessagesInErrorStore } from "../../Stores/ErrorStore";
|
import { errorStore, hasClosableMessagesInErrorStore } from "../../Stores/ErrorStore";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
function close(): boolean {
|
function close(): boolean {
|
||||||
errorStore.clearClosableMessages();
|
errorStore.clearClosableMessages();
|
||||||
@ -8,7 +9,7 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="error-div nes-container is-dark is-rounded" open>
|
<div class="error-div nes-container is-dark is-rounded" open>
|
||||||
<p class="nes-text is-error title">Error</p>
|
<p class="nes-text is-error title">{$LL.error.error()}</p>
|
||||||
<div class="body">
|
<div class="body">
|
||||||
{#each $errorStore as error}
|
{#each $errorStore as error}
|
||||||
<p>{error.message}</p>
|
<p>{error.message}</p>
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import { fly } from "svelte/transition";
|
import { fly } from "svelte/transition";
|
||||||
import { requestVisitCardsStore } from "../../Stores/GameStore";
|
import { requestVisitCardsStore } from "../../Stores/GameStore";
|
||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let visitCardUrl: string;
|
export let visitCardUrl: string;
|
||||||
let w = "500px";
|
let w = "500px";
|
||||||
@ -40,7 +41,7 @@
|
|||||||
/>
|
/>
|
||||||
{#if !hidden}
|
{#if !hidden}
|
||||||
<div class="buttonContainer">
|
<div class="buttonContainer">
|
||||||
<button class="nes-btn is-popUpElement" on:click={closeCard}>Close</button>
|
<button class="nes-btn is-popUpElement" on:click={closeCard}>{$LL.menu.visitCard.close()}</button>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import { fly } from "svelte/transition";
|
import { fly } from "svelte/transition";
|
||||||
import { userIsAdminStore, limitMapStore } from "../../Stores/GameStore";
|
import { userIsAdminStore, limitMapStore } from "../../Stores/GameStore";
|
||||||
import { ADMIN_URL } from "../../Enum/EnvironmentVariable";
|
import { ADMIN_URL } from "../../Enum/EnvironmentVariable";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
const upgradeLink = ADMIN_URL + "/pricing";
|
const upgradeLink = ADMIN_URL + "/pricing";
|
||||||
const registerLink = ADMIN_URL + "/second-step-register";
|
const registerLink = ADMIN_URL + "/second-step-register";
|
||||||
@ -9,19 +10,17 @@
|
|||||||
|
|
||||||
<main class="warningMain" transition:fly={{ y: -200, duration: 500 }}>
|
<main class="warningMain" transition:fly={{ y: -200, duration: 500 }}>
|
||||||
{#if $userIsAdminStore}
|
{#if $userIsAdminStore}
|
||||||
<h2>Warning!</h2>
|
<h2>{$LL.warning.title()}</h2>
|
||||||
<p>
|
<p>
|
||||||
This world is close to its limit!. You can upgrade its capacity <a href={upgradeLink} target="_blank"
|
{$LL.warning.content({ upgradeLink })}
|
||||||
>here</a
|
|
||||||
>
|
|
||||||
</p>
|
</p>
|
||||||
{:else if $limitMapStore}
|
{:else if $limitMapStore}
|
||||||
<p>
|
<p>
|
||||||
This map is available for 2 days. You can register your domain <a href={registerLink}>here</a>!
|
This map is available for 2 days. You can register your domain <a href={registerLink}>here</a>!
|
||||||
</p>
|
</p>
|
||||||
{:else}
|
{:else}
|
||||||
<h2>Warning!</h2>
|
<h2>{$LL.warning.title()}</h2>
|
||||||
<p>This world is close to its limit!</p>
|
<p>{$LL.warning.limit()}</p>
|
||||||
{/if}
|
{/if}
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script lang="typescript">
|
<script lang="typescript">
|
||||||
import type { Game } from "../../Phaser/Game/Game";
|
import type { Game } from "../../Phaser/Game/Game";
|
||||||
import { SelectCharacterScene, SelectCharacterSceneName } from "../../Phaser/Login/SelectCharacterScene";
|
import { SelectCharacterScene, SelectCharacterSceneName } from "../../Phaser/Login/SelectCharacterScene";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export let game: Game;
|
export let game: Game;
|
||||||
|
|
||||||
@ -25,7 +26,7 @@
|
|||||||
|
|
||||||
<form class="selectCharacterScene">
|
<form class="selectCharacterScene">
|
||||||
<section class="text-center">
|
<section class="text-center">
|
||||||
<h2>Select your WOKA</h2>
|
<h2>{$LL.woka.selectWoka.title()}</h2>
|
||||||
<button class="selectCharacterButton selectCharacterButtonLeft nes-btn" on:click|preventDefault={selectLeft}>
|
<button class="selectCharacterButton selectCharacterButtonLeft nes-btn" on:click|preventDefault={selectLeft}>
|
||||||
<
|
<
|
||||||
</button>
|
</button>
|
||||||
@ -37,12 +38,12 @@
|
|||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
class="selectCharacterSceneFormSubmit nes-btn is-primary"
|
class="selectCharacterSceneFormSubmit nes-btn is-primary"
|
||||||
on:click|preventDefault={cameraScene}>Continue</button
|
on:click|preventDefault={cameraScene}>{$LL.woka.selectWoka.continue()}</button
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
class="selectCharacterSceneFormCustomYourOwnSubmit nes-btn"
|
class="selectCharacterSceneFormCustomYourOwnSubmit nes-btn"
|
||||||
on:click|preventDefault={customizeScene}>Customize your WOKA</button
|
on:click|preventDefault={customizeScene}>{$LL.woka.selectWoka.customize()}</button
|
||||||
>
|
>
|
||||||
</section>
|
</section>
|
||||||
</form>
|
</form>
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import * as rax from "retry-axios";
|
import * as rax from "retry-axios";
|
||||||
import { errorStore } from "../Stores/ErrorStore";
|
import { errorStore } from "../Stores/ErrorStore";
|
||||||
|
import LL from "../i18n/i18n-svelte";
|
||||||
|
import { get } from "svelte/store";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This instance of Axios will retry in case of an issue and display an error message as a HTML overlay.
|
* This instance of Axios will retry in case of an issue and display an error message as a HTML overlay.
|
||||||
@ -26,7 +28,7 @@ axiosWithRetry.defaults.raxConfig = {
|
|||||||
console.log(err);
|
console.log(err);
|
||||||
console.log(cfg);
|
console.log(cfg);
|
||||||
console.log(`Retry attempt #${cfg?.currentRetryAttempt} on URL '${err.config.url}'`);
|
console.log(`Retry attempt #${cfg?.currentRetryAttempt} on URL '${err.config.url}'`);
|
||||||
errorStore.addErrorMessage("Unable to connect to WorkAdventure. Are you connected to internet?", {
|
errorStore.addErrorMessage(get(LL).error.connectionRetry.unableConnect(), {
|
||||||
closable: false,
|
closable: false,
|
||||||
id: "axios_retry",
|
id: "axios_retry",
|
||||||
});
|
});
|
||||||
|
@ -25,6 +25,7 @@ export const POSTHOG_API_KEY: string = (process.env.POSTHOG_API_KEY as string) |
|
|||||||
export const POSTHOG_URL = process.env.POSTHOG_URL || undefined;
|
export const POSTHOG_URL = process.env.POSTHOG_URL || undefined;
|
||||||
export const DISABLE_ANONYMOUS: boolean = process.env.DISABLE_ANONYMOUS === "true";
|
export const DISABLE_ANONYMOUS: boolean = process.env.DISABLE_ANONYMOUS === "true";
|
||||||
export const OPID_LOGIN_SCREEN_PROVIDER = process.env.OPID_LOGIN_SCREEN_PROVIDER;
|
export const OPID_LOGIN_SCREEN_PROVIDER = process.env.OPID_LOGIN_SCREEN_PROVIDER;
|
||||||
|
const FALLBACK_LOCALE = process.env.FALLBACK_LOCALE || undefined;
|
||||||
|
|
||||||
export const isMobile = (): boolean => window.innerWidth <= 800 || window.innerHeight <= 600;
|
export const isMobile = (): boolean => window.innerWidth <= 800 || window.innerHeight <= 600;
|
||||||
|
|
||||||
@ -44,4 +45,5 @@ export {
|
|||||||
TURN_PASSWORD,
|
TURN_PASSWORD,
|
||||||
JITSI_URL,
|
JITSI_URL,
|
||||||
JITSI_PRIVATE_MODE,
|
JITSI_PRIVATE_MODE,
|
||||||
|
FALLBACK_LOCALE,
|
||||||
};
|
};
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
import { GameScene } from "./GameScene";
|
import { get } from "svelte/store";
|
||||||
import { connectionManager } from "../../Connexion/ConnectionManager";
|
import { connectionManager } from "../../Connexion/ConnectionManager";
|
||||||
|
import { localUserStore } from "../../Connexion/LocalUserStore";
|
||||||
import type { Room } from "../../Connexion/Room";
|
import type { Room } from "../../Connexion/Room";
|
||||||
|
import { helpCameraSettingsVisibleStore } from "../../Stores/HelpCameraSettingsStore";
|
||||||
|
import { requestedCameraState, requestedMicrophoneState } from "../../Stores/MediaStore";
|
||||||
|
import { menuIconVisiblilityStore } from "../../Stores/MenuStore";
|
||||||
|
import { EnableCameraSceneName } from "../Login/EnableCameraScene";
|
||||||
import { LoginSceneName } from "../Login/LoginScene";
|
import { LoginSceneName } from "../Login/LoginScene";
|
||||||
import { SelectCharacterSceneName } from "../Login/SelectCharacterScene";
|
import { SelectCharacterSceneName } from "../Login/SelectCharacterScene";
|
||||||
import { EnableCameraSceneName } from "../Login/EnableCameraScene";
|
import { GameScene } from "./GameScene";
|
||||||
import { localUserStore } from "../../Connexion/LocalUserStore";
|
|
||||||
import { get } from "svelte/store";
|
|
||||||
import { requestedCameraState, requestedMicrophoneState } from "../../Stores/MediaStore";
|
|
||||||
import { helpCameraSettingsVisibleStore } from "../../Stores/HelpCameraSettingsStore";
|
|
||||||
import { menuIconVisiblilityStore } from "../../Stores/MenuStore";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class should be responsible for any scene starting/stopping
|
* This class should be responsible for any scene starting/stopping
|
||||||
|
@ -91,6 +91,7 @@ import { MapStore } from "../../Stores/Utils/MapStore";
|
|||||||
import { followUsersColorStore } from "../../Stores/FollowStore";
|
import { followUsersColorStore } from "../../Stores/FollowStore";
|
||||||
import Camera = Phaser.Cameras.Scene2D.Camera;
|
import Camera = Phaser.Cameras.Scene2D.Camera;
|
||||||
import { GameSceneUserInputHandler } from "../UserInput/GameSceneUserInputHandler";
|
import { GameSceneUserInputHandler } from "../UserInput/GameSceneUserInputHandler";
|
||||||
|
import { locale } from "../../i18n/i18n-svelte";
|
||||||
|
|
||||||
export interface GameSceneInitInterface {
|
export interface GameSceneInitInterface {
|
||||||
initPosition: PointInterface | null;
|
initPosition: PointInterface | null;
|
||||||
@ -1325,6 +1326,7 @@ ${escapedMessage}
|
|||||||
startLayerName: this.startPositionCalculator.startLayerName,
|
startLayerName: this.startPositionCalculator.startLayerName,
|
||||||
uuid: localUserStore.getLocalUser()?.uuid,
|
uuid: localUserStore.getLocalUser()?.uuid,
|
||||||
nickname: this.playerName,
|
nickname: this.playerName,
|
||||||
|
language: get(locale),
|
||||||
roomId: this.roomUrl,
|
roomId: this.roomUrl,
|
||||||
tags: this.connection ? this.connection.getAllTags() : [],
|
tags: this.connection ? this.connection.getAllTags() : [],
|
||||||
variables: this.sharedVariablesManager.variables,
|
variables: this.sharedVariablesManager.variables,
|
||||||
|
@ -4,6 +4,11 @@ import { ErrorScene, ErrorSceneName } from "../Reconnecting/ErrorScene";
|
|||||||
import { WAError } from "../Reconnecting/WAError";
|
import { WAError } from "../Reconnecting/WAError";
|
||||||
import { waScaleManager } from "../Services/WaScaleManager";
|
import { waScaleManager } from "../Services/WaScaleManager";
|
||||||
import { ReconnectingTextures } from "../Reconnecting/ReconnectingScene";
|
import { ReconnectingTextures } from "../Reconnecting/ReconnectingScene";
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
import { get } from "svelte/store";
|
||||||
|
import { localeDetector } from "../../i18n/locales";
|
||||||
|
|
||||||
|
const $LL = get(LL);
|
||||||
|
|
||||||
export const EntrySceneName = "EntryScene";
|
export const EntrySceneName = "EntryScene";
|
||||||
|
|
||||||
@ -27,38 +32,44 @@ export class EntryScene extends Scene {
|
|||||||
}
|
}
|
||||||
|
|
||||||
create() {
|
create() {
|
||||||
gameManager
|
localeDetector()
|
||||||
.init(this.scene)
|
.then(() => {
|
||||||
.then((nextSceneName) => {
|
gameManager
|
||||||
// Let's rescale before starting the game
|
.init(this.scene)
|
||||||
// We can do it at this stage.
|
.then((nextSceneName) => {
|
||||||
waScaleManager.applyNewSize();
|
// Let's rescale before starting the game
|
||||||
this.scene.start(nextSceneName);
|
// We can do it at this stage.
|
||||||
|
waScaleManager.applyNewSize();
|
||||||
|
this.scene.start(nextSceneName);
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
if (err.response && err.response.status == 404) {
|
||||||
|
ErrorScene.showError(
|
||||||
|
new WAError(
|
||||||
|
$LL.error.accessLink.title(),
|
||||||
|
$LL.error.accessLink.subTitle(),
|
||||||
|
$LL.error.accessLink.details()
|
||||||
|
),
|
||||||
|
this.scene
|
||||||
|
);
|
||||||
|
} else if (err.response && err.response.status == 403) {
|
||||||
|
ErrorScene.showError(
|
||||||
|
new WAError(
|
||||||
|
$LL.error.connectionRejected.title(),
|
||||||
|
$LL.error.connectionRejected.subTitle({
|
||||||
|
error: err.response.data ? ". \n\r \n\r" + `${err.response.data}` : "",
|
||||||
|
}),
|
||||||
|
$LL.error.connectionRejected.details()
|
||||||
|
),
|
||||||
|
this.scene
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
ErrorScene.showError(err, this.scene);
|
||||||
|
}
|
||||||
|
});
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch(() => {
|
||||||
if (err.response && err.response.status == 404) {
|
throw new Error("Cannot load locale!");
|
||||||
ErrorScene.showError(
|
|
||||||
new WAError(
|
|
||||||
"Access link incorrect",
|
|
||||||
"Could not find map. Please check your access link.",
|
|
||||||
"If you want more information, you may contact administrator or contact us at: hello@workadventu.re"
|
|
||||||
),
|
|
||||||
this.scene
|
|
||||||
);
|
|
||||||
} else if (err.response && err.response.status == 403) {
|
|
||||||
ErrorScene.showError(
|
|
||||||
new WAError(
|
|
||||||
"Connection rejected",
|
|
||||||
"You cannot join the World. Try again later" +
|
|
||||||
(err.response.data ? ". \n\r \n\r" + `${err.response.data}` : "") +
|
|
||||||
".",
|
|
||||||
"If you want more information, you may contact administrator or contact us at: hello@workadventu.re"
|
|
||||||
),
|
|
||||||
this.scene
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
ErrorScene.showError(err, this.scene);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import { TextField } from "../Components/TextField";
|
import { TextField } from "../Components/TextField";
|
||||||
import Image = Phaser.GameObjects.Image;
|
import Image = Phaser.GameObjects.Image;
|
||||||
import Sprite = Phaser.GameObjects.Sprite;
|
import Sprite = Phaser.GameObjects.Sprite;
|
||||||
|
import LL from "../../i18n/i18n-svelte";
|
||||||
|
import { get } from "svelte/store";
|
||||||
|
|
||||||
export const ReconnectingSceneName = "ReconnectingScene";
|
export const ReconnectingSceneName = "ReconnectingScene";
|
||||||
export enum ReconnectingTextures {
|
export enum ReconnectingTextures {
|
||||||
@ -38,7 +40,7 @@ export class ReconnectingScene extends Phaser.Scene {
|
|||||||
this,
|
this,
|
||||||
this.game.renderer.width / 2,
|
this.game.renderer.width / 2,
|
||||||
this.game.renderer.height / 2,
|
this.game.renderer.height / 2,
|
||||||
"Connection lost. Reconnecting..."
|
get(LL).warning.connectionLost()
|
||||||
);
|
);
|
||||||
|
|
||||||
const cat = this.add.sprite(this.game.renderer.width / 2, this.game.renderer.height / 2 - 32, "cat");
|
const cat = this.add.sprite(this.game.renderer.width / 2, this.game.renderer.height / 2 - 32, "cat");
|
||||||
|
@ -3,6 +3,7 @@ import Timeout = NodeJS.Timeout;
|
|||||||
import { userIsAdminStore } from "./GameStore";
|
import { userIsAdminStore } from "./GameStore";
|
||||||
import { CONTACT_URL } from "../Enum/EnvironmentVariable";
|
import { CONTACT_URL } from "../Enum/EnvironmentVariable";
|
||||||
import { analyticsClient } from "../Administration/AnalyticsClient";
|
import { analyticsClient } from "../Administration/AnalyticsClient";
|
||||||
|
import type { Translation } from "../i18n/i18n-types";
|
||||||
|
|
||||||
export const menuIconVisiblilityStore = writable(false);
|
export const menuIconVisiblilityStore = writable(false);
|
||||||
export const menuVisiblilityStore = writable(false);
|
export const menuVisiblilityStore = writable(false);
|
||||||
@ -32,37 +33,95 @@ function createWarningContainerStore() {
|
|||||||
export const warningContainerStore = createWarningContainerStore();
|
export const warningContainerStore = createWarningContainerStore();
|
||||||
|
|
||||||
export enum SubMenusInterface {
|
export enum SubMenusInterface {
|
||||||
settings = "Settings",
|
settings = "settings",
|
||||||
profile = "Profile",
|
profile = "profile",
|
||||||
invite = "Invite",
|
invite = "invite",
|
||||||
aboutRoom = "Credit",
|
aboutRoom = "credit",
|
||||||
globalMessages = "Global Messages",
|
globalMessages = "globalMessages",
|
||||||
contact = "Contact",
|
contact = "contact",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type MenuKeys = keyof Translation["menu"]["sub"];
|
||||||
|
|
||||||
|
interface TranslatedMenu {
|
||||||
|
type: "translated";
|
||||||
|
key: MenuKeys;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A menu item from the scripting API
|
||||||
|
*/
|
||||||
|
interface ScriptingMenu {
|
||||||
|
type: "scripting";
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type MenuItem = TranslatedMenu | ScriptingMenu;
|
||||||
|
|
||||||
function createSubMenusStore() {
|
function createSubMenusStore() {
|
||||||
const { subscribe, update } = writable<string[]>([
|
const { subscribe, update } = writable<MenuItem[]>([
|
||||||
SubMenusInterface.profile,
|
{
|
||||||
SubMenusInterface.globalMessages,
|
type: "translated",
|
||||||
SubMenusInterface.contact,
|
key: SubMenusInterface.profile,
|
||||||
SubMenusInterface.settings,
|
},
|
||||||
SubMenusInterface.invite,
|
{
|
||||||
SubMenusInterface.aboutRoom,
|
type: "translated",
|
||||||
|
key: SubMenusInterface.globalMessages,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "translated",
|
||||||
|
key: SubMenusInterface.contact,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "translated",
|
||||||
|
key: SubMenusInterface.settings,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "translated",
|
||||||
|
key: SubMenusInterface.invite,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "translated",
|
||||||
|
key: SubMenusInterface.aboutRoom,
|
||||||
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
subscribe,
|
subscribe,
|
||||||
addMenu(menuCommand: string) {
|
addTranslatedMenu(menuCommand: MenuKeys) {
|
||||||
update((menuList: string[]) => {
|
update((menuList) => {
|
||||||
if (!menuList.find((menu) => menu === menuCommand)) {
|
if (!menuList.find((menu) => menu.type === "translated" && menu.key === menuCommand)) {
|
||||||
menuList.push(menuCommand);
|
menuList.push({
|
||||||
|
type: "translated",
|
||||||
|
key: menuCommand,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
return menuList;
|
return menuList;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
removeMenu(menuCommand: string) {
|
removeTranslatedMenu(menuCommand: MenuKeys) {
|
||||||
update((menuList: string[]) => {
|
update((menuList) => {
|
||||||
const index = menuList.findIndex((menu) => menu === menuCommand);
|
const index = menuList.findIndex((menu) => menu.type === "translated" && menu.key === menuCommand);
|
||||||
|
if (index !== -1) {
|
||||||
|
menuList.splice(index, 1);
|
||||||
|
}
|
||||||
|
return menuList;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
addScriptingMenu(menuCommand: string) {
|
||||||
|
update((menuList) => {
|
||||||
|
if (!menuList.find((menu) => menu.type === "scripting" && menu.label === menuCommand)) {
|
||||||
|
menuList.push({
|
||||||
|
type: "scripting",
|
||||||
|
label: menuCommand,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return menuList;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
removeScriptingMenu(menuCommand: string) {
|
||||||
|
update((menuList) => {
|
||||||
|
const index = menuList.findIndex((menu) => menu.type === "scripting" && menu.label === menuCommand);
|
||||||
if (index !== -1) {
|
if (index !== -1) {
|
||||||
menuList.splice(index, 1);
|
menuList.splice(index, 1);
|
||||||
}
|
}
|
||||||
@ -77,15 +136,15 @@ export const subMenusStore = createSubMenusStore();
|
|||||||
export const contactPageStore = writable<string | undefined>(CONTACT_URL);
|
export const contactPageStore = writable<string | undefined>(CONTACT_URL);
|
||||||
|
|
||||||
export function checkSubMenuToShow() {
|
export function checkSubMenuToShow() {
|
||||||
subMenusStore.removeMenu(SubMenusInterface.globalMessages);
|
subMenusStore.removeTranslatedMenu(SubMenusInterface.globalMessages);
|
||||||
subMenusStore.removeMenu(SubMenusInterface.contact);
|
subMenusStore.removeTranslatedMenu(SubMenusInterface.contact);
|
||||||
|
|
||||||
if (get(userIsAdminStore)) {
|
if (get(userIsAdminStore)) {
|
||||||
subMenusStore.addMenu(SubMenusInterface.globalMessages);
|
subMenusStore.addTranslatedMenu(SubMenusInterface.globalMessages);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (get(contactPageStore) !== undefined) {
|
if (get(contactPageStore) !== undefined) {
|
||||||
subMenusStore.addMenu(SubMenusInterface.contact);
|
subMenusStore.addTranslatedMenu(SubMenusInterface.contact);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,12 +156,12 @@ export function handleMenuRegistrationEvent(
|
|||||||
source: string | undefined = undefined,
|
source: string | undefined = undefined,
|
||||||
options: { allowApi: boolean }
|
options: { allowApi: boolean }
|
||||||
) {
|
) {
|
||||||
if (get(subMenusStore).includes(menuName)) {
|
if (get(subMenusStore).find((item) => item.type === "scripting" && item.label === menuName)) {
|
||||||
console.warn("The menu " + menuName + " already exist.");
|
console.warn("The menu " + menuName + " already exist.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
subMenusStore.addMenu(menuName);
|
subMenusStore.addScriptingMenu(menuName);
|
||||||
|
|
||||||
if (iframeUrl !== undefined) {
|
if (iframeUrl !== undefined) {
|
||||||
const url = new URL(iframeUrl, source);
|
const url = new URL(iframeUrl, source);
|
||||||
@ -111,12 +170,6 @@ export function handleMenuRegistrationEvent(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function handleMenuUnregisterEvent(menuName: string) {
|
export function handleMenuUnregisterEvent(menuName: string) {
|
||||||
const subMenuGeneral: string[] = Object.values(SubMenusInterface);
|
subMenusStore.removeScriptingMenu(menuName);
|
||||||
if (subMenuGeneral.includes(menuName)) {
|
|
||||||
console.warn("The menu " + menuName + " is a mandatory menu. It can't be remove");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
subMenusStore.removeMenu(menuName);
|
|
||||||
customMenuIframe.delete(menuName);
|
customMenuIframe.delete(menuName);
|
||||||
}
|
}
|
||||||
|
@ -9,10 +9,12 @@ export type StopScreenSharingCallback = (media: MediaStream) => void;
|
|||||||
|
|
||||||
import { cowebsiteCloseButtonId } from "./CoWebsiteManager";
|
import { cowebsiteCloseButtonId } from "./CoWebsiteManager";
|
||||||
import { gameOverlayVisibilityStore } from "../Stores/GameOverlayStoreVisibility";
|
import { gameOverlayVisibilityStore } from "../Stores/GameOverlayStoreVisibility";
|
||||||
import { layoutManagerActionStore, layoutManagerVisibilityStore } from "../Stores/LayoutManagerStore";
|
import { layoutManagerActionStore } from "../Stores/LayoutManagerStore";
|
||||||
import { get } from "svelte/store";
|
|
||||||
import { localUserStore } from "../Connexion/LocalUserStore";
|
|
||||||
import { MediaStreamConstraintsError } from "../Stores/Errors/MediaStreamConstraintsError";
|
import { MediaStreamConstraintsError } from "../Stores/Errors/MediaStreamConstraintsError";
|
||||||
|
import { localUserStore } from "../Connexion/LocalUserStore";
|
||||||
|
import LL from "../i18n/i18n-svelte";
|
||||||
|
import { get } from "svelte/store";
|
||||||
|
import { localeDetector } from "../i18n/locales";
|
||||||
|
|
||||||
export class MediaManager {
|
export class MediaManager {
|
||||||
startScreenSharingCallBacks: Set<StartScreenSharingCallback> = new Set<StartScreenSharingCallback>();
|
startScreenSharingCallBacks: Set<StartScreenSharingCallback> = new Set<StartScreenSharingCallback>();
|
||||||
@ -23,46 +25,52 @@ export class MediaManager {
|
|||||||
private userInputManager?: UserInputManager;
|
private userInputManager?: UserInputManager;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
localStreamStore.subscribe((result) => {
|
localeDetector()
|
||||||
if (result.type === "error") {
|
.catch(() => {
|
||||||
if (result.error.name !== MediaStreamConstraintsError.NAME) {
|
throw new Error("Cannot load locale on media manager");
|
||||||
layoutManagerActionStore.addAction({
|
})
|
||||||
uuid: "cameraAccessDenied",
|
.finally(() => {
|
||||||
type: "warning",
|
localStreamStore.subscribe((result) => {
|
||||||
message: "Camera access denied. Click here and check your browser permissions.",
|
if (result.type === "error") {
|
||||||
callback: () => {
|
if (result.error.name !== MediaStreamConstraintsError.NAME) {
|
||||||
helpCameraSettingsVisibleStore.set(true);
|
layoutManagerActionStore.addAction({
|
||||||
},
|
uuid: "cameraAccessDenied",
|
||||||
userInputManager: this.userInputManager,
|
type: "warning",
|
||||||
});
|
message: get(LL).warning.accessDenied.camera(),
|
||||||
}
|
callback: () => {
|
||||||
//remove it after 10 sec
|
helpCameraSettingsVisibleStore.set(true);
|
||||||
setTimeout(() => {
|
},
|
||||||
layoutManagerActionStore.removeAction("cameraAccessDenied");
|
userInputManager: this.userInputManager,
|
||||||
}, 10000);
|
});
|
||||||
return;
|
}
|
||||||
}
|
//remove it after 10 sec
|
||||||
});
|
setTimeout(() => {
|
||||||
|
layoutManagerActionStore.removeAction("cameraAccessDenied");
|
||||||
screenSharingLocalStreamStore.subscribe((result) => {
|
}, 10000);
|
||||||
if (result.type === "error") {
|
return;
|
||||||
console.error(result.error);
|
}
|
||||||
layoutManagerActionStore.addAction({
|
|
||||||
uuid: "screenSharingAccessDenied",
|
|
||||||
type: "warning",
|
|
||||||
message: "Screen sharing denied. Click here and check your browser permissions.",
|
|
||||||
callback: () => {
|
|
||||||
helpCameraSettingsVisibleStore.set(true);
|
|
||||||
},
|
|
||||||
userInputManager: this.userInputManager,
|
|
||||||
});
|
});
|
||||||
//remove it after 10 sec
|
|
||||||
setTimeout(() => {
|
screenSharingLocalStreamStore.subscribe((result) => {
|
||||||
layoutManagerActionStore.removeAction("screenSharingAccessDenied");
|
if (result.type === "error") {
|
||||||
}, 10000);
|
console.error(result.error);
|
||||||
return;
|
layoutManagerActionStore.addAction({
|
||||||
}
|
uuid: "screenSharingAccessDenied",
|
||||||
});
|
type: "warning",
|
||||||
|
message: get(LL).warning.accessDenied.screenSharing(),
|
||||||
|
callback: () => {
|
||||||
|
helpCameraSettingsVisibleStore.set(true);
|
||||||
|
},
|
||||||
|
userInputManager: this.userInputManager,
|
||||||
|
});
|
||||||
|
//remove it after 10 sec
|
||||||
|
setTimeout(() => {
|
||||||
|
layoutManagerActionStore.removeAction("screenSharingAccessDenied");
|
||||||
|
}, 10000);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public showGameOverlay(): void {
|
public showGameOverlay(): void {
|
||||||
|
3
front/src/i18n/.gitignore
vendored
Normal file
3
front/src/i18n/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
i18n-svelte.ts
|
||||||
|
i18n-types.ts
|
||||||
|
i18n-util.ts
|
10
front/src/i18n/en-US/audio.ts
Normal file
10
front/src/i18n/en-US/audio.ts
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const audio: BaseTranslation = {
|
||||||
|
manager: {
|
||||||
|
reduce: "reduce in conversations",
|
||||||
|
},
|
||||||
|
message: "Audio message",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default audio;
|
22
front/src/i18n/en-US/camera.ts
Normal file
22
front/src/i18n/en-US/camera.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const camera: BaseTranslation = {
|
||||||
|
enable: {
|
||||||
|
title: "Turn on your camera and microphone",
|
||||||
|
start: "Let's go!",
|
||||||
|
},
|
||||||
|
help: {
|
||||||
|
title: "Camera / Microphone access needed",
|
||||||
|
permissionDenied: "Permission denied",
|
||||||
|
content: "You must allow camera and microphone access in your browser.",
|
||||||
|
firefoxContent:
|
||||||
|
'Please click the "Remember this decision" checkbox, if you don\'t want Firefox to keep asking you the authorization.',
|
||||||
|
refresh: "Refresh",
|
||||||
|
continue: "Continue without webcam",
|
||||||
|
},
|
||||||
|
my: {
|
||||||
|
silentZone: "Silent zone",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default camera;
|
12
front/src/i18n/en-US/chat.ts
Normal file
12
front/src/i18n/en-US/chat.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const chat: BaseTranslation = {
|
||||||
|
intro: "Here is your chat history:",
|
||||||
|
enter: "Enter your message...",
|
||||||
|
menu: {
|
||||||
|
visitCard: "Visit card",
|
||||||
|
addFriend: "Add friend",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default chat;
|
11
front/src/i18n/en-US/companion.ts
Normal file
11
front/src/i18n/en-US/companion.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const companion: BaseTranslation = {
|
||||||
|
select: {
|
||||||
|
title: "Select your companion",
|
||||||
|
any: "No companion",
|
||||||
|
continue: "Continue",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default companion;
|
20
front/src/i18n/en-US/error.ts
Normal file
20
front/src/i18n/en-US/error.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const error: BaseTranslation = {
|
||||||
|
accessLink: {
|
||||||
|
title: "Access link incorrect",
|
||||||
|
subTitle: "Could not find map. Please check your access link.",
|
||||||
|
details: "If you want more information, you may contact administrator or contact us at: hello@workadventu.re",
|
||||||
|
},
|
||||||
|
connectionRejected: {
|
||||||
|
title: "Connection rejected",
|
||||||
|
subTitle: "You cannot join the World. Try again later {error}.",
|
||||||
|
details: "If you want more information, you may contact administrator or contact us at: hello@workadventu.re",
|
||||||
|
},
|
||||||
|
connectionRetry: {
|
||||||
|
unableConnect: "Unable to connect to WorkAdventure. Are you connected to internet?",
|
||||||
|
},
|
||||||
|
error: "Error",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default error;
|
27
front/src/i18n/en-US/follow.ts
Normal file
27
front/src/i18n/en-US/follow.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const follow: BaseTranslation = {
|
||||||
|
interactStatus: {
|
||||||
|
following: "Following {leader}",
|
||||||
|
waitingFollowers: "Waiting for followers confirmation",
|
||||||
|
followed: {
|
||||||
|
one: "{follower} is following you",
|
||||||
|
two: "{firstFollower} and {secondFollower} are following you",
|
||||||
|
many: "{followers} and {lastFollower} are following you",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
interactMenu: {
|
||||||
|
title: {
|
||||||
|
interact: "Interaction",
|
||||||
|
follow: "Do you want to follow {leader}?",
|
||||||
|
},
|
||||||
|
stop: {
|
||||||
|
leader: "Do you want to stop leading the way?",
|
||||||
|
follower: "Do you want to stop following {leader}?",
|
||||||
|
},
|
||||||
|
yes: "Yes",
|
||||||
|
no: "No",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default follow;
|
30
front/src/i18n/en-US/index.ts
Normal file
30
front/src/i18n/en-US/index.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
import audio from "./audio";
|
||||||
|
import camera from "./camera";
|
||||||
|
import chat from "./chat";
|
||||||
|
import companion from "./companion";
|
||||||
|
import woka from "./woka";
|
||||||
|
import error from "./error";
|
||||||
|
import follow from "./follow";
|
||||||
|
import login from "./login";
|
||||||
|
import menu from "./menu";
|
||||||
|
import report from "./report";
|
||||||
|
import warning from "./warning";
|
||||||
|
|
||||||
|
const en_US: BaseTranslation = {
|
||||||
|
language: "English",
|
||||||
|
country: "United States",
|
||||||
|
audio,
|
||||||
|
camera,
|
||||||
|
chat,
|
||||||
|
companion,
|
||||||
|
woka,
|
||||||
|
error,
|
||||||
|
follow,
|
||||||
|
login,
|
||||||
|
menu,
|
||||||
|
report,
|
||||||
|
warning,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default en_US;
|
14
front/src/i18n/en-US/login.ts
Normal file
14
front/src/i18n/en-US/login.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const login: BaseTranslation = {
|
||||||
|
input: {
|
||||||
|
name: {
|
||||||
|
placeholder: "Enter your name",
|
||||||
|
empty: "The name is empty",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
terms: 'By continuing, you are agreeing our <a href="https://workadventu.re/terms-of-use" target="_blank">terms of use</a>, <a href="https://workadventu.re/privacy-policy" target="_blank">privacy policy</a> and <a href="https://workadventu.re/cookie-policy" target="_blank">cookie policy</a>.',
|
||||||
|
continue: "Continue",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default login;
|
124
front/src/i18n/en-US/menu.ts
Normal file
124
front/src/i18n/en-US/menu.ts
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const menu: BaseTranslation = {
|
||||||
|
title: "Menu",
|
||||||
|
icon: {
|
||||||
|
open: {
|
||||||
|
menu: "Open menu",
|
||||||
|
invite: "Show invite",
|
||||||
|
register: "Register",
|
||||||
|
chat: "Open chat",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
visitCard: {
|
||||||
|
close: "Close",
|
||||||
|
},
|
||||||
|
profile: {
|
||||||
|
edit: {
|
||||||
|
name: "Edit your name",
|
||||||
|
woka: "Edit your WOKA",
|
||||||
|
companion: "Edit your companion",
|
||||||
|
camera: "Edit your camera",
|
||||||
|
},
|
||||||
|
login: "Sign in",
|
||||||
|
logout: "Log out",
|
||||||
|
},
|
||||||
|
settings: {
|
||||||
|
gameQuality: {
|
||||||
|
title: "Game quality",
|
||||||
|
short: {
|
||||||
|
high: "High (120 fps)",
|
||||||
|
medium: "Medium (60 fps)",
|
||||||
|
minimum: "Minimum (40 fps)",
|
||||||
|
small: "Small (20 fps)",
|
||||||
|
},
|
||||||
|
long: {
|
||||||
|
high: "High video quality (120 fps)",
|
||||||
|
medium: "Medium video quality (60 fps, recommended)",
|
||||||
|
minimum: "Minimum video quality (40 fps)",
|
||||||
|
small: "Small video quality (20 fps)",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
videoQuality: {
|
||||||
|
title: "Video quality",
|
||||||
|
short: {
|
||||||
|
high: "High (30 fps)",
|
||||||
|
medium: "Medium (20 fps)",
|
||||||
|
minimum: "Minimum (10 fps)",
|
||||||
|
small: "Small (5 fps)",
|
||||||
|
},
|
||||||
|
long: {
|
||||||
|
high: "High video quality (30 fps)",
|
||||||
|
medium: "Medium video quality (20 fps, recommended)",
|
||||||
|
minimum: "Minimum video quality (10 fps)",
|
||||||
|
small: "Small video quality (5 fps)",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
language: {
|
||||||
|
title: "Language",
|
||||||
|
},
|
||||||
|
save: {
|
||||||
|
warning: "(Saving these settings will restart the game)",
|
||||||
|
button: "Save",
|
||||||
|
},
|
||||||
|
fullscreen: "Fullscreen",
|
||||||
|
notifications: "Notifications",
|
||||||
|
cowebsiteTrigger: "Always ask before opening websites and Jitsi Meet rooms",
|
||||||
|
ignoreFollowRequest: "Ignore requests to follow other users",
|
||||||
|
},
|
||||||
|
invite: {
|
||||||
|
description: "Share the link of the room!",
|
||||||
|
copy: "Copy",
|
||||||
|
share: "Share",
|
||||||
|
},
|
||||||
|
globalMessage: {
|
||||||
|
text: "Text",
|
||||||
|
audio: "Audio",
|
||||||
|
warning: "Broadcast to all rooms of the world",
|
||||||
|
enter: "Enter your message here...",
|
||||||
|
send: "Send",
|
||||||
|
},
|
||||||
|
globalAudio: {
|
||||||
|
uploadInfo: "Upload a file",
|
||||||
|
error: "No file selected. You need to upload a file before sending it.",
|
||||||
|
},
|
||||||
|
contact: {
|
||||||
|
gettingStarted: {
|
||||||
|
title: "Getting started",
|
||||||
|
description:
|
||||||
|
"WorkAdventure allows you to create an online space to communicate spontaneously with others. And it all starts with creating your own space. Choose from a large selection of prefabricated maps by our team.",
|
||||||
|
},
|
||||||
|
createMap: {
|
||||||
|
title: "Create your map",
|
||||||
|
description: "You can also create your own custom map by following the step of the documentation.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
about: {
|
||||||
|
mapInfo: "Information on the map",
|
||||||
|
mapLink: "link to this map",
|
||||||
|
copyrights: {
|
||||||
|
map: {
|
||||||
|
title: "Copyrights of the map",
|
||||||
|
empty: "The map creator did not declare a copyright for the map.",
|
||||||
|
},
|
||||||
|
tileset: {
|
||||||
|
title: "Copyrights of the tilesets",
|
||||||
|
empty: "The map creator did not declare a copyright for the tilesets. This doesn't mean that those tilesets have no license.",
|
||||||
|
},
|
||||||
|
audio: {
|
||||||
|
title: "Copyrights of audio files",
|
||||||
|
empty: "The map creator did not declare a copyright for audio files. This doesn't mean that those audio files have no license.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
sub: {
|
||||||
|
profile: "Profile",
|
||||||
|
settings: "Settings",
|
||||||
|
invite: "Invite",
|
||||||
|
credit: "Credit",
|
||||||
|
globalMessages: "Global Messages",
|
||||||
|
contact: "Contact",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default menu;
|
25
front/src/i18n/en-US/report.ts
Normal file
25
front/src/i18n/en-US/report.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const report: BaseTranslation = {
|
||||||
|
block: {
|
||||||
|
title: "Block",
|
||||||
|
content: "Block any communication from and to {userName}. This can be reverted.",
|
||||||
|
unblock: "Unblock this user",
|
||||||
|
block: "Block this user",
|
||||||
|
},
|
||||||
|
title: "Report",
|
||||||
|
content: "Send a report message to the administrators of this room. They may later ban this user.",
|
||||||
|
message: {
|
||||||
|
title: "Your message: ",
|
||||||
|
empty: "Report message cannot to be empty.",
|
||||||
|
},
|
||||||
|
submit: "Report this user",
|
||||||
|
moderate: {
|
||||||
|
title: "Moderate {userName}",
|
||||||
|
block: "Block",
|
||||||
|
report: "Report",
|
||||||
|
noSelect: "ERROR : There is no action selected.",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default report;
|
16
front/src/i18n/en-US/warning.ts
Normal file
16
front/src/i18n/en-US/warning.ts
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const warning: BaseTranslation = {
|
||||||
|
title: "Warning!",
|
||||||
|
content:
|
||||||
|
'This world is close to its limit!. You can upgrade its capacity <a href={upgradeLink} target="_blank">here</a>',
|
||||||
|
limit: "This world is close to its limit!",
|
||||||
|
accessDenied: {
|
||||||
|
camera: "Camera access denied. Click here and check your browser permissions.",
|
||||||
|
screenSharing: "Screen sharing denied. Click here and check your browser permissions.",
|
||||||
|
},
|
||||||
|
importantMessage: "Important message",
|
||||||
|
connectionLost: "Connection lost. Reconnecting...",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default warning;
|
20
front/src/i18n/en-US/woka.ts
Normal file
20
front/src/i18n/en-US/woka.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import type { BaseTranslation } from "../i18n-types";
|
||||||
|
|
||||||
|
const woka: BaseTranslation = {
|
||||||
|
customWoka: {
|
||||||
|
title: "Customize your WOKA",
|
||||||
|
navigation: {
|
||||||
|
return: "Return",
|
||||||
|
back: "Back",
|
||||||
|
finish: "Finish",
|
||||||
|
next: "Next",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
selectWoka: {
|
||||||
|
title: "Select your WOKA",
|
||||||
|
continue: "Continue",
|
||||||
|
customize: "Customize your WOKA",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default woka;
|
11
front/src/i18n/formatters.ts
Normal file
11
front/src/i18n/formatters.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { AsyncFormattersInitializer } from "typesafe-i18n";
|
||||||
|
import type { Locales, Formatters } from "./i18n-types";
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/require-await
|
||||||
|
export const initFormatters: AsyncFormattersInitializer<Locales, Formatters> = async (locale: Locales) => {
|
||||||
|
const formatters: Formatters = {
|
||||||
|
// add your formatter functions here
|
||||||
|
};
|
||||||
|
|
||||||
|
return formatters;
|
||||||
|
};
|
10
front/src/i18n/fr-FR/audio.ts
Normal file
10
front/src/i18n/fr-FR/audio.ts
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const audio: NonNullable<Translation["audio"]> = {
|
||||||
|
manager: {
|
||||||
|
reduce: "réduit dans les conversations",
|
||||||
|
},
|
||||||
|
message: "Message audio",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default audio;
|
22
front/src/i18n/fr-FR/camera.ts
Normal file
22
front/src/i18n/fr-FR/camera.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const camera: NonNullable<Translation["camera"]> = {
|
||||||
|
enable: {
|
||||||
|
title: "Allumez votre caméra et votre microphone",
|
||||||
|
start: "C'est partie!",
|
||||||
|
},
|
||||||
|
help: {
|
||||||
|
title: "Accès à la caméra / au microphone nécessaire",
|
||||||
|
permissionDenied: "Permission refusée",
|
||||||
|
content: "Vous devez autoriser l'accès à la caméra et au microphone dans votre navigateur.",
|
||||||
|
firefoxContent:
|
||||||
|
'Veuillez cocher la case "Se souvenir de cette décision" si vous ne voulez pas que Firefox vous demande sans cesse l\'autorisation.',
|
||||||
|
refresh: "Rafraîchir",
|
||||||
|
continue: "Continuer sans webcam",
|
||||||
|
},
|
||||||
|
my: {
|
||||||
|
silentZone: "Zone silencieuse",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default camera;
|
12
front/src/i18n/fr-FR/chat.ts
Normal file
12
front/src/i18n/fr-FR/chat.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const chat: NonNullable<Translation["chat"]> = {
|
||||||
|
intro: "Voici l'historique de votre chat:",
|
||||||
|
enter: "Entrez votre message...",
|
||||||
|
menu: {
|
||||||
|
visitCard: "Carte de visite",
|
||||||
|
addFriend: "Ajouter un ami",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default chat;
|
11
front/src/i18n/fr-FR/companion.ts
Normal file
11
front/src/i18n/fr-FR/companion.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const companion: NonNullable<Translation["companion"]> = {
|
||||||
|
select: {
|
||||||
|
title: "Sélectionnez votre compagnon",
|
||||||
|
any: "Pas de compagnon",
|
||||||
|
continue: "Continuer",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default companion;
|
22
front/src/i18n/fr-FR/error.ts
Normal file
22
front/src/i18n/fr-FR/error.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const error: NonNullable<Translation["error"]> = {
|
||||||
|
accessLink: {
|
||||||
|
title: "Lien d'accès incorrect",
|
||||||
|
subTitle: "Impossible de trouver la carte. Veuillez vérifier votre lien d'accès.",
|
||||||
|
details:
|
||||||
|
"Si vous souhaitez obtenir de plus amples informations, vous pouvez contacter l'administrateur ou nous contacter à l'adresse suivante: hello@workadventu.re",
|
||||||
|
},
|
||||||
|
connectionRejected: {
|
||||||
|
title: "Connexion rejetée",
|
||||||
|
subTitle: "Vous ne pouvez pas rejoindre le monde. Réessayer plus tard {error}.",
|
||||||
|
details:
|
||||||
|
"Si vous souhaitez obtenir de plus amples informations, vous pouvez contacter l'administrateur ou nous contacter à l'adresse suivante: hello@workadventu.re",
|
||||||
|
},
|
||||||
|
connectionRetry: {
|
||||||
|
unableConnect: "Impossible de se connecter à WorkAdventure. Etes vous connecté à Internet?",
|
||||||
|
},
|
||||||
|
error: "Erreur",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default error;
|
27
front/src/i18n/fr-FR/follow.ts
Normal file
27
front/src/i18n/fr-FR/follow.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const follow: NonNullable<Translation["follow"]> = {
|
||||||
|
interactStatus: {
|
||||||
|
following: "Vous suivez {leader}",
|
||||||
|
waitingFollowers: "En attente de la confirmation des suiveurs",
|
||||||
|
followed: {
|
||||||
|
one: "{follower} vous suit",
|
||||||
|
two: "{firstFollower} et {secondFollower} vous suivent",
|
||||||
|
many: "{followers} et {lastFollower} vous suivent",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
interactMenu: {
|
||||||
|
title: {
|
||||||
|
interact: "Interaction",
|
||||||
|
follow: "Voulez-vous suivre {leader}?",
|
||||||
|
},
|
||||||
|
stop: {
|
||||||
|
leader: "Voulez-vous qu'on arrête de vous suivre?",
|
||||||
|
follower: "Voulez-vous arrêter de suivre {leader}?",
|
||||||
|
},
|
||||||
|
yes: "Oui",
|
||||||
|
no: "Non",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default follow;
|
32
front/src/i18n/fr-FR/index.ts
Normal file
32
front/src/i18n/fr-FR/index.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import en_US from "../en-US";
|
||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
import audio from "./audio";
|
||||||
|
import camera from "./camera";
|
||||||
|
import chat from "./chat";
|
||||||
|
import companion from "./companion";
|
||||||
|
import error from "./error";
|
||||||
|
import follow from "./follow";
|
||||||
|
import login from "./login";
|
||||||
|
import menu from "./menu";
|
||||||
|
import report from "./report";
|
||||||
|
import warning from "./warning";
|
||||||
|
import woka from "./woka";
|
||||||
|
|
||||||
|
const fr_FR: Translation = {
|
||||||
|
...en_US,
|
||||||
|
language: "Français",
|
||||||
|
country: "France",
|
||||||
|
audio,
|
||||||
|
camera,
|
||||||
|
chat,
|
||||||
|
companion,
|
||||||
|
woka,
|
||||||
|
error,
|
||||||
|
follow,
|
||||||
|
login,
|
||||||
|
menu,
|
||||||
|
report,
|
||||||
|
warning,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default fr_FR;
|
14
front/src/i18n/fr-FR/login.ts
Normal file
14
front/src/i18n/fr-FR/login.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const login: NonNullable<Translation["login"]> = {
|
||||||
|
input: {
|
||||||
|
name: {
|
||||||
|
placeholder: "Entrez votre nom",
|
||||||
|
empty: "Le nom est vide",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
terms: 'En continuant, vous acceptez nos <a href="https://workadventu.re/terms-of-use" target="_blank">conditions d\'utilisation</a>, notre <a href="https://workadventu.re/privacy-policy" target="_blank">politique de confidentialité</a> et notre <a href="https://workadventu.re/cookie-policy" target="_blank">politique relative aux cookies</a>.',
|
||||||
|
continue: "Continuer",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default login;
|
124
front/src/i18n/fr-FR/menu.ts
Normal file
124
front/src/i18n/fr-FR/menu.ts
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const menu: NonNullable<Translation["menu"]> = {
|
||||||
|
title: "Menu",
|
||||||
|
icon: {
|
||||||
|
open: {
|
||||||
|
menu: "Ouvrir le menu",
|
||||||
|
invite: "Afficher l'invitation",
|
||||||
|
register: "Enregistrez vous",
|
||||||
|
chat: "Ouvrir le chat",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
visitCard: {
|
||||||
|
close: "Fermer",
|
||||||
|
},
|
||||||
|
profile: {
|
||||||
|
edit: {
|
||||||
|
name: "Modifier votre nom",
|
||||||
|
woka: "Modifier votre WOKA",
|
||||||
|
companion: "Modifier votre compagnon",
|
||||||
|
camera: "Modifier votre caméra",
|
||||||
|
},
|
||||||
|
login: "S'identifier",
|
||||||
|
logout: "Déconnexion",
|
||||||
|
},
|
||||||
|
settings: {
|
||||||
|
gameQuality: {
|
||||||
|
title: "Qualité du jeu",
|
||||||
|
short: {
|
||||||
|
high: "Haute (120 fps)",
|
||||||
|
medium: "Moyenne (60 fps)",
|
||||||
|
minimum: "Minimale (40 fps)",
|
||||||
|
small: "Reduite (20 fps)",
|
||||||
|
},
|
||||||
|
long: {
|
||||||
|
high: "Haute (120 fps)",
|
||||||
|
medium: "Moyenne (60 fps, recommandée)",
|
||||||
|
minimum: "Minimale (40 fps)",
|
||||||
|
small: "Reduite (20 fps)",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
videoQuality: {
|
||||||
|
title: "Qualité de la vidéo",
|
||||||
|
short: {
|
||||||
|
high: "Haute (30 fps)",
|
||||||
|
medium: "Moyenne (20 fps)",
|
||||||
|
minimum: "Minimale (10 fps)",
|
||||||
|
small: "Reduite (5 fps)",
|
||||||
|
},
|
||||||
|
long: {
|
||||||
|
high: "Haute (30 fps)",
|
||||||
|
medium: "Moyenne (20 fps, recommandée)",
|
||||||
|
minimum: "Minimale (10 fps)",
|
||||||
|
small: "Reduite (5 fps)",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
language: {
|
||||||
|
title: "Langage",
|
||||||
|
},
|
||||||
|
save: {
|
||||||
|
warning: "(La sauvegarde de ces paramètres redémarre le jeu)",
|
||||||
|
button: "Sauvegarder",
|
||||||
|
},
|
||||||
|
fullscreen: "Plein écran",
|
||||||
|
notifications: "Notifications",
|
||||||
|
cowebsiteTrigger: "Demander toujours avant d'ouvrir des sites web et des salles de réunion Jitsi",
|
||||||
|
ignoreFollowRequest: "Ignorer les demandes de suivi des autres utilisateurs",
|
||||||
|
},
|
||||||
|
invite: {
|
||||||
|
description: "Partager le lien de la salle!",
|
||||||
|
copy: "Copier",
|
||||||
|
share: "Partager",
|
||||||
|
},
|
||||||
|
globalMessage: {
|
||||||
|
text: "Texte",
|
||||||
|
audio: "Audio",
|
||||||
|
warning: "Diffusion dans toutes les salles du monde",
|
||||||
|
enter: "Entrez votre message ici...",
|
||||||
|
send: "Envoyer",
|
||||||
|
},
|
||||||
|
globalAudio: {
|
||||||
|
uploadInfo: "Télécharger un fichier",
|
||||||
|
error: "Aucun fichier sélectionné. Vous devez télécharger un fichier avant de l'envoyer.",
|
||||||
|
},
|
||||||
|
contact: {
|
||||||
|
gettingStarted: {
|
||||||
|
title: "Pour commencer",
|
||||||
|
description:
|
||||||
|
"WorkAdventure vous permet de créer un espace en ligne pour communiquer spontanément avec d'autres personnes. Et tout commence par la création de votre propre espace. Choisissez parmi une large sélection de cartes préfabriquées par notre équipe.",
|
||||||
|
},
|
||||||
|
createMap: {
|
||||||
|
title: "Créer votre carte",
|
||||||
|
description: "Vous pouvez également créer votre propre carte personnalisée en suivant la documentation.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
about: {
|
||||||
|
mapInfo: "Informations sur la carte",
|
||||||
|
mapLink: "lien vers cette carte",
|
||||||
|
copyrights: {
|
||||||
|
map: {
|
||||||
|
title: "Droits d'auteur de la carte",
|
||||||
|
empty: "Le créateur de la carte n'a pas déclaré de droit d'auteur pour la carte.",
|
||||||
|
},
|
||||||
|
tileset: {
|
||||||
|
title: "Droits d'auteur des tilesets",
|
||||||
|
empty: "Le créateur de la carte n'a pas déclaré de droit d'auteur pour les tilesets. Cela ne signifie pas que les tilesets n'ont pas de licence.",
|
||||||
|
},
|
||||||
|
audio: {
|
||||||
|
title: "Droits d'auteur des fichiers audio",
|
||||||
|
empty: "aré de droit d'auteur pour les fichiers audio. Cela ne signifie pas que les fichiers audio n'ont pas de licence.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
sub: {
|
||||||
|
profile: "Profile",
|
||||||
|
settings: "Paramètres",
|
||||||
|
invite: "Inviter",
|
||||||
|
credit: "Crédits",
|
||||||
|
globalMessages: "Messages globaux",
|
||||||
|
contact: "Contact",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default menu;
|
25
front/src/i18n/fr-FR/report.ts
Normal file
25
front/src/i18n/fr-FR/report.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const report: NonNullable<Translation["report"]> = {
|
||||||
|
block: {
|
||||||
|
title: "Bloquer",
|
||||||
|
content: "Bloquer toute communication en provenance et à destination de {userName}. Cela peut être annulé.",
|
||||||
|
unblock: "Débloquer cet utilisateur",
|
||||||
|
block: "Bloquer cet utilisateur",
|
||||||
|
},
|
||||||
|
title: "Signaler",
|
||||||
|
content: "Signaler aux administrateurs de cette salle. Ils pourront par la suite bannir cet utilisateur.",
|
||||||
|
message: {
|
||||||
|
title: "Votre message: ",
|
||||||
|
empty: "Le message du signalement ne peut pas être vide.",
|
||||||
|
},
|
||||||
|
submit: "Signaler cet utilisateur",
|
||||||
|
moderate: {
|
||||||
|
title: "Modérer {userName}",
|
||||||
|
block: "Bloquer",
|
||||||
|
report: "Signaler",
|
||||||
|
noSelect: "ERREUR : Il n'y a pas d'action sélectionnée.",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default report;
|
16
front/src/i18n/fr-FR/warning.ts
Normal file
16
front/src/i18n/fr-FR/warning.ts
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const warning: NonNullable<Translation["warning"]> = {
|
||||||
|
title: "Attention!",
|
||||||
|
content:
|
||||||
|
'Ce monde est proche de sa limite ! Vous pouvez améliorer sa capacité <a href={upgradeLink} target="_blank">içi</a>',
|
||||||
|
limit: "Ce monde est proche de ses limites!",
|
||||||
|
accessDenied: {
|
||||||
|
camera: "Accès à la caméra refusé. Cliquez ici et vérifiez les autorisations de votre navigateur.",
|
||||||
|
screenSharing: "Partage d'écran refusé. Cliquez ici et vérifiez les autorisations de votre navigateur.",
|
||||||
|
},
|
||||||
|
importantMessage: "Message important",
|
||||||
|
connectionLost: "Connexion perdue. Reconnexion...",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default warning;
|
20
front/src/i18n/fr-FR/woka.ts
Normal file
20
front/src/i18n/fr-FR/woka.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import type { Translation } from "../i18n-types";
|
||||||
|
|
||||||
|
const woka: NonNullable<Translation["woka"]> = {
|
||||||
|
customWoka: {
|
||||||
|
title: "Personnalisez votre WOKA",
|
||||||
|
navigation: {
|
||||||
|
return: "Retour",
|
||||||
|
back: "Précédent",
|
||||||
|
finish: "Terminer",
|
||||||
|
next: "Suivant",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
selectWoka: {
|
||||||
|
title: "Sélectionnez votre WOKA",
|
||||||
|
continue: "Continuer",
|
||||||
|
customize: "Personnalisez votre WOKA",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default woka;
|
52
front/src/i18n/locales.ts
Normal file
52
front/src/i18n/locales.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import { detectLocale, navigatorDetector, initLocalStorageDetector } from "typesafe-i18n/detectors";
|
||||||
|
import { FALLBACK_LOCALE } from "../Enum/EnvironmentVariable";
|
||||||
|
import { initI18n, setLocale } from "./i18n-svelte";
|
||||||
|
import type { Locales, Translation } from "./i18n-types";
|
||||||
|
import { baseLocale, getTranslationForLocale, locales } from "./i18n-util";
|
||||||
|
|
||||||
|
const fallbackLocale = FALLBACK_LOCALE || baseLocale;
|
||||||
|
const localStorageProperty = "language";
|
||||||
|
|
||||||
|
export const localeDetector = async () => {
|
||||||
|
const exist = localStorage.getItem(localStorageProperty);
|
||||||
|
let detectedLocale: Locales = fallbackLocale as Locales;
|
||||||
|
|
||||||
|
if (exist) {
|
||||||
|
const localStorageDetector = initLocalStorageDetector(localStorageProperty);
|
||||||
|
detectedLocale = detectLocale(fallbackLocale, locales, localStorageDetector) as Locales;
|
||||||
|
} else {
|
||||||
|
detectedLocale = detectLocale(fallbackLocale, locales, navigatorDetector) as Locales;
|
||||||
|
}
|
||||||
|
|
||||||
|
await initI18n(detectedLocale);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setCurrentLocale = (locale: Locales) => {
|
||||||
|
localStorage.setItem(localStorageProperty, locale);
|
||||||
|
setLocale(locale).catch(() => {
|
||||||
|
console.log("Cannot reload the locale!");
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DisplayableLocale = { id: Locales; language: string; country: string };
|
||||||
|
|
||||||
|
function getDisplayableLocales() {
|
||||||
|
const localesObject: DisplayableLocale[] = [];
|
||||||
|
locales.forEach((locale) => {
|
||||||
|
getTranslationForLocale(locale)
|
||||||
|
.then((translations) => {
|
||||||
|
localesObject.push({
|
||||||
|
id: locale,
|
||||||
|
language: translations.language,
|
||||||
|
country: translations.country,
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.log(error);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return localesObject;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const displayableLocales = getDisplayableLocales();
|
@ -15,7 +15,7 @@ import ui from "./Api/iframe/ui";
|
|||||||
import sound from "./Api/iframe/sound";
|
import sound from "./Api/iframe/sound";
|
||||||
import room, { setMapURL, setRoomId } from "./Api/iframe/room";
|
import room, { setMapURL, setRoomId } from "./Api/iframe/room";
|
||||||
import { createState } from "./Api/iframe/state";
|
import { createState } from "./Api/iframe/state";
|
||||||
import player, { setPlayerName, setTags, setUserRoomToken, setUuid } from "./Api/iframe/player";
|
import player, { setPlayerName, setPlayerLanguage, setTags, setUserRoomToken, setUuid } from "./Api/iframe/player";
|
||||||
import type { ButtonDescriptor } from "./Api/iframe/Ui/ButtonDescriptor";
|
import type { ButtonDescriptor } from "./Api/iframe/Ui/ButtonDescriptor";
|
||||||
import type { Popup } from "./Api/iframe/Ui/Popup";
|
import type { Popup } from "./Api/iframe/Ui/Popup";
|
||||||
import type { Sound } from "./Api/iframe/Sound/Sound";
|
import type { Sound } from "./Api/iframe/Sound/Sound";
|
||||||
@ -30,6 +30,7 @@ const initPromise = queryWorkadventure({
|
|||||||
data: undefined,
|
data: undefined,
|
||||||
}).then((gameState) => {
|
}).then((gameState) => {
|
||||||
setPlayerName(gameState.nickname);
|
setPlayerName(gameState.nickname);
|
||||||
|
setPlayerLanguage(gameState.language);
|
||||||
setRoomId(gameState.roomId);
|
setRoomId(gameState.roomId);
|
||||||
setMapURL(gameState.mapUrl);
|
setMapURL(gameState.mapUrl);
|
||||||
setTags(gameState.tags);
|
setTags(gameState.tags);
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import type { Configuration } from "webpack";
|
import ForkTsCheckerWebpackPlugin from "fork-ts-checker-webpack-plugin";
|
||||||
import type WebpackDevServer from "webpack-dev-server";
|
|
||||||
import path from "path";
|
|
||||||
import webpack from "webpack";
|
|
||||||
import HtmlWebpackPlugin from "html-webpack-plugin";
|
import HtmlWebpackPlugin from "html-webpack-plugin";
|
||||||
import MiniCssExtractPlugin from "mini-css-extract-plugin";
|
import MiniCssExtractPlugin from "mini-css-extract-plugin";
|
||||||
import sveltePreprocess from "svelte-preprocess";
|
|
||||||
import ForkTsCheckerWebpackPlugin from "fork-ts-checker-webpack-plugin";
|
|
||||||
import NodePolyfillPlugin from "node-polyfill-webpack-plugin";
|
import NodePolyfillPlugin from "node-polyfill-webpack-plugin";
|
||||||
|
import path from "path";
|
||||||
|
import sveltePreprocess from "svelte-preprocess";
|
||||||
|
import type { Configuration } from "webpack";
|
||||||
|
import webpack from "webpack";
|
||||||
|
import type WebpackDevServer from "webpack-dev-server";
|
||||||
|
|
||||||
const mode = process.env.NODE_ENV ?? "development";
|
const mode = process.env.NODE_ENV ?? "development";
|
||||||
const buildNpmTypingsForApi = !!process.env.BUILD_TYPINGS;
|
const buildNpmTypingsForApi = !!process.env.BUILD_TYPINGS;
|
||||||
@ -141,6 +141,11 @@ module.exports = {
|
|||||||
filename: "fonts/[name][ext]",
|
filename: "fonts/[name][ext]",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
test: /\.json$/,
|
||||||
|
exclude: /node_modules/,
|
||||||
|
type: "asset",
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
resolve: {
|
resolve: {
|
||||||
@ -210,6 +215,7 @@ module.exports = {
|
|||||||
NODE_ENV: mode,
|
NODE_ENV: mode,
|
||||||
DISABLE_ANONYMOUS: false,
|
DISABLE_ANONYMOUS: false,
|
||||||
OPID_LOGIN_SCREEN_PROVIDER: null,
|
OPID_LOGIN_SCREEN_PROVIDER: null,
|
||||||
|
FALLBACK_LANGUAGE: null,
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
} as Configuration & WebpackDevServer.Configuration;
|
} as Configuration & WebpackDevServer.Configuration;
|
||||||
|
@ -6044,6 +6044,11 @@ type-is@~1.6.17, type-is@~1.6.18:
|
|||||||
media-typer "0.3.0"
|
media-typer "0.3.0"
|
||||||
mime-types "~2.1.24"
|
mime-types "~2.1.24"
|
||||||
|
|
||||||
|
typesafe-i18n@^2.59.0:
|
||||||
|
version "2.59.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/typesafe-i18n/-/typesafe-i18n-2.59.0.tgz#09a9a32e61711418d927a389fa52e1c06a5fa5c4"
|
||||||
|
integrity sha512-Qv3Mrwmb8b73VNzQDPHPECzwymdBRVyDiZ3w2qnp4c2iv/7TGuiJegNHT/l3MooEN7IPbSpc5tbXw2x3MbGtFg==
|
||||||
|
|
||||||
typescript@*:
|
typescript@*:
|
||||||
version "4.3.2"
|
version "4.3.2"
|
||||||
resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.3.2.tgz#399ab18aac45802d6f2498de5054fcbbe716a805"
|
resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.3.2.tgz#399ab18aac45802d6f2498de5054fcbbe716a805"
|
||||||
@ -6534,3 +6539,8 @@ yocto-queue@^0.1.0:
|
|||||||
version "0.1.0"
|
version "0.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
|
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
|
||||||
integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
|
integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
|
||||||
|
|
||||||
|
zod@^3.11.6:
|
||||||
|
version "3.11.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/zod/-/zod-3.11.6.tgz#e43a5e0c213ae2e02aefe7cb2b1a6fa3d7f1f483"
|
||||||
|
integrity sha512-daZ80A81I3/9lIydI44motWe6n59kRBfNzTuS2bfzVh1nAXi667TOTWWtatxyG+fwgNUiagSj/CWZwRRbevJIg==
|
||||||
|
@ -3,7 +3,7 @@ import {assertLogMessage} from "./utils/log";
|
|||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const Docker = require('dockerode');
|
const Docker = require('dockerode');
|
||||||
import { Selector } from 'testcafe';
|
import { Selector } from 'testcafe';
|
||||||
import {login} from "./utils/roles";
|
import {login, resetLanguage} from "./utils/roles";
|
||||||
import {findContainer, rebootBack, rebootPusher, resetRedis, startContainer, stopContainer} from "./utils/containers";
|
import {findContainer, rebootBack, rebootPusher, resetRedis, startContainer, stopContainer} from "./utils/containers";
|
||||||
|
|
||||||
fixture `Reconnection`
|
fixture `Reconnection`
|
||||||
@ -16,6 +16,8 @@ test("Test that connection can succeed even if WorkAdventure starts while pusher
|
|||||||
|
|
||||||
const errorMessage = Selector('.error-div');
|
const errorMessage = Selector('.error-div');
|
||||||
|
|
||||||
|
await resetLanguage('en-US');
|
||||||
|
|
||||||
await t
|
await t
|
||||||
.navigateTo('http://play.workadventure.localhost/_/global/maps.workadventure.localhost/tests/mousewheel.json')
|
.navigateTo('http://play.workadventure.localhost/_/global/maps.workadventure.localhost/tests/mousewheel.json')
|
||||||
.expect(errorMessage.innerText).contains('Unable to connect to WorkAdventure')
|
.expect(errorMessage.innerText).contains('Unable to connect to WorkAdventure')
|
||||||
|
34
tests/tests/translate.ts
Normal file
34
tests/tests/translate.ts
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
import { Selector } from "testcafe";
|
||||||
|
import { login } from "./utils/roles";
|
||||||
|
|
||||||
|
fixture`Translation`
|
||||||
|
.page`http://play.workadventure.localhost/_/global/maps.workadventure.localhost/tests/mousewheel.json`;
|
||||||
|
|
||||||
|
test("Test that I can switch to French", async (t: TestController) => {
|
||||||
|
const languageSelect = Selector(".languages-switcher");
|
||||||
|
const languageOption = languageSelect.find("option");
|
||||||
|
|
||||||
|
await login(
|
||||||
|
t,
|
||||||
|
"http://play.workadventure.localhost/_/global/maps.workadventure.localhost/tests/mousewheel.json"
|
||||||
|
);
|
||||||
|
|
||||||
|
await t
|
||||||
|
.click(".menuIcon img:first-child")
|
||||||
|
.click(Selector("button").withText("Settings"))
|
||||||
|
.click(".languages-switcher")
|
||||||
|
.click(languageOption.withText("Français (France)"))
|
||||||
|
.click(Selector("button").withText("Save"))
|
||||||
|
.wait(5000)
|
||||||
|
|
||||||
|
.click(".menuIcon img:first-child")
|
||||||
|
.expect(Selector("button").withText("Paramètres").innerText)
|
||||||
|
.contains("Paramètres");
|
||||||
|
|
||||||
|
t.ctx.passed = true;
|
||||||
|
}).after(async (t) => {
|
||||||
|
if (!t.ctx.passed) {
|
||||||
|
console.log("Test failed. Browser logs:");
|
||||||
|
console.log(await t.getBrowserConsoleMessages());
|
||||||
|
}
|
||||||
|
});
|
@ -1,6 +1,11 @@
|
|||||||
import { Role } from 'testcafe';
|
import { Role, ClientFunction } from 'testcafe';
|
||||||
|
|
||||||
|
export const resetLanguage = ClientFunction((browserLanguage) => window.localStorage.setItem('language', browserLanguage));
|
||||||
|
|
||||||
|
export async function login(t: TestController, url: string, userName: string = "Alice", characterNumber: number = 2, browserLanguage: string|null = 'en-US') {
|
||||||
|
|
||||||
|
await resetLanguage(browserLanguage);
|
||||||
|
|
||||||
export function login(t: TestController, url: string, userName: string = "Alice", characterNumber: number = 2) {
|
|
||||||
t = t
|
t = t
|
||||||
.navigateTo(url)
|
.navigateTo(url)
|
||||||
.typeText('input[name="loginSceneName"]', userName)
|
.typeText('input[name="loginSceneName"]', userName)
|
||||||
|
Loading…
Reference in New Issue
Block a user