Skip to content

Commit fed4162

Browse files
committed
lib: add navigator.language & navigator.languages
1 parent 14af167 commit fed4162

File tree

3 files changed

+111
-1
lines changed

3 files changed

+111
-1
lines changed

doc/api/globals.md

+44
Original file line numberDiff line numberDiff line change
@@ -637,6 +637,50 @@ logical processors available to the current Node.js instance.
637637
console.log(`This process is running on ${navigator.hardwareConcurrency} logical processors`);
638638
```
639639

640+
### `navigator.language`
641+
642+
<!-- YAML
643+
added: REPLACEME
644+
-->
645+
646+
* {string}
647+
648+
The `navigator.language` read-only property returns a string representing the
649+
preferred language of the Node.js instance.
650+
651+
The value is representing the language version as defined in RFC <5646>.
652+
653+
You can override the value of `navigator.language` by setting the
654+
`LC_ALL` environment variable.
655+
656+
For example:
657+
```sh
658+
LC_ALL=fr-FR node -p "console.log(navigator.language)"
659+
```
660+
661+
The fallback value on builds without ICU is `'en-US'`.
662+
663+
```js
664+
console.log(`The preferred language of the Node.js instance has the tag '${navigator.language}'`);
665+
```
666+
667+
### `navigator.languages`
668+
669+
<!-- YAML
670+
added: REPLACEME
671+
-->
672+
673+
* {Array<string>}
674+
675+
The `navigator.languages` read-only property returns an array of strings
676+
representing the preferred languages of the Node.js instance.
677+
By default it contains only the value of `navigator.language`.
678+
The fallback value on builds without ICU is `['en-US']`.
679+
680+
```js
681+
console.log(`The preferred languages are '${navigator.language}'`);
682+
```
683+
640684
### `navigator.userAgent`
641685

642686
<!-- YAML

lib/internal/navigator.js

+24
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,15 @@
22

33
const {
44
ObjectDefineProperties,
5+
ObjectFreeze,
6+
globalThis,
57
Symbol,
68
} = primordials;
79

10+
const {
11+
Intl,
12+
} = globalThis;
13+
814
const {
915
ERR_ILLEGAL_CONSTRUCTOR,
1016
} = require('internal/errors').codes;
@@ -24,6 +30,8 @@ class Navigator {
2430
// Private properties are used to avoid brand validations.
2531
#availableParallelism;
2632
#userAgent = `Node.js/${nodeVersion.slice(1, nodeVersion.indexOf('.'))}`;
33+
#language = Intl?.Collator().resolvedOptions().locale || 'en-US';
34+
#languages = ObjectFreeze([this.#language]);
2735

2836
constructor() {
2937
if (arguments[0] === kInitialize) {
@@ -40,6 +48,20 @@ class Navigator {
4048
return this.#availableParallelism;
4149
}
4250

51+
/**
52+
* @return {string}
53+
*/
54+
get language() {
55+
return this.#language;
56+
}
57+
58+
/**
59+
* @return {Array<string>}
60+
*/
61+
get languages() {
62+
return this.#languages;
63+
}
64+
4365
/**
4466
* @return {string}
4567
*/
@@ -50,6 +72,8 @@ class Navigator {
5072

5173
ObjectDefineProperties(Navigator.prototype, {
5274
hardwareConcurrency: kEnumerableProperty,
75+
language: kEnumerableProperty,
76+
languages: kEnumerableProperty,
5377
userAgent: kEnumerableProperty,
5478
});
5579

test/parallel/test-navigator.js

+43-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
'use strict';
22

3-
require('../common');
3+
const common = require('../common');
44
const assert = require('assert');
5+
const { execFile } = require('child_process');
56

67
const is = {
78
number: (value, key) => {
@@ -15,3 +16,44 @@ is.number(navigator.hardwareConcurrency, 'hardwareConcurrency');
1516
assert.ok(navigator.hardwareConcurrency > 0);
1617
assert.strictEqual(typeof navigator.userAgent, 'string');
1718
assert.match(navigator.userAgent, /^Node\.js\/\d+$/);
19+
20+
assert.strictEqual(typeof navigator.language, 'string');
21+
assert.strictEqual(navigator.language, 'en-US');
22+
23+
assert.ok(Array.isArray(navigator.languages));
24+
assert.strictEqual(navigator.languages.length, 1);
25+
assert.strictEqual(typeof navigator.languages[0], 'string');
26+
27+
assert.throws(() => {
28+
navigator.languages[0] = 'foo';
29+
}, new TypeError("Cannot assign to read only property '0' of object '[object Array]'"));
30+
assert.notStrictEqual(navigator.languages[0], 'foo');
31+
assert.strictEqual(navigator.languages[0], 'en-US');
32+
33+
Object.defineProperty(navigator, 'language', { value: 'de-DE' });
34+
assert.strictEqual(navigator.language, 'de-DE');
35+
assert.strictEqual(navigator.languages.length, 1);
36+
assert.strictEqual(navigator.languages[0], 'en-US');
37+
38+
39+
if (common.hasIntl) {
40+
{
41+
const env = { ...process.env, LC_ALL: 'fr-FR' };
42+
execFile(
43+
process.execPath,
44+
['-p', 'process.exit(navigator.language === "fr-FR" ? 0 : 1)'],
45+
{ env },
46+
common.mustSucceed()
47+
);
48+
}
49+
50+
{
51+
const env = { ...process.env, LC_ALL: 'fr-FR' };
52+
execFile(
53+
process.execPath,
54+
['-p', 'process.exit(navigator.languages[0] === "fr-FR" ? 0 : 1)'],
55+
{ env },
56+
common.mustSucceed()
57+
);
58+
}
59+
}

0 commit comments

Comments
 (0)