From 2b17a2e812e05443f740bcabf7db1291337492a7 Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 16 Apr 2021 22:54:46 -0400 Subject: [PATCH 001/278] Upgrade chokidar to support M1 macs --- package-lock.json | 29 +++++++++++------------------ package.json | 4 ++-- 2 files changed, 13 insertions(+), 20 deletions(-) diff --git a/package-lock.json b/package-lock.json index 5eafa5ff8..ae09a7799 100644 --- a/package-lock.json +++ b/package-lock.json @@ -536,13 +536,12 @@ "dev": true }, "@types/chokidar": { - "version": "1.7.5", - "resolved": "https://registry.npmjs.org/@types/chokidar/-/chokidar-1.7.5.tgz", - "integrity": "sha512-PDkSRY7KltW3M60hSBlerxI8SFPXsO3AL/aRVsO4Kh9IHRW74Ih75gUuTd/aE4LSSFqypb10UIX3QzOJwBQMGQ==", + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/@types/chokidar/-/chokidar-2.1.3.tgz", + "integrity": "sha512-6qK3xoLLAhQVTucQGHTySwOVA1crHRXnJeLwqK6KIFkkKa2aoMFXh+WEi8PotxDtvN6MQJLyYN9ag9P6NLV81w==", "dev": true, "requires": { - "@types/events": "*", - "@types/node": "*" + "chokidar": "*" } }, "@types/command-line-args": { @@ -572,12 +571,6 @@ "integrity": "sha512-mjcCf//DAUQ6YLQMhqYJAv/+a4BsE1GQFmy1el5K62wLJJmQwGi3TsnshhOFynPpuBF9Gh2Vvb+5ImPi47KaZw==", "dev": true }, - "@types/events": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz", - "integrity": "sha512-EaObqwIvayI5a8dCzhFrjKzVwKLxjoG9T6Ppd5CEo07LRKfQ8Yokw54r5+Wq7FaBQ+yXRvQAYPrHwya1/UFt9g==", - "dev": true - }, "@types/form-data": { "version": "0.0.33", "resolved": "https://registry.npmjs.org/@types/form-data/-/form-data-0.0.33.tgz", @@ -1326,13 +1319,13 @@ } }, "chokidar": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz", - "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==", + "version": "3.5.1", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.1.tgz", + "integrity": "sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw==", "requires": { "anymatch": "~3.1.1", "braces": "~3.0.2", - "fsevents": "~2.1.2", + "fsevents": "~2.3.1", "glob-parent": "~5.1.0", "is-binary-path": "~2.1.0", "is-glob": "~4.0.1", @@ -2740,9 +2733,9 @@ "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" }, "fsevents": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", - "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", "optional": true }, "function-bind": { diff --git a/package.json b/package.json index 499f212c1..2f3731b6a 100644 --- a/package.json +++ b/package.json @@ -71,7 +71,7 @@ "devDependencies": { "@types/benchmark": "^1.0.31", "@types/chai": "^4.1.2", - "@types/chokidar": "^1.7.5", + "@types/chokidar": "^2.1.3", "@types/command-line-args": "^5.0.0", "@types/command-line-usage": "^5.0.1", "@types/debounce-promise": "^3.1.1", @@ -113,7 +113,7 @@ "array-flat-polyfill": "^1.0.1", "chalk": "^2.4.2", "chevrotain": "^7.0.1", - "chokidar": "^3.0.2", + "chokidar": "^3.5.1", "clear": "^0.1.0", "cross-platform-clear-console": "^2.3.0", "debounce-promise": "^3.1.0", From cf19c70b86f630184c2163b81b57cdbed3f3864e Mon Sep 17 00:00:00 2001 From: George Cook Date: Tue, 4 May 2021 04:47:52 +0200 Subject: [PATCH 002/278] adds warning for mismatched class member accessibility (#402) * adds warning for mismatched class member accessibility * updates overridden accessType message, as requested in pr review --- src/DiagnosticMessages.ts | 5 +++ src/files/BrsFile.Class.spec.ts | 53 ++++++++++++++++++++++++++++++++ src/validators/ClassValidator.ts | 20 ++++++++++++ 3 files changed, 78 insertions(+) diff --git a/src/DiagnosticMessages.ts b/src/DiagnosticMessages.ts index b9f5703e9..542b624b7 100644 --- a/src/DiagnosticMessages.ts +++ b/src/DiagnosticMessages.ts @@ -625,6 +625,11 @@ export let DiagnosticMessages = { message: `Missing expression(s) in 'dim' statement`, code: 1121, severity: DiagnosticSeverity.Error + }), + mismatchedOverriddenMemberVisibility: (childClassName: string, memberName: string, childAccessModifier: string, ancestorAccessModifier: string, ancestorClassName: string) => ({ + message: `Access modifier mismatch: '${memberName}' is ${childAccessModifier} in type '${childClassName}' but is ${ancestorAccessModifier} in base type '${ancestorClassName}'.`, + code: 1122, + severity: DiagnosticSeverity.Error }) }; diff --git a/src/files/BrsFile.Class.spec.ts b/src/files/BrsFile.Class.spec.ts index fe25e42ec..f5ed1f527 100644 --- a/src/files/BrsFile.Class.spec.ts +++ b/src/files/BrsFile.Class.spec.ts @@ -830,6 +830,59 @@ describe('BrsFile BrighterScript classes', () => { }); }); + it('detects overridden methods with different visibility', () => { + program.addOrReplaceFile({ src: `${rootDir}/source/main.bs`, dest: 'source/main.bs' }, ` + class Animal + sub speakInPublic() + end sub + protected sub speakWithFriends() + end sub + private sub speakWithFamily() + end sub + end class + class Duck extends Animal + private override sub speakInPublic() + end sub + public override sub speakWithFriends() + end sub + override sub speakWithFamily() + end sub + end class + `); + program.validate(); + expect(program.getDiagnostics()[0]).to.exist.and.to.include({ + ...DiagnosticMessages.mismatchedOverriddenMemberVisibility('Duck', 'speakInPublic', 'private', 'public', 'Animal') + }); + expect(program.getDiagnostics()[1]).to.exist.and.to.include({ + ...DiagnosticMessages.mismatchedOverriddenMemberVisibility('Duck', 'speakWithFriends', 'public', 'protected', 'Animal') + }); + expect(program.getDiagnostics()[2]).to.exist.and.to.include({ + ...DiagnosticMessages.mismatchedOverriddenMemberVisibility('Duck', 'speakWithFamily', 'public', 'private', 'Animal') + }); + }); + it('allows overridden methods with matching visibility', () => { + program.addOrReplaceFile({ src: `${rootDir}/source/main.bs`, dest: 'source/main.bs' }, ` + class Animal + sub speakInPublic() + end sub + protected sub speakWithFriends() + end sub + private sub speakWithFamily() + end sub + end class + class Duck extends Animal + override sub speakInPublic() + end sub + protected override sub speakWithFriends() + end sub + private override sub speakWithFamily() + end sub + end class + `); + program.validate(); + expect(program.getDiagnostics()).to.be.empty; + }); + it('detects extending unknown parent class', () => { program.addOrReplaceFile('source/main.brs', ` class Duck extends Animal diff --git a/src/validators/ClassValidator.ts b/src/validators/ClassValidator.ts index f2955c6e3..252adece5 100644 --- a/src/validators/ClassValidator.ts +++ b/src/validators/ClassValidator.ts @@ -10,6 +10,7 @@ import { isCallExpression, isClassFieldStatement, isClassMethodStatement, isCust import type { BscFile, BsDiagnostic } from '../interfaces'; import { createVisitor, WalkMode } from '../astUtils'; import type { BrsFile } from '../files/BrsFile'; +import { TokenKind } from '../lexer'; export class BsClassValidator { private scope: Scope; @@ -223,6 +224,25 @@ export class BsClassValidator { range: member.range }); } + + //child member has different visiblity + if ( + //is a method + isClassMethodStatement(member) && + (member.accessModifier?.kind ?? TokenKind.Public) !== (ancestorAndMember.member.accessModifier?.kind ?? TokenKind.Public) + ) { + this.diagnostics.push({ + ...DiagnosticMessages.mismatchedOverriddenMemberVisibility( + classStatement.name.text, + ancestorAndMember.member.name?.text, + member.accessModifier?.text || 'public', + ancestorAndMember.member.accessModifier?.text || 'public', + ancestorAndMember.classStatement.getName(ParseMode.BrighterScript) + ), + file: classStatement.file, + range: member.range + }); + } } if (isClassMethodStatement(member)) { From 46b2fa175af27db89db2d8cfbad133e8f80bf99a Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Tue, 4 May 2021 06:59:32 -0400 Subject: [PATCH 003/278] Allow field overrides, disallow type changes (#394) --- src/DiagnosticMessages.ts | 9 +++++-- src/Scope.spec.ts | 4 ++-- src/astUtils/reflection.ts | 4 ++++ src/files/BrsFile.Class.spec.ts | 40 ++++++++++++++++++++++++++++---- src/parser/Parser.ts | 2 +- src/parser/Statement.ts | 2 +- src/types/CustomType.ts | 28 +++++++++++++++++++--- src/validators/ClassValidator.ts | 36 +++++++++++++++++----------- 8 files changed, 98 insertions(+), 27 deletions(-) diff --git a/src/DiagnosticMessages.ts b/src/DiagnosticMessages.ts index 542b624b7..07a5b577e 100644 --- a/src/DiagnosticMessages.ts +++ b/src/DiagnosticMessages.ts @@ -511,8 +511,8 @@ export let DiagnosticMessages = { code: 1097, severity: DiagnosticSeverity.Error }), - memberAlreadyExistsInParentClass: (memberType: string, parentClassName: string) => ({ - message: `A ${memberType} with this name already exists in inherited class '${parentClassName}'`, + childFieldTypeNotAssignableToBaseProperty: (childTypeName: string, baseTypeName: string, fieldName: string, childFieldType: string, parentFieldType: string) => ({ + message: `Field '${fieldName}' in class '${childTypeName}' is not assignable to the same field in base class '${baseTypeName}'. Type '${childFieldType}' is not assignable to type '${parentFieldType}'.`, code: 1098, severity: DiagnosticSeverity.Error }), @@ -630,6 +630,11 @@ export let DiagnosticMessages = { message: `Access modifier mismatch: '${memberName}' is ${childAccessModifier} in type '${childClassName}' but is ${ancestorAccessModifier} in base type '${ancestorClassName}'.`, code: 1122, severity: DiagnosticSeverity.Error + }), + cannotFindType: (typeName: string) => ({ + message: `Cannot find type with name '${typeName}'`, + code: 1123, + severity: DiagnosticSeverity.Error }) }; diff --git a/src/Scope.spec.ts b/src/Scope.spec.ts index 7bac5bab6..0b3ae056d 100644 --- a/src/Scope.spec.ts +++ b/src/Scope.spec.ts @@ -599,8 +599,8 @@ describe('Scope', () => { `); program.validate(); expect(program.getDiagnostics().map(x => x.message)).to.eql([ - DiagnosticMessages.expectedValidTypeToFollowAsKeyword().message, - DiagnosticMessages.expectedValidTypeToFollowAsKeyword().message + DiagnosticMessages.cannotFindType('unknownType').message, + DiagnosticMessages.cannotFindType('unknownType').message ]); }); diff --git a/src/astUtils/reflection.ts b/src/astUtils/reflection.ts index f099fa566..d04d4591e 100644 --- a/src/astUtils/reflection.ts +++ b/src/astUtils/reflection.ts @@ -16,6 +16,7 @@ import { DoubleType } from '../types/DoubleType'; import { CustomType } from '../types/CustomType'; import type { Scope } from '../Scope'; import type { XmlScope } from '../XmlScope'; +import { DynamicType } from '../types/DynamicType'; // File reflection @@ -238,6 +239,9 @@ export function isVoidType(e: any): e is VoidType { export function isCustomType(e: any): e is CustomType { return e?.constructor.name === CustomType.name; } +export function isDynamicType(e: any): e is DynamicType { + return e?.constructor.name === DynamicType.name; +} const numberConstructorNames = [ IntegerType.name, diff --git a/src/files/BrsFile.Class.spec.ts b/src/files/BrsFile.Class.spec.ts index f5ed1f527..20b1b1fab 100644 --- a/src/files/BrsFile.Class.spec.ts +++ b/src/files/BrsFile.Class.spec.ts @@ -797,7 +797,7 @@ describe('BrsFile BrighterScript classes', () => { ); }); - it('detects overridden property name in child class', () => { + it('allows untyped overridden field in child class', () => { program.addOrReplaceFile({ src: `${rootDir}/source/main.bs`, dest: 'source/main.bs' }, ` class Animal public name @@ -807,9 +807,41 @@ describe('BrsFile BrighterScript classes', () => { end class `); program.validate(); - let diagnostics = program.getDiagnostics().map(x => x.message); - expect(diagnostics).to.eql([ - DiagnosticMessages.memberAlreadyExistsInParentClass('field', 'Animal').message + expectZeroDiagnostics(program); + }); + + it('allows overridden property name in child class', () => { + program.addOrReplaceFile('source/main.bs', ` + class Bird + public name = "bird" + end class + class Duck extends Bird + public name = "duck" + end class + `); + program.validate(); + expectZeroDiagnostics(program); + }); + + it('flags incompatible child field type changes', () => { + program.addOrReplaceFile('source/main.bs', ` + class Bird + public age = 12 + public name = "bird" + public owner as Person + end class + class Duck extends Bird + public age = 12.2 'should be integer but is float + public name = 12 'should be string but is integer + public owner as string + end class + `); + program.validate(); + expect(program.getDiagnostics().map(x => x.message).sort()).to.eql([ + DiagnosticMessages.cannotFindType('Person').message, + DiagnosticMessages.childFieldTypeNotAssignableToBaseProperty('Duck', 'Bird', 'age', 'float', 'integer').message, + DiagnosticMessages.childFieldTypeNotAssignableToBaseProperty('Duck', 'Bird', 'name', 'integer', 'string').message, + DiagnosticMessages.childFieldTypeNotAssignableToBaseProperty('Duck', 'Bird', 'owner', 'string', 'Person').message ]); }); diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index bef4ee9b4..b9be0de98 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -492,7 +492,7 @@ export class Parser { fieldType = this.typeToken(); //no field type specified - if (!util.tokenToBscType(fieldType) && !this.check(TokenKind.Identifier)) { + if (!util.tokenToBscType(fieldType)) { this.diagnostics.push({ ...DiagnosticMessages.expectedValidTypeToFollowAsKeyword(), range: this.peek().range diff --git a/src/parser/Statement.ts b/src/parser/Statement.ts index 73d091e10..8e389a7d0 100644 --- a/src/parser/Statement.ts +++ b/src/parser/Statement.ts @@ -1757,7 +1757,7 @@ export class ClassFieldStatement extends Statement implements TypedefProvider { /** * Derive a ValueKind from the type token, or the initial value. - * Defaults to `ValueKind.Dynamic` + * Defaults to `DynamicType` */ getType() { if (this.type) { diff --git a/src/types/CustomType.ts b/src/types/CustomType.ts index 5ce447259..9dad1d431 100644 --- a/src/types/CustomType.ts +++ b/src/types/CustomType.ts @@ -1,9 +1,31 @@ +import { isCustomType, isDynamicType } from '../astUtils/reflection'; import type { BscType } from './BscType'; -import { ObjectType } from './ObjectType'; -export class CustomType extends ObjectType implements BscType { +export class CustomType implements BscType { constructor(public name: string) { - super(); + } + + public toString() { + return this.name; + } + + public toTypeString(): string { + return 'object'; + } + + public isAssignableTo(targetType: BscType) { + //TODO for now, if the custom types have the same name, assume they're the same thing + if (isCustomType(targetType) && targetType.name === this.name) { + return true; + } else if (isDynamicType(targetType)) { + return true; + } else { + return false; + } + } + + public isConvertibleTo(targetType: BscType) { + return this.isAssignableTo(targetType); } } diff --git a/src/validators/ClassValidator.ts b/src/validators/ClassValidator.ts index 252adece5..a23c970b5 100644 --- a/src/validators/ClassValidator.ts +++ b/src/validators/ClassValidator.ts @@ -2,7 +2,7 @@ import type { Scope } from '../Scope'; import { DiagnosticMessages } from '../DiagnosticMessages'; import type { CallExpression } from '../parser/Expression'; import { ParseMode } from '../parser/Parser'; -import type { ClassMethodStatement, ClassStatement } from '../parser/Statement'; +import type { ClassFieldStatement, ClassMethodStatement, ClassStatement } from '../parser/Statement'; import { CancellationTokenSource, Location } from 'vscode-languageserver'; import { URI } from 'vscode-uri'; import util from '../util'; @@ -180,14 +180,14 @@ export class BsClassValidator { let memberType = isClassFieldStatement(member) ? 'field' : 'method'; let ancestorAndMember = this.getAncestorMember(classStatement, lowerMemberName); if (ancestorAndMember) { - let ancestorMemberType = isClassFieldStatement(ancestorAndMember.member) ? 'field' : 'method'; + let ancestorMemberKind = isClassFieldStatement(ancestorAndMember.member) ? 'field' : 'method'; - //mismatched member type (field/method in child, opposite in parent) - if (memberType !== ancestorMemberType) { + //mismatched member type (field/method in child, opposite in ancestor) + if (memberType !== ancestorMemberKind) { this.diagnostics.push({ ...DiagnosticMessages.classChildMemberDifferentMemberTypeThanAncestor( memberType, - ancestorMemberType, + ancestorMemberKind, ancestorAndMember.classStatement.getName(ParseMode.BrighterScript) ), file: classStatement.file, @@ -197,14 +197,22 @@ export class BsClassValidator { //child field has same name as parent if (isClassFieldStatement(member)) { - this.diagnostics.push({ - ...DiagnosticMessages.memberAlreadyExistsInParentClass( - memberType, - ancestorAndMember.classStatement.getName(ParseMode.BrighterScript) - ), - file: classStatement.file, - range: member.range - }); + const ancestorFieldType = (ancestorAndMember.member as ClassFieldStatement).getType(); + const childFieldType = member.getType(); + if (!childFieldType.isAssignableTo(ancestorFieldType)) { + //flag incompatible child field type to ancestor field type + this.diagnostics.push({ + ...DiagnosticMessages.childFieldTypeNotAssignableToBaseProperty( + classStatement.getName(ParseMode.BrighterScript), + ancestorAndMember.classStatement.getName(ParseMode.BrighterScript), + member.name.text, + childFieldType.toString(), + ancestorFieldType.toString() + ), + file: classStatement.file, + range: member.range + }); + } } //child method missing the override keyword @@ -275,7 +283,7 @@ export class BsClassValidator { //check if this custom type is in our class map if (!this.getClassByName(lowerFieldTypeName, currentNamespaceName)) { this.diagnostics.push({ - ...DiagnosticMessages.expectedValidTypeToFollowAsKeyword(), + ...DiagnosticMessages.cannotFindType(fieldTypeName), range: statement.type.range, file: classStatement.file }); From 42f0760bab44c61b27be4fec6d4e0450ab74922d Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 4 May 2021 07:01:17 -0400 Subject: [PATCH 004/278] update changelog for v0.38.0 --- CHANGELOG.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 546403bf9..d28007949 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.38.0] - 2021-05-04 +### Adds + - warning for mismatched class method accessibility ([#402](https://github.com/rokucommunity/brighterscript/pull/402)) + - allow class field overrides in child classes as long as they are the same type ([#394](https://github.com/rokucommunity/brighterscript/pull/394)) + + + ## [0.37.4] - 2021-04-20 ### Fixed - bug validating namespace function calls ([#390](https://github.com/rokucommunity/brighterscript/pull/390)) @@ -1091,3 +1098,4 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 [0.37.2]: https://github.com/rokucommunity/brighterscript/compare/v0.37.1...v0.37.2 [0.37.3]: https://github.com/rokucommunity/brighterscript/compare/v0.37.2...v0.37.3 [0.37.4]: https://github.com/rokucommunity/brighterscript/compare/v0.37.3...v0.37.4 +[0.38.0]: https://github.com/rokucommunity/brighterscript/compare/v0.37.4...v0.38.0 From 940a0cdb12a1a8506a0f910f97888827e0663da6 Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 4 May 2021 07:02:51 -0400 Subject: [PATCH 005/278] 0.38.0 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index feff83a97..27cb655ae 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.37.4", + "version": "0.38.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 7e510a713..744723015 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.37.4", + "version": "0.38.0", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 735aa9676670a3cfa65c26573dbd67de72e62147 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 11 May 2021 06:58:40 -0400 Subject: [PATCH 006/278] Bump hosted-git-info from 2.8.8 to 2.8.9 (#411) Bumps [hosted-git-info](https://github.com/npm/hosted-git-info) from 2.8.8 to 2.8.9. - [Release notes](https://github.com/npm/hosted-git-info/releases) - [Changelog](https://github.com/npm/hosted-git-info/blob/v2.8.9/CHANGELOG.md) - [Commits](https://github.com/npm/hosted-git-info/compare/v2.8.8...v2.8.9) Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index 273759bd0..efcbf7856 100644 --- a/package-lock.json +++ b/package-lock.json @@ -2951,9 +2951,9 @@ "dev": true }, "hosted-git-info": { - "version": "2.8.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", - "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", + "version": "2.8.9", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", + "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", "dev": true }, "html-escaper": { From 25b1acd51a83fb1c414815287979ac286a692320 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 11 May 2021 07:10:17 -0400 Subject: [PATCH 007/278] Bump lodash from 4.17.20 to 4.17.21 (#410) Bumps [lodash](https://github.com/lodash/lodash) from 4.17.20 to 4.17.21. - [Release notes](https://github.com/lodash/lodash/releases) - [Commits](https://github.com/lodash/lodash/compare/4.17.20...4.17.21) Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index efcbf7856..1fe6d808a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3535,9 +3535,9 @@ } }, "lodash": { - "version": "4.17.20", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", - "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" }, "lodash.defaults": { "version": "4.2.0", From c4240377ced940752c059a5af3d3f26da6a18f07 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 12 May 2021 06:22:05 -0400 Subject: [PATCH 008/278] Bump postcss from 8.2.8 to 8.2.15 (#413) Bumps [postcss](https://github.com/postcss/postcss) from 8.2.8 to 8.2.15. - [Release notes](https://github.com/postcss/postcss/releases) - [Changelog](https://github.com/postcss/postcss/blob/main/CHANGELOG.md) - [Commits](https://github.com/postcss/postcss/compare/8.2.8...8.2.15) Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/package-lock.json b/package-lock.json index 1fe6d808a..68f0445fa 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4711,16 +4711,22 @@ "dev": true }, "postcss": { - "version": "8.2.8", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.2.8.tgz", - "integrity": "sha512-1F0Xb2T21xET7oQV9eKuctbM9S7BC0fetoHCc4H13z0PT6haiRLP4T0ZY4XWh7iLP0usgqykT6p9B2RtOf4FPw==", + "version": "8.2.15", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.2.15.tgz", + "integrity": "sha512-2zO3b26eJD/8rb106Qu2o7Qgg52ND5HPjcyQiK2B98O388h43A448LCslC0dI2P97wCAQRJsFvwTRcXxTKds+Q==", "dev": true, "requires": { "colorette": "^1.2.2", - "nanoid": "^3.1.20", + "nanoid": "^3.1.23", "source-map": "^0.6.1" }, "dependencies": { + "nanoid": { + "version": "3.1.23", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.23.tgz", + "integrity": "sha512-FiB0kzdP0FFVGDKlRLEQ1BgDzU87dy5NnzjeW9YZNt+/c3+q82EQDUwniSAUxp/F0gFNI1ZhKU1FqYsMuqZVnw==", + "dev": true + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", From 894ace84d5cdee0eb4ab1ed869d0401adf3df62d Mon Sep 17 00:00:00 2001 From: Mark Pearce Date: Fri, 14 May 2021 00:55:06 -0300 Subject: [PATCH 009/278] Fix/bs config schema allows string diagnostic codes (#416) * Allow string diagnostic codes in bsconfig when specifying diagnostic filters by paths * Cleaner representation of actual codes types Co-authored-by: Mark Pearce Co-authored-by: Bronley Plumb --- bsconfig.schema.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bsconfig.schema.json b/bsconfig.schema.json index e57651c54..41f659abe 100644 --- a/bsconfig.schema.json +++ b/bsconfig.schema.json @@ -165,7 +165,7 @@ "items": { "anyOf": [ { - "type": "number", + "type": ["number", "string"], "description": "A code of diagnostics that should be filtered out from the files matched in 'src'" } ] From 3bf228663985be7cd9c6809b5c7e29babdf35e35 Mon Sep 17 00:00:00 2001 From: George Cook Date: Fri, 14 May 2021 12:59:57 +0200 Subject: [PATCH 010/278] concatanates source literal file paths so that the transpiled output does not cause issues with roku's static analysis tool (#415) * concatanates source literal file paths so that the transpiled output does not cause issues with roku's static analysis tool * minor docs fix Co-authored-by: Bronley Plumb --- docs/source-literals.md | 9 +++++++-- src/parser/Expression.ts | 6 ++++-- .../tests/expression/SourceLiteralExpression.spec.ts | 8 ++++---- 3 files changed, 15 insertions(+), 8 deletions(-) diff --git a/docs/source-literals.md b/docs/source-literals.md index b0b41e225..ff40c0fcc 100644 --- a/docs/source-literals.md +++ b/docs/source-literals.md @@ -14,9 +14,11 @@ print SOURCE_FILE_PATH transpiles to: ```BrightScript -print "file:///c:/projects/roku/brighterscript/scripts/rootDir/source/main.bs" +print "file" + ":///c:/projects/roku/brighterscript/scripts/rootDir/source/main.bs" ``` +_note: the literal is concatenated to keep the roku static analysis tool happy_ + ## SOURCE_LINE_NUM The 1-based line number of the source file. @@ -153,10 +155,13 @@ transpiles to: ```BrightScript function main() - print "file:///c:/projects/roku/brighterscript/scripts/rootDir/source/main.bs:2" + print "file" + ":///c:/projects/roku/brighterscript/scripts/rootDir/source/main.bs:2" end function ``` +_note: the literal is concatenated to keep the roku static analysis tool happy_ + + ## PKG_PATH The pkg path of the file. diff --git a/src/parser/Expression.ts b/src/parser/Expression.ts index 5486478b9..07790e7f7 100644 --- a/src/parser/Expression.ts +++ b/src/parser/Expression.ts @@ -844,7 +844,8 @@ export class SourceLiteralExpression extends Expression { let text: string; switch (this.token.kind) { case TokenKind.SourceFilePathLiteral: - text = `"${fileUrl(state.srcPath)}"`; + const pathUrl = fileUrl(state.srcPath); + text = `"${pathUrl.substring(0, 4)}" + "${pathUrl.substring(4)}"`; break; case TokenKind.SourceLineNumLiteral: text = `${this.token.range.start.line + 1}`; @@ -856,7 +857,8 @@ export class SourceLiteralExpression extends Expression { text = `"${this.getFunctionName(state, ParseMode.BrighterScript)}"`; break; case TokenKind.SourceLocationLiteral: - text = `"${fileUrl(state.srcPath)}:${this.token.range.start.line + 1}"`; + const locationUrl = fileUrl(state.srcPath); + text = `"${locationUrl.substring(0, 4)}" + "${locationUrl.substring(4)}:${this.token.range.start.line + 1}"`; break; case TokenKind.PkgPathLiteral: let pkgPath1 = `pkg:/${state.file.pkgPath}` diff --git a/src/parser/tests/expression/SourceLiteralExpression.spec.ts b/src/parser/tests/expression/SourceLiteralExpression.spec.ts index 50b7aef6b..d418021a9 100644 --- a/src/parser/tests/expression/SourceLiteralExpression.spec.ts +++ b/src/parser/tests/expression/SourceLiteralExpression.spec.ts @@ -47,7 +47,7 @@ describe('SourceLiteralExpression', () => { end sub `, ` sub main() - print "${fileUrl(`${rootDir}/source/main.bs`)}" + print "${fileUrl(`${rootDir}/source/main.bs`).substring(0, 4)}" + "${fileUrl(`${rootDir}/source/main.bs`).substring(4)}" end sub `, undefined, 'source/main.bs'); }); @@ -139,7 +139,7 @@ describe('SourceLiteralExpression', () => { end sub `, ` sub main() - print "${fileUrl(`${rootDir}/source/main.bs`)}:3" + print "${fileUrl(`${rootDir}/source/main.bs`).substring(0, 4)}" + "${fileUrl(`${rootDir}/source/main.bs`).substring(4)}:3" end sub `, undefined, 'source/main.bs'); }); @@ -188,7 +188,7 @@ describe('SourceLiteralExpression', () => { end sub `, ` sub main() - print "${fileUrl(s`${sourceRoot}/source/main.bs`)}" + print "${fileUrl(s`${sourceRoot}/source/main.bs`).substring(0, 4)}" + "${fileUrl(s`${sourceRoot}/source/main.bs`).substring(4)}" end sub `, undefined, 'source/main.bs'); }); @@ -205,7 +205,7 @@ describe('SourceLiteralExpression', () => { end sub `, ` sub main() - print "${fileUrl(`${sourceRoot}/source/main.bs`)}:3" + print "${fileUrl(s`${sourceRoot}/source/main.bs`).substring(0, 4)}" + "${fileUrl(s`${sourceRoot}/source/main.bs`).substring(4)}:3" end sub `, undefined, 'source/main.bs'); }); From 4d2eef594bb781128f29e952edcbc6e65d24a210 Mon Sep 17 00:00:00 2001 From: Mark Pearce Date: Fri, 14 May 2021 16:43:59 -0300 Subject: [PATCH 011/278] ParseJson allows two parameters as of Roku OS 9.4 (#418) Co-authored-by: Mark Pearce Co-authored-by: Bronley Plumb --- src/globalCallables.spec.ts | 22 ++++++++++++++++++++++ src/globalCallables.ts | 8 +++++++- 2 files changed, 29 insertions(+), 1 deletion(-) diff --git a/src/globalCallables.spec.ts b/src/globalCallables.spec.ts index 5ae53a38e..f9aeacf30 100644 --- a/src/globalCallables.spec.ts +++ b/src/globalCallables.spec.ts @@ -75,5 +75,27 @@ describe('globalCallables', () => { }); }); + describe('parseJson', () => { + it('allows single parameter', () => { + program.addOrReplaceFile('source/main.brs', ` + sub main() + print ParseJson("{}") + end sub + `); + program.validate(); + expect(program.getDiagnostics()[0]?.message).not.to.exist; + }); + + it('allows 2 parameters', () => { + program.addOrReplaceFile('source/main.brs', ` + sub main() + print ParseJson("{}", "i") + end sub + `); + program.validate(); + expect(program.getDiagnostics()[0]?.message).not.to.exist; + }); + }); + }); diff --git a/src/globalCallables.ts b/src/globalCallables.ts index 2a7f8c2f5..4c98fafb3 100644 --- a/src/globalCallables.ts +++ b/src/globalCallables.ts @@ -466,13 +466,19 @@ The characters '?', '*' and '[' lose their special meaning if preceded by a sing shortDescription: `This function will parse a string formatted according to RFC4627 and return an equivalent BrightScript object (consisting of booleans, integer and floating point numbers, strings, roArray, and roAssociativeArray objects). If the string is not syntactically correct, Invalid will be returned. A few other things to note: -Any roAssociativeArray objects in the returned objects will be case sensitive. +Any roAssociativeArray objects in the returned objects will be case sensitive. As of Roku OS 9.4, to return a case-insensitive structure, set the flags parameter to "i". +If the "i" option is used, and the jsonString includes multiple keys that match case-insensitively, duplicates are overwritten and only the last matching values are preserved. An error will be returned if arrays/associative arrays are nested more than 256 levels deep.`, type: new FunctionType(new ObjectType()), file: globalFile, params: [{ name: 'jsonString', type: new StringType() + }, + { + name: 'flags', + type: new StringType(), + isOptional: true }] }, { name: 'FormatJson', From 6ed13e198e422a33a9831f10f66148f7072a3dde Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 14 May 2021 16:10:45 -0400 Subject: [PATCH 012/278] update changelog for v0.38.1 --- CHANGELOG.md | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d28007949..5d39daff9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,8 +6,20 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.38.1] - 2021-05-14 +[0.38.1]: https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.38.1 +### Changed + - SOURCE_FILE_PATH and SOURCE_LOCATION source literals are now string concatenations in order to avoid triggering Roku's static analysis rule against `file:/` protocol strings ([#415](https://github.com/rokucommunity/brighterscript/pull/415)) +### Fixed + - ParseJson function signature to include second parameter ([#418](https://github.com/rokucommunity/brighterscript/pull/418)) + - bsconfig.schema.json support for string diagnostic codes ([#416](https://github.com/rokucommunity/brighterscript/pull/416)) + - upgrade chokidar to add `--watch` cli support for M1 mac computers ([#386](https://github.com/rokucommunity/brighterscript/pull/386)) + - several dependency vulnerability fixes ([#413](https://github.com/rokucommunity/brighterscript/pull/413), [#410](https://github.com/rokucommunity/brighterscript/pull/410), [#411](https://github.com/rokucommunity/brighterscript/pull/411)) + + + ## [0.38.0] - 2021-05-04 -### Adds +### Added - warning for mismatched class method accessibility ([#402](https://github.com/rokucommunity/brighterscript/pull/402)) - allow class field overrides in child classes as long as they are the same type ([#394](https://github.com/rokucommunity/brighterscript/pull/394)) From 7e20d8c5bc9116e4301eab57a29b2e4474106120 Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 14 May 2021 16:11:56 -0400 Subject: [PATCH 013/278] 0.38.1 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 68f0445fa..bf1f79553 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.0", + "version": "0.38.1", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 84827301b..83dc0a187 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.0", + "version": "0.38.1", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 93658e3a8183af8fa64bd1dc39f1e86b38d66326 Mon Sep 17 00:00:00 2001 From: Philippe Date: Mon, 17 May 2021 10:12:30 +0100 Subject: [PATCH 014/278] Incomplete statement in namespace can crash the server (#419) --- src/Scope.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Scope.ts b/src/Scope.ts index 73afc2344..8bbd97ae8 100644 --- a/src/Scope.ts +++ b/src/Scope.ts @@ -364,9 +364,9 @@ export class Scope { let ns = namespaceLookup[name.toLowerCase()]; ns.statements.push(...namespace.body.statements); for (let statement of namespace.body.statements) { - if (isClassStatement(statement)) { + if (isClassStatement(statement) && statement.name) { ns.classStatements[statement.name.text.toLowerCase()] = statement; - } else if (isFunctionStatement(statement)) { + } else if (isFunctionStatement(statement) && statement.name) { ns.functionStatements[statement.name.text.toLowerCase()] = statement; } } From 904ddcbf1a0dcb8e7bccac574901faa142bc244a Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 17 May 2021 06:49:34 -0400 Subject: [PATCH 015/278] Unit tests for #419 --- src/Scope.spec.ts | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/src/Scope.spec.ts b/src/Scope.spec.ts index 0b3ae056d..beace4cc1 100644 --- a/src/Scope.spec.ts +++ b/src/Scope.spec.ts @@ -8,6 +8,8 @@ import { ParseMode } from './parser/Parser'; import PluginInterface from './PluginInterface'; import { trim } from './testHelpers.spec'; import { Logger } from './Logger'; +import { BrsFile } from './files/BrsFile'; +import { FunctionStatement, NamespaceStatement } from './parser'; describe('Scope', () => { let sinon = sinonImport.createSandbox(); @@ -823,4 +825,27 @@ describe('Scope', () => { expect(completions.filter(x => !!x.documentation)).to.have.length.greaterThan(0); }); }); + + describe('buildNamespaceLookup', () => { + it('does not crash when class statement is missing `name` prop', () => { + program.addOrReplaceFile('source/main.bs', ` + namespace NameA + class + end class + end namespace + `); + program['scopes']['source'].buildNamespaceLookup(); + }); + + it('does not crash when function statement is missing `name` prop', () => { + const file = program.addOrReplaceFile('source/main.bs', ` + namespace NameA + function doSomething() + end function + end namespace + `); + delete ((file.ast.statements[0] as NamespaceStatement).body.statements[0] as FunctionStatement).name; + program['scopes']['source'].buildNamespaceLookup(); + }); + }); }); From 1994bc1bcc584c802d509b948468389d69a6a783 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 17 May 2021 06:50:49 -0400 Subject: [PATCH 016/278] update changelog for v0.38.2 --- CHANGELOG.md | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 5d39daff9..1d986ea05 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.38.2] +[0.38.2]: https://github.com/rokucommunity/brighterscript/compare/v0.38.1...v0.38.2 +### Fixed + - language server crash when namespaced function or class didn't have a name ([#419](https://github.com/rokucommunity/brighterscript/pull/419)) + + + ## [0.38.1] - 2021-05-14 [0.38.1]: https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.38.1 ### Changed From 37852377f1c9ed5f0231c0b5ba36305f0ac5b7d3 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 17 May 2021 06:52:21 -0400 Subject: [PATCH 017/278] lint fixes --- src/Scope.spec.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Scope.spec.ts b/src/Scope.spec.ts index beace4cc1..12a93ab92 100644 --- a/src/Scope.spec.ts +++ b/src/Scope.spec.ts @@ -8,8 +8,8 @@ import { ParseMode } from './parser/Parser'; import PluginInterface from './PluginInterface'; import { trim } from './testHelpers.spec'; import { Logger } from './Logger'; -import { BrsFile } from './files/BrsFile'; -import { FunctionStatement, NamespaceStatement } from './parser'; +import type { BrsFile } from './files/BrsFile'; +import type { FunctionStatement, NamespaceStatement } from './parser'; describe('Scope', () => { let sinon = sinonImport.createSandbox(); From 96a0638c17454e8360d8c2d63a347d9811589454 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 17 May 2021 06:53:37 -0400 Subject: [PATCH 018/278] 0.38.2 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index bf1f79553..a9f1c27df 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.1", + "version": "0.38.2", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 83dc0a187..93417b291 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.1", + "version": "0.38.2", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 8e687fc7fbf4b224e03b426013384b3083498db5 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Mon, 17 May 2021 07:00:50 -0400 Subject: [PATCH 019/278] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1d986ea05..5075c1fe3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,7 +6,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.38.2] +## [0.38.2] - 2021-05-17 [0.38.2]: https://github.com/rokucommunity/brighterscript/compare/v0.38.1...v0.38.2 ### Fixed - language server crash when namespaced function or class didn't have a name ([#419](https://github.com/rokucommunity/brighterscript/pull/419)) From efdcc2f8cadab1e8e5edcd93ceacf6360b602626 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Tue, 18 May 2021 22:58:45 -0400 Subject: [PATCH 020/278] Plugin semantic highlighting (#396) * Add baseline semantic token support * Colorize parameter types. * Implemented modifiers * fix lint issue --- package-lock.json | 34 ++++--- package.json | 4 +- src/LanguageServer.spec.ts | 1 + src/LanguageServer.ts | 46 +++++++-- src/Program.ts | 19 +++- src/Scope.ts | 2 +- src/SemanticTokenUtils.spec.ts | 95 +++++++++++++++++++ src/SemanticTokenUtils.ts | 92 ++++++++++++++++++ src/bscPlugin/BscPlugin.ts | 7 +- .../SemanticTokensProcessor.spec.ts | 48 ++++++++++ .../semanticTokens/SemanticTokensProcessor.ts | 71 ++++++++++++++ src/interfaces.ts | 19 +++- src/util.spec.ts | 58 +++++++++++ src/util.ts | 66 ++++++++++++- 14 files changed, 533 insertions(+), 29 deletions(-) create mode 100644 src/SemanticTokenUtils.spec.ts create mode 100644 src/SemanticTokenUtils.ts create mode 100644 src/bscPlugin/semanticTokens/SemanticTokensProcessor.spec.ts create mode 100644 src/bscPlugin/semanticTokens/SemanticTokensProcessor.ts diff --git a/package-lock.json b/package-lock.json index a9f1c27df..41d43454c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5962,23 +5962,31 @@ "vscode-jsonrpc": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/vscode-jsonrpc/-/vscode-jsonrpc-5.0.1.tgz", - "integrity": "sha512-JvONPptw3GAQGXlVV2utDcHx0BiY34FupW/kI6mZ5x06ER5DdPG/tXWMVHjTNULF5uKPOUUD0SaXg5QaubJL0A==" + "integrity": "sha512-JvONPptw3GAQGXlVV2utDcHx0BiY34FupW/kI6mZ5x06ER5DdPG/tXWMVHjTNULF5uKPOUUD0SaXg5QaubJL0A==", + "dev": true }, "vscode-languageserver": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/vscode-languageserver/-/vscode-languageserver-6.1.1.tgz", - "integrity": "sha512-DueEpkUAkD5XTR4MLYNr6bQIp/UFR0/IPApgXU3YfCBCB08u2sm9hRCs6DxYZELkk++STPjpcjksR2H8qI3cDQ==", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver/-/vscode-languageserver-7.0.0.tgz", + "integrity": "sha512-60HTx5ID+fLRcgdHfmz0LDZAXYEV68fzwG0JWwEPBode9NuMYTIxuYXPg4ngO8i8+Ou0lM7y6GzaYWbiDL0drw==", "requires": { - "vscode-languageserver-protocol": "^3.15.3" + "vscode-languageserver-protocol": "3.16.0" } }, "vscode-languageserver-protocol": { - "version": "3.15.3", - "resolved": "https://registry.npmjs.org/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.15.3.tgz", - "integrity": "sha512-zrMuwHOAQRhjDSnflWdJG+O2ztMWss8GqUUB8dXLR/FPenwkiBNkMIJJYfSN6sgskvsF0rHAoBowNQfbyZnnvw==", + "version": "3.16.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.16.0.tgz", + "integrity": "sha512-sdeUoAawceQdgIfTI+sdcwkiK2KU+2cbEYA0agzM2uqaUy2UpnnGHtWTHVEtS0ES4zHU0eMFRGN+oQgDxlD66A==", "requires": { - "vscode-jsonrpc": "^5.0.1", - "vscode-languageserver-types": "3.15.1" + "vscode-jsonrpc": "6.0.0", + "vscode-languageserver-types": "3.16.0" + }, + "dependencies": { + "vscode-jsonrpc": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/vscode-jsonrpc/-/vscode-jsonrpc-6.0.0.tgz", + "integrity": "sha512-wnJA4BnEjOSyFMvjZdpiOwhSq9uDoK8e/kpRJDTaMYzwlkrhG1fwDIZI94CLsLzlCK5cIbMMtFlJlfR57Lavmg==" + } } }, "vscode-languageserver-textdocument": { @@ -5987,9 +5995,9 @@ "integrity": "sha512-UIcJDjX7IFkck7cSkNNyzIz5FyvpQfY7sdzVy+wkKN/BLaD4DQ0ppXQrKePomCxTS7RrolK1I0pey0bG9eh8dA==" }, "vscode-languageserver-types": { - "version": "3.15.1", - "resolved": "https://registry.npmjs.org/vscode-languageserver-types/-/vscode-languageserver-types-3.15.1.tgz", - "integrity": "sha512-+a9MPUQrNGRrGU630OGbYVQ+11iOIovjCkqxajPa9w57Sd5ruK8WQNsslzpa0x/QJqC8kRc2DUxWjIFwoNm4ZQ==" + "version": "3.16.0", + "resolved": "https://registry.npmjs.org/vscode-languageserver-types/-/vscode-languageserver-types-3.16.0.tgz", + "integrity": "sha512-k8luDIWJWyenLc5ToFQQMaSrqCHiLwyKPHKPQZ5zz21vM+vIVUSvsRpcbiECH4WR88K2XZqc4ScRcZ7nk/jbeA==" }, "vscode-uri": { "version": "2.1.2", diff --git a/package.json b/package.json index 93417b291..750f91a20 100644 --- a/package.json +++ b/package.json @@ -131,8 +131,8 @@ "roku-deploy": "^3.2.4", "serialize-error": "^7.0.1", "source-map": "^0.7.3", - "vscode-languageserver": "^6.1.1", - "vscode-languageserver-protocol": "~3.15.3", + "vscode-languageserver": "7.0.0", + "vscode-languageserver-protocol": "3.16.0", "vscode-languageserver-textdocument": "^1.0.1", "vscode-uri": "^2.1.1", "xml2js": "^0.4.19", diff --git a/src/LanguageServer.spec.ts b/src/LanguageServer.spec.ts index 15787343e..06af66b38 100644 --- a/src/LanguageServer.spec.ts +++ b/src/LanguageServer.spec.ts @@ -59,6 +59,7 @@ describe('LanguageServer', () => { onWillSaveTextDocument: () => null, onWillSaveTextDocumentWaitUntil: () => null, onDidSaveTextDocument: () => null, + onRequest: () => null, workspace: { getWorkspaceFolders: () => workspaceFolders, getConfiguration: () => { diff --git a/src/LanguageServer.ts b/src/LanguageServer.ts index 0efb81b36..422c66ae5 100644 --- a/src/LanguageServer.ts +++ b/src/LanguageServer.ts @@ -18,9 +18,13 @@ import type { ReferenceParams, SignatureHelp, SignatureHelpParams, - CodeActionParams -} from 'vscode-languageserver'; + CodeActionParams, + SemanticTokensOptions, + SemanticTokens, + SemanticTokensParams +} from 'vscode-languageserver/node'; import { + SemanticTokensRequest, createConnection, DidChangeConfigurationNotification, FileChangeType, @@ -28,7 +32,7 @@ import { TextDocuments, TextDocumentSyncKind, CodeActionKind -} from 'vscode-languageserver'; +} from 'vscode-languageserver/node'; import { URI } from 'vscode-uri'; import { TextDocument } from 'vscode-languageserver-textdocument'; import type { BsConfig } from './BsConfig'; @@ -41,10 +45,10 @@ import { Throttler } from './Throttler'; import { KeyedThrottler } from './KeyedThrottler'; import { DiagnosticCollection } from './DiagnosticCollection'; import { isBrsFile } from './astUtils/reflection'; +import { encodeSemanticTokens, semanticTokensLegend } from './SemanticTokenUtils'; export class LanguageServer { - //cast undefined as any to get around strictNullChecks...it's ok in this case - private connection: Connection = undefined; + private connection = undefined as Connection; public workspaces = [] as Workspace[]; @@ -146,6 +150,9 @@ export class LanguageServer { this.connection.onCodeAction(this.onCodeAction.bind(this)); + //TODO switch to a more specific connection function call once they actually add it + this.connection.onRequest(SemanticTokensRequest.method, this.onFullSemanticTokens.bind(this)); + /* this.connection.onDidOpenTextDocument((params) => { // A text document got opened in VSCode. @@ -199,6 +206,10 @@ export class LanguageServer { }, documentSymbolProvider: true, workspaceSymbolProvider: true, + semanticTokensProvider: { + legend: semanticTokensLegend, + full: true + } as SemanticTokensOptions, referencesProvider: true, codeActionProvider: { codeActionKinds: [CodeActionKind.Refactor] @@ -318,8 +329,8 @@ export class LanguageServer { * Event handler for when the program wants to load file contents. * anytime the program wants to load a file, check with our in-memory document cache first */ - private documentFileResolver(pathAbsolute) { - let pathUri = URI.file(pathAbsolute).toString(); + private documentFileResolver(srcPath: string) { + let pathUri = URI.file(srcPath).toString(); let document = this.documents.get(pathUri); if (document) { return document.getText(); @@ -766,14 +777,14 @@ export class LanguageServer { }); return files.map(x => { return { - type: FileChangeType.Created as FileChangeType, + type: FileChangeType.Created, pathAbsolute: s`${x}` }; }); }); //add the new file changes to the changes array. - changes.push(...newFileChanges); + changes.push(...newFileChanges as any); //give every workspace the chance to handle file changes await Promise.all( @@ -1064,6 +1075,23 @@ export class LanguageServer { return results.filter((r) => r); } + @AddStackToErrorMessage + private async onFullSemanticTokens(params: SemanticTokensParams) { + await this.waitAllProgramFirstRuns(); + await this.keyedThrottler.onIdleOnce(util.uriToPath(params.textDocument.uri), true); + + const srcPath = util.uriToPath(params.textDocument.uri); + for (const workspace of this.workspaces) { + //find the first program that has this file, since it would be incredibly inefficient to generate semantic tokens for the same file multiple times. + if (workspace.builder.program.hasFile(srcPath)) { + let semanticTokens = workspace.builder.program.getSemanticTokens(srcPath); + return { + data: encodeSemanticTokens(semanticTokens) + } as SemanticTokens; + } + } + } + private diagnosticCollection = new DiagnosticCollection(); private async sendDiagnostics() { diff --git a/src/Program.ts b/src/Program.ts index e7b7ee382..66b2fc657 100644 --- a/src/Program.ts +++ b/src/Program.ts @@ -8,7 +8,7 @@ import { Scope } from './Scope'; import { DiagnosticMessages } from './DiagnosticMessages'; import { BrsFile } from './files/BrsFile'; import { XmlFile } from './files/XmlFile'; -import type { BsDiagnostic, File, FileReference, FileObj, BscFile } from './interfaces'; +import type { BsDiagnostic, File, FileReference, FileObj, BscFile, SemanticToken } from './interfaces'; import { standardizePath as s, util } from './util'; import { XmlScope } from './XmlScope'; import { DiagnosticFilterer } from './DiagnosticFilterer'; @@ -841,6 +841,23 @@ export class Program { return codeActions; } + /** + * Get semantic tokens for the specified file + */ + public getSemanticTokens(srcPath: string) { + const file = this.getFile(srcPath); + if (file) { + const result = [] as SemanticToken[]; + this.plugins.emit('onGetSemanticTokens', { + program: this, + file: file, + scopes: this.getScopesForFile(file), + semanticTokens: result + }); + return result; + } + } + public getSignatureHelp(filepath: string, position: Position): SignatureInfoObj[] { let file: BrsFile = this.getFile(filepath); if (!file || !isBrsFile(file)) { diff --git a/src/Scope.ts b/src/Scope.ts index 8bbd97ae8..cd3c640ea 100644 --- a/src/Scope.ts +++ b/src/Scope.ts @@ -84,7 +84,7 @@ export class Scope { /** * Tests if a class exists with the specified name * @param className - the all-lower-case namespace-included class name - * @param namespaceName - teh current namespace name + * @param containingNamespace - The namespace used to resolve relative class names. (i.e. the namespace around the current statement trying to find a class) */ public hasClass(className: string, namespaceName?: string): boolean { return !!this.getClass(className, namespaceName); diff --git a/src/SemanticTokenUtils.spec.ts b/src/SemanticTokenUtils.spec.ts new file mode 100644 index 000000000..bdf2fff8e --- /dev/null +++ b/src/SemanticTokenUtils.spec.ts @@ -0,0 +1,95 @@ +/* eslint-disable no-bitwise */ +import { expect } from 'chai'; +import { SemanticTokenModifiers, SemanticTokenTypes } from 'vscode-languageserver-protocol'; +import { encodeSemanticTokens, getModifierBitFlags, semanticTokensLegend } from './SemanticTokenUtils'; +import util from './util'; + +describe('SemanticTokenUtils', () => { + describe('encodeSemanticTokens', () => { + it('encodes single entry at start of line', () => { + expectEncodeEquals( + encodeSemanticTokens([{ + range: util.createRange(1, 0, 1, 2), + tokenType: SemanticTokenTypes.class + }]), + [ + 1, 0, 2, semanticTokensLegend.tokenTypes.indexOf(SemanticTokenTypes.class), 0 + ] + ); + }); + + it('encodes two non-touching entries on same line', () => { + expectEncodeEquals( + encodeSemanticTokens([{ + range: util.createRange(1, 0, 1, 2), + tokenType: SemanticTokenTypes.class + }, { + range: util.createRange(1, 10, 1, 12), + tokenType: SemanticTokenTypes.namespace + }]), + [ + 1, 0, 2, semanticTokensLegend.tokenTypes.indexOf(SemanticTokenTypes.class), 0, + 0, 10, 2, semanticTokensLegend.tokenTypes.indexOf(SemanticTokenTypes.namespace), 0 + ] + ); + }); + }); + + //these tests depend on the semanticTokensLegend.tokenModifiers being in a specific order. If those change order, then this test needs changed as well + describe('getModifierBitFlags', () => { + it('works for single modifier', () => { + expect(getModifierBitFlags([SemanticTokenModifiers.declaration])).to.eql(1 << 0); + expect(getModifierBitFlags([SemanticTokenModifiers.definition])).to.eql(1 << 1); + expect(getModifierBitFlags([SemanticTokenModifiers.readonly])).to.eql(1 << 2); + expect(getModifierBitFlags([SemanticTokenModifiers.static])).to.eql(1 << 3); + expect(getModifierBitFlags([SemanticTokenModifiers.deprecated])).to.eql(1 << 4); + expect(getModifierBitFlags([SemanticTokenModifiers.abstract])).to.eql(1 << 5); + expect(getModifierBitFlags([SemanticTokenModifiers.async])).to.eql(1 << 6); + expect(getModifierBitFlags([SemanticTokenModifiers.modification])).to.eql(1 << 7); + expect(getModifierBitFlags([SemanticTokenModifiers.documentation])).to.eql(1 << 8); + expect(getModifierBitFlags([SemanticTokenModifiers.defaultLibrary])).to.eql(1 << 9); + }); + + it('properly combines multiple modifiers', () => { + expect(getModifierBitFlags([ + SemanticTokenModifiers.declaration, //idx=0 + SemanticTokenModifiers.static, //idx=3 + SemanticTokenModifiers.documentation //idx=8 + ])).to.eql(0b100001001); + }); + }); +}); + +function expectEncodeEquals(actual: number[], expected: number[]) { + //results should be in multiples of 5 + expect(actual.length % 5).to.eql(0); + expect(expected.length % 5).to.eql(0); + + expect( + decodeSemanticTokens(actual) + ).to.eql( + decodeSemanticTokens(expected) + ); +} + +function decodeSemanticTokens(data: number[]) { + const result = [] as SemanticTokenDelta[]; + for (let i = 0; i < data.length; i += 5) { + result.push({ + deltaLine: data[i], + deltaCharacter: data[i + 1], + length: data[i + 2], + tokenTypeIndex: data[i + 3], + tokenModifierIndex: data[i + 4] + }); + } + return result; +} + +interface SemanticTokenDelta { + deltaLine: number; + deltaCharacter: number; + length: number; + tokenTypeIndex: number; + tokenModifierIndex: number; +} diff --git a/src/SemanticTokenUtils.ts b/src/SemanticTokenUtils.ts new file mode 100644 index 000000000..a382fec9f --- /dev/null +++ b/src/SemanticTokenUtils.ts @@ -0,0 +1,92 @@ +import type { + SemanticTokensLegend +} from 'vscode-languageserver-protocol/node'; +import { + SemanticTokenTypes, + SemanticTokenModifiers +} from 'vscode-languageserver-protocol/node'; +import { SemanticTokensBuilder } from 'vscode-languageserver/node'; +import type { + SemanticToken +} from './interfaces'; +import util from './util'; + +export const semanticTokensLegend = { + tokenTypes: [ + SemanticTokenTypes.namespace, + SemanticTokenTypes.type, + SemanticTokenTypes.class, + SemanticTokenTypes.enum, + SemanticTokenTypes.interface, + SemanticTokenTypes.struct, + SemanticTokenTypes.typeParameter, + SemanticTokenTypes.parameter, + SemanticTokenTypes.variable, + SemanticTokenTypes.property, + SemanticTokenTypes.enumMember, + SemanticTokenTypes.event, + SemanticTokenTypes.function, + SemanticTokenTypes.method, + SemanticTokenTypes.macro, + SemanticTokenTypes.keyword, + SemanticTokenTypes.modifier, + SemanticTokenTypes.comment, + SemanticTokenTypes.string, + SemanticTokenTypes.number, + SemanticTokenTypes.regexp, + SemanticTokenTypes.operator + ], + tokenModifiers: [ + SemanticTokenModifiers.declaration, + SemanticTokenModifiers.definition, + SemanticTokenModifiers.readonly, + SemanticTokenModifiers.static, + SemanticTokenModifiers.deprecated, + SemanticTokenModifiers.abstract, + SemanticTokenModifiers.async, + SemanticTokenModifiers.modification, + SemanticTokenModifiers.documentation, + SemanticTokenModifiers.defaultLibrary + ] +} as SemanticTokensLegend; + +/** + * The LSP has a very specific format for semantic tokens, so this encodes our internal representation into the LSP format. + * Currently all tokens must be single-line + * See for more info: https://microsoft.github.io/language-server-protocol/specifications/specification-current/#textDocument_semanticTokens + */ +export function encodeSemanticTokens(tokens: SemanticToken[]) { + util.sortByRange(tokens); + const builder = new SemanticTokensBuilder(); + for (const token of tokens) { + builder.push( + token.range.start.line, + token.range.start.character, + token.range.end.character - token.range.start.character, + //token type index + semanticTokensLegend.tokenTypes.indexOf(token.tokenType), + //modifier bit flags + token.tokenModifiers ? getModifierBitFlags(token.tokenModifiers) : 0 + ); + } + return builder.build().data; +} + +/** + * Convert an array of strings into a binary bitflag integer, where each non-zero bit indiciates the index of the modifier from `semanticTokensLegend.tokenModifiers` + */ +export function getModifierBitFlags(modifiers: SemanticTokenModifiers[]) { + let result = 0; + for (const modifier of modifiers) { + const idx = semanticTokensLegend.tokenModifiers.indexOf(modifier); + if (idx === -1) { + throw new Error(`Unknown semantic token modifier: '${modifier}'`); + } + + //convert the index into a bit flag by bitshifting 1 the by the number of zeros for the idx. + //example: idx=3. binary should be 0b1000, so we bitshift 1 << 3 + // eslint-disable-next-line no-bitwise + result |= 1 << idx; + } + return result; +} diff --git a/src/bscPlugin/BscPlugin.ts b/src/bscPlugin/BscPlugin.ts index a60c38197..ab5af61c9 100644 --- a/src/bscPlugin/BscPlugin.ts +++ b/src/bscPlugin/BscPlugin.ts @@ -1,5 +1,6 @@ -import type { OnGetCodeActionsEvent, CompilerPlugin } from '../interfaces'; +import type { CompilerPlugin, OnGetCodeActionsEvent, OnGetSemanticTokensEvent } from '../interfaces'; import { CodeActionsProcessor } from './codeActions/CodeActionsProcessor'; +import { SemanticTokensProcessor } from './semanticTokens/SemanticTokensProcessor'; export class BscPlugin implements CompilerPlugin { public name = 'BscPlugin'; @@ -7,4 +8,8 @@ export class BscPlugin implements CompilerPlugin { public onGetCodeActions(event: OnGetCodeActionsEvent) { new CodeActionsProcessor(event).process(); } + + public onGetSemanticTokens(event: OnGetSemanticTokensEvent) { + return new SemanticTokensProcessor(event).process(); + } } diff --git a/src/bscPlugin/semanticTokens/SemanticTokensProcessor.spec.ts b/src/bscPlugin/semanticTokens/SemanticTokensProcessor.spec.ts new file mode 100644 index 000000000..aa2a6e8d4 --- /dev/null +++ b/src/bscPlugin/semanticTokens/SemanticTokensProcessor.spec.ts @@ -0,0 +1,48 @@ +import { expect } from 'chai'; +import { SemanticTokenTypes } from 'vscode-languageserver-protocol'; +import type { BrsFile } from '../../files/BrsFile'; +import { Program } from '../../Program'; +import { expectZeroDiagnostics } from '../../testHelpers.spec'; +import { standardizePath as s, util } from '../../util'; + +const rootDir = s`${process.cwd()}/.tmp/rootDir`; + +describe('SemanticTokensProcessor', () => { + let program: Program; + beforeEach(() => { + program = new Program({ + rootDir: rootDir + }); + }); + afterEach(() => { + program.dispose(); + }); + + it('matches each namespace section', () => { + const file = program.addOrReplaceFile('source/main.bs', ` + namespace Earthlings.Humanoids + class Person + end class + end namespace + class Dog + sub new() + m.owner = new Earthlings.Humanoids.Person() + end sub + end class + `); + program.validate(); + expectZeroDiagnostics(program); + expect( + program.getSemanticTokens(file.pathAbsolute) + ).to.eql([{ + range: util.createRange(7, 34, 7, 44), + tokenType: SemanticTokenTypes.namespace + }, { + range: util.createRange(7, 45, 7, 54), + tokenType: SemanticTokenTypes.namespace + }, { + range: util.createRange(7, 55, 7, 61), + tokenType: SemanticTokenTypes.class + }]); + }); +}); diff --git a/src/bscPlugin/semanticTokens/SemanticTokensProcessor.ts b/src/bscPlugin/semanticTokens/SemanticTokensProcessor.ts new file mode 100644 index 000000000..812b4103a --- /dev/null +++ b/src/bscPlugin/semanticTokens/SemanticTokensProcessor.ts @@ -0,0 +1,71 @@ +import type { Range } from 'vscode-languageserver-protocol'; +import { SemanticTokenTypes } from 'vscode-languageserver-protocol'; +import { isBrsFile, isCustomType } from '../../astUtils/reflection'; +import type { BrsFile } from '../../files/BrsFile'; +import type { OnGetSemanticTokensEvent } from '../../interfaces'; +import { ParseMode } from '../../parser'; +import util from '../../util'; + +export class SemanticTokensProcessor { + public constructor( + public event: OnGetSemanticTokensEvent + ) { + + } + + public process() { + if (isBrsFile(this.event.file)) { + this.processBrsFile(this.event.file); + } + } + + private processBrsFile(file: BrsFile) { + + const classes = [] as Array<{ className: string; namespaceName: string; range: Range }>; + + //classes used in function param types + for (const func of file.parser.references.functionExpressions) { + for (const parm of func.parameters) { + if (isCustomType(parm.type)) { + classes.push({ + className: parm.typeToken.text, + namespaceName: parm.namespaceName?.getName(ParseMode.BrighterScript), + range: parm.typeToken.range + }); + } + } + } + //classes used in `new` expressions + for (const expr of file.parser.references.newExpressions) { + classes.push({ + className: expr.className.getName(ParseMode.BrighterScript), + namespaceName: expr.namespaceName?.getName(ParseMode.BrighterScript), + range: expr.className.range + }); + } + + for (const cls of classes) { + if ( + cls.className.length > 0 && + //only highlight classes that are in scope + this.event.scopes.some(x => x.hasClass(cls.className, cls.namespaceName)) + ) { + const tokens = util.splitGetRange('.', cls.className, cls.range); + //namespace parts (skip the final array entry) + for (let i = 0; i < tokens.length - 1; i++) { + const token = tokens[i]; + this.event.semanticTokens.push({ + range: token.range, + tokenType: SemanticTokenTypes.namespace + }); + } + //class name + this.event.semanticTokens.push({ + range: tokens.pop().range, + tokenType: SemanticTokenTypes.class + }); + } + } + + } +} diff --git a/src/interfaces.ts b/src/interfaces.ts index 90d50f781..9a243f32d 100644 --- a/src/interfaces.ts +++ b/src/interfaces.ts @@ -1,4 +1,4 @@ -import type { Range, Diagnostic, CodeAction } from 'vscode-languageserver'; +import type { Range, Diagnostic, CodeAction, SemanticTokenTypes, SemanticTokenModifiers } from 'vscode-languageserver'; import type { Scope } from './Scope'; import type { BrsFile } from './files/BrsFile'; import type { XmlFile } from './files/XmlFile'; @@ -193,6 +193,7 @@ export interface CompilerPlugin { beforeProgramTranspile?: (program: Program, entries: TranspileObj[]) => void; afterProgramTranspile?: (program: Program, entries: TranspileObj[]) => void; onGetCodeActions?: PluginHandler; + onGetSemanticTokens?: PluginHandler; //scope events afterScopeCreate?: (scope: Scope) => void; beforeScopeDispose?: (scope: Scope) => void; @@ -219,6 +220,22 @@ export interface OnGetCodeActionsEvent { codeActions: CodeAction[]; } +export interface OnGetSemanticTokensEvent { + program: Program; + file: BscFile; + scopes: Scope[]; + semanticTokens: SemanticToken[]; +} + +export interface SemanticToken { + range: Range; + tokenType: SemanticTokenTypes; + /** + * An optional array of modifiers for this token + */ + tokenModifiers?: SemanticTokenModifiers[]; +} + export interface TypedefProvider { getTypedef(state: TranspileState): Array; } diff --git a/src/util.spec.ts b/src/util.spec.ts index a8d877f64..f1b85c9bc 100644 --- a/src/util.spec.ts +++ b/src/util.spec.ts @@ -605,4 +605,62 @@ describe('util', () => { )).to.be.true; }); }); + + it('sortByRange', () => { + const front = { + range: util.createRange(1, 1, 1, 2) + }; + const middle = { + range: util.createRange(1, 3, 1, 4) + }; + const back = { + range: util.createRange(1, 5, 1, 6) + }; + expect( + util.sortByRange([middle, front, back]) + ).to.eql([ + front, middle, back + ]); + }); + + describe('splitWithLocation', () => { + it('works with no split items', () => { + expect( + util.splitGetRange('.', 'hello', util.createRange(2, 10, 2, 15)) + ).to.eql([{ + text: 'hello', + range: util.createRange(2, 10, 2, 15) + }]); + }); + + it('handles empty chunks', () => { + expect( + util.splitGetRange('l', 'hello', util.createRange(2, 10, 2, 15)) + ).to.eql([{ + text: 'he', + range: util.createRange(2, 10, 2, 12) + }, { + text: 'o', + range: util.createRange(2, 14, 2, 15) + }]); + }); + + it('handles multiple non-empty chunks', () => { + expect( + util.splitGetRange('.', 'abc.d.efgh.i', util.createRange(2, 10, 2, 2)) + ).to.eql([{ + text: 'abc', + range: util.createRange(2, 10, 2, 13) + }, { + text: 'd', + range: util.createRange(2, 14, 2, 15) + }, { + text: 'efgh', + range: util.createRange(2, 16, 2, 20) + }, { + text: 'i', + range: util.createRange(2, 21, 2, 22) + }]); + }); + }); }); diff --git a/src/util.ts b/src/util.ts index 66f034dc8..b13b0d3c5 100644 --- a/src/util.ts +++ b/src/util.ts @@ -24,7 +24,7 @@ import { VoidType } from './types/VoidType'; import { ParseMode } from './parser/Parser'; import type { DottedGetExpression, Expression, VariableExpression } from './parser/Expression'; import { Logger, LogLevel } from './Logger'; -import type { Token } from './lexer'; +import type { Locatable, Token } from './lexer'; import { TokenKind } from './lexer'; import { isDottedGetExpression, isExpression, isVariableExpression, WalkMode } from './astUtils'; import { CustomType } from './types/CustomType'; @@ -1136,6 +1136,70 @@ export class Util { source: 'brs' }; } + + /** + * Sort an array of objects that have a Range + */ + public sortByRange(locatables: Locatable[]) { + //sort the tokens by range + return locatables.sort((a, b) => { + //start line + if (a.range.start.line < b.range.start.line) { + return -1; + } + if (a.range.start.line > b.range.start.line) { + return 1; + } + //start char + if (a.range.start.character < b.range.start.character) { + return -1; + } + if (a.range.start.character > b.range.start.character) { + return 1; + } + //end line + if (a.range.end.line < b.range.end.line) { + return -1; + } + if (a.range.end.line > b.range.end.line) { + return 1; + } + //end char + if (a.range.end.character < b.range.end.character) { + return -1; + } else if (a.range.end.character > b.range.end.character) { + return 1; + } + return 0; + }); + } + + /** + * Split the given text and return ranges for each chunk. + * Only works for single-line strings + */ + public splitGetRange(separator: string, text: string, range: Range) { + const chunks = text.split(separator); + const result = [] as Array<{ text: string; range: Range }>; + let offset = 0; + for (let i = 0; i < chunks.length; i++) { + const chunk = chunks[i]; + //only keep nonzero chunks + if (chunk.length > 0) { + result.push({ + text: chunk, + range: this.createRange( + range.start.line, + range.start.character + offset, + range.end.line, + range.start.character + offset + chunk.length + ) + }); + } + offset += chunk.length + separator.length; + } + return result; + } } /** From e71b517031ceadce88fb437c756076172b8ea23f Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 18 May 2021 23:06:42 -0400 Subject: [PATCH 021/278] update changelog for v0.39.0 --- CHANGELOG.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 5075c1fe3..adec7ede4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.39.0] - 2021-05-18 +[0.39.0]: https://github.com/rokucommunity/brighterscript/compare/v0.38.2...v0.39.0 +### Added + - semantic token support for plugins + - basic semantic tokens for `new` statements and namespace usage + + + ## [0.38.2] - 2021-05-17 [0.38.2]: https://github.com/rokucommunity/brighterscript/compare/v0.38.1...v0.38.2 ### Fixed From 4a0f857115ba3061bbf9e7f15f3ddc5cd056db98 Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 18 May 2021 23:08:20 -0400 Subject: [PATCH 022/278] 0.39.0 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 41d43454c..47ca95acf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.2", + "version": "0.39.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 750f91a20..bf4c2ed28 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.38.2", + "version": "0.39.0", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From aeb63d5bb34edf329fdef0a166d3d9c0eaeb457e Mon Sep 17 00:00:00 2001 From: Philippe Date: Sat, 22 May 2021 00:05:39 +0100 Subject: [PATCH 023/278] Fix code actions construction for "replace" tasks (#421) * Fix code actions construction for "replace" tasks Also add missing information in the action. * Re-export CodeActionKind --- src/CodeActionUtil.ts | 26 ++++++++++++++++++++++---- 1 file changed, 22 insertions(+), 4 deletions(-) diff --git a/src/CodeActionUtil.ts b/src/CodeActionUtil.ts index 76b4b71ee..c80d5d101 100644 --- a/src/CodeActionUtil.ts +++ b/src/CodeActionUtil.ts @@ -1,5 +1,5 @@ -import type { CodeActionKind, Diagnostic, Position, Range, WorkspaceEdit } from 'vscode-languageserver'; -import { CodeAction, TextEdit } from 'vscode-languageserver'; +import type { Diagnostic, Position, Range, WorkspaceEdit } from 'vscode-languageserver'; +import { CodeActionKind, CodeAction, TextEdit } from 'vscode-languageserver'; import { URI } from 'vscode-uri'; export class CodeActionUtil { @@ -20,13 +20,31 @@ export class CodeActionUtil { TextEdit.insert(change.position, change.newText) ); } else if (change.type === 'replace') { - TextEdit.replace(change.range, change.newText); + edit.changes[uri].push( + TextEdit.replace(change.range, change.newText) + ); } } - return CodeAction.create(obj.title, edit); + const action = CodeAction.create(obj.title, edit, obj.kind); + action.isPreferred = obj.isPreferred; + action.diagnostics = this.serializableDiagnostics(obj.diagnostics); + return action; + } + + public serializableDiagnostics(diagnostics: Diagnostic[]) { + return diagnostics?.map(({ range, severity, code, source, message, relatedInformation }) => ({ + range: range, + severity: severity, + source: source, + code: code, + message: message, + relatedInformation: relatedInformation + })); } } +export { CodeActionKind }; + export interface CodeActionShorthand { title: string; diagnostics?: Diagnostic[]; From 12756f50b9e645ed3ff62e8bc18037a86cdf49a8 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Sat, 22 May 2021 11:17:49 -0400 Subject: [PATCH 024/278] Add unit tests for Logger and Stopwatch (#420) --- src/Logger.spec.ts | 147 ++++++++++++++++++++++++++++++++++++++++++ src/Logger.ts | 7 +- src/Stopwatch.spec.ts | 55 +++++++++++++++- 3 files changed, 205 insertions(+), 4 deletions(-) create mode 100644 src/Logger.spec.ts diff --git a/src/Logger.spec.ts b/src/Logger.spec.ts new file mode 100644 index 000000000..e34a7eee8 --- /dev/null +++ b/src/Logger.spec.ts @@ -0,0 +1,147 @@ +import { expect } from 'chai'; +import { Logger, LogLevel, noop } from './Logger'; +import chalk from 'chalk'; +import { createSandbox } from 'sinon'; +const sinon = createSandbox(); + +describe('Logger', () => { + let logger: Logger; + + beforeEach(() => { + logger = new Logger(LogLevel.trace); + sinon.restore(); + //disable chalk colors for testing + sinon.stub(chalk, 'grey').callsFake((arg) => arg as any); + }); + + it('noop does nothing', () => { + noop(); + }); + + it('loglevel setter converts string to enum', () => { + (logger as any).logLevel = 'error'; + expect(logger.logLevel).to.eql(LogLevel.error); + (logger as any).logLevel = 'info'; + expect(logger.logLevel).to.eql(LogLevel.info); + }); + + it('uses LogLevel.log by default', () => { + logger = new Logger(); + expect(logger.logLevel).to.eql(LogLevel.log); + }); + + describe('log methods call correct error type', () => { + it('error', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.error(); + expect(stub.getCalls()[0].args[0]).to.eql(console.error); + }); + + it('warn', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.warn(); + expect(stub.getCalls()[0].args[0]).to.eql(console.warn); + }); + + it('log', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.log(); + expect(stub.getCalls()[0].args[0]).to.eql(console.log); + }); + + it('info', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.info(); + expect(stub.getCalls()[0].args[0]).to.eql(console.info); + }); + + it('debug', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.debug(); + expect(stub.getCalls()[0].args[0]).to.eql(console.debug); + }); + + it('trace', () => { + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.trace(); + expect(stub.getCalls()[0].args[0]).to.eql(console.trace); + }); + }); + + it('skips all errors on error level', () => { + logger.logLevel = LogLevel.off; + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.trace(); + logger.debug(); + logger.info(); + logger.log(); + logger.warn(); + logger.error(); + + expect( + stub.getCalls().map(x => x.args[0]) + ).to.eql([]); + }); + + it('does not skip when log level is high enough', () => { + logger.logLevel = LogLevel.trace; + const stub = sinon.stub(logger as any, 'writeToLog').callsFake(() => { }); + logger.trace(); + logger.debug(); + logger.info(); + logger.log(); + logger.warn(); + logger.error(); + + expect( + stub.getCalls().map(x => x.args[0]) + ).to.eql([ + console.trace, + console.debug, + console.info, + console.log, + console.warn, + console.error + ]); + }); + + describe('time', () => { + it('calls action even if logLevel is wrong', () => { + logger.logLevel = LogLevel.error; + const spy = sinon.spy(); + logger.time(LogLevel.info, null, spy); + expect(spy.called).to.be.true; + }); + + it('runs timer when loglevel is right', () => { + logger.logLevel = LogLevel.log; + const spy = sinon.spy(); + logger.time(LogLevel.log, null, spy); + expect(spy.called).to.be.true; + }); + + it('returns value', () => { + logger.logLevel = LogLevel.log; + const spy = sinon.spy(() => { + return true; + }); + expect( + logger.time(LogLevel.log, null, spy) + ).to.be.true; + expect(spy.called).to.be.true; + }); + + it('gives callable pause and resume functions even when not running timer', () => { + logger.time(LogLevel.info, null, (pause, resume) => { + pause(); + resume(); + }); + }); + + it('waits for and returns a promise when a promise is returned from the action', () => { + expect(logger.time(LogLevel.info, ['message'], () => { + return Promise.resolve(); + })).to.be.instanceof(Promise); + }); + }); +}); diff --git a/src/Logger.ts b/src/Logger.ts index 28e9f5cc0..16f595aac 100644 --- a/src/Logger.ts +++ b/src/Logger.ts @@ -124,7 +124,7 @@ export class Logger { let logLevelString = LogLevel[logLevel]; //write the initial log - this[logLevelString](...messages); + this[logLevelString](...messages ?? []); this.indent += ' '; stopwatch.start(); @@ -135,7 +135,7 @@ export class Logger { //return a function to call when the timer is complete let done = () => { this.indent = this.indent.substring(2); - this[logLevelString](...messages, `finished. (${chalk.blue(stopwatch.getDurationText())})`); + this[logLevelString](...messages ?? [], `finished. (${chalk.blue(stopwatch.getDurationText())})`); }; //if this is a promise, wait for it to resolve and then return the original result @@ -154,11 +154,12 @@ export class Logger { } } -function noop() { +export function noop() { } export enum LogLevel { + off = 0, error = 1, warn = 2, log = 3, diff --git a/src/Stopwatch.spec.ts b/src/Stopwatch.spec.ts index 93a1eaa4a..911c46413 100644 --- a/src/Stopwatch.spec.ts +++ b/src/Stopwatch.spec.ts @@ -1,7 +1,60 @@ -import { Stopwatch } from './Stopwatch'; import { expect } from 'chai'; +import { Stopwatch } from './Stopwatch'; +import { util } from './util'; describe('Stopwatch', () => { + let stopwatch: Stopwatch; + beforeEach(() => { + stopwatch = new Stopwatch(); + }); + + it('constructs', () => { + stopwatch = new Stopwatch(); + }); + + it('starts', () => { + expect(stopwatch['startTime']).to.not.exist; + stopwatch.start(); + expect(stopwatch['startTime']).to.exist; + }); + + it('resets', () => { + stopwatch.start(); + expect(stopwatch['startTime']).to.exist; + stopwatch.reset(); + expect(stopwatch['startTime']).to.not.exist; + }); + + it('stops', async () => { + stopwatch.start(); + expect(stopwatch['startTime']).to.exist; + await util.sleep(3); + stopwatch.stop(); + expect(stopwatch['startTime']).to.not.exist; + expect(stopwatch['totalMilliseconds']).to.be.gte(2); + }); + + it('stop multiple times has no effect', () => { + stopwatch.start(); + stopwatch.stop(); + stopwatch.stop(); + }); + + it('breaks out hours, minutes, and seconds', () => { + stopwatch['totalMilliseconds'] = (17 * 60 * 1000) + (43 * 1000) + 30; + expect(stopwatch.getDurationText()).to.eql('17m43s30.0ms'); + }); + + it('returns only seconds and milliseconds', () => { + stopwatch['totalMilliseconds'] = (43 * 1000) + 30; + expect(stopwatch.getDurationText()).to.eql('43s30.0ms'); + }); + + it('returns only milliseconds', () => { + stopwatch['totalMilliseconds'] = 30; + expect(stopwatch.getDurationText()).to.eql('30.0ms'); + }); + it('works for single run', async () => { let stopwatch = new Stopwatch(); stopwatch.start(); From a8198728688334eea94e192de2042f4002dfa6d5 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 24 May 2021 06:46:15 -0400 Subject: [PATCH 025/278] update changelog for v0.39.1 --- CHANGELOG.md | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index adec7ede4..4a5735556 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,16 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.39.1] - 2021-05-24 +[0.39.1]: https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1 +### Changed + - re-export `CodeActionKind` so plugins don't need to import from vscode-brightscript-language directly. +### Fixed + - code action for "replace" tasks bug + - include missing information in the CodeAction construction + + + ## [0.39.0] - 2021-05-18 [0.39.0]: https://github.com/rokucommunity/brighterscript/compare/v0.38.2...v0.39.0 ### Added From 00313a29a69de044cdb99a3d23729b1bb982fea9 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 24 May 2021 06:52:41 -0400 Subject: [PATCH 026/278] 0.39.1 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 47ca95acf..1874fa7f9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.0", + "version": "0.39.1", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index bf4c2ed28..bb37835e8 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.0", + "version": "0.39.1", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From a90c484f9821f877a1a6d3e978abbe03e0fff88a Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 28 May 2021 16:54:59 -0400 Subject: [PATCH 027/278] roku-deploy@3.4.0 --- package-lock.json | 276 ++++++++++++---------------------------------- package.json | 2 +- 2 files changed, 69 insertions(+), 209 deletions(-) diff --git a/package-lock.json b/package-lock.json index 1874fa7f9..fadbf7821 100644 --- a/package-lock.json +++ b/package-lock.json @@ -879,53 +879,6 @@ "default-require-extensions": "^3.0.0" } }, - "archiver": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/archiver/-/archiver-3.1.1.tgz", - "integrity": "sha512-5Hxxcig7gw5Jod/8Gq0OneVgLYET+oNHcxgWItq4TbhOzRLKNAFUb9edAftiMKXvXfCB0vbGrJdZDNq0dWMsxg==", - "requires": { - "archiver-utils": "^2.1.0", - "async": "^2.6.3", - "buffer-crc32": "^0.2.1", - "glob": "^7.1.4", - "readable-stream": "^3.4.0", - "tar-stream": "^2.1.0", - "zip-stream": "^2.1.2" - } - }, - "archiver-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", - "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", - "requires": { - "glob": "^7.1.4", - "graceful-fs": "^4.2.0", - "lazystream": "^1.0.0", - "lodash.defaults": "^4.2.0", - "lodash.difference": "^4.5.0", - "lodash.flatten": "^4.4.0", - "lodash.isplainobject": "^4.0.6", - "lodash.union": "^4.6.0", - "normalize-path": "^3.0.0", - "readable-stream": "^2.0.0" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - } - } - }, "archy": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz", @@ -1087,14 +1040,6 @@ "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", "dev": true }, - "async": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", - "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", - "requires": { - "lodash": "^4.17.14" - } - }, "asynckit": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", @@ -1118,7 +1063,8 @@ "base64-js": { "version": "1.5.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true }, "bcrypt-pbkdf": { "version": "1.0.2", @@ -1143,16 +1089,6 @@ "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==" }, - "bl": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/bl/-/bl-4.0.3.tgz", - "integrity": "sha512-fs4G6/Hu4/EE+F75J8DuN/0IpQqNjAdC7aEQv7Qt8MHGUH7Ckv2MwTEEeN9QehD0pfIDkMI1bkHYkKy7xHyKIg==", - "requires": { - "buffer": "^5.5.0", - "inherits": "^2.0.4", - "readable-stream": "^3.4.0" - } - }, "brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -1177,32 +1113,28 @@ "dev": true }, "browserslist": { - "version": "4.16.3", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.16.3.tgz", - "integrity": "sha512-vIyhWmIkULaq04Gt93txdh+j02yX/JzlyhLYbV3YQCn/zvES3JnY7TifHHvvr1w5hTDluNKMkV05cs4vy8Q7sw==", + "version": "4.16.6", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.16.6.tgz", + "integrity": "sha512-Wspk/PqO+4W9qp5iUTJsa1B/QrYn1keNCcEP5OvP7WBwT4KaDly0uONYmC6Xa3Z5IqnUgS0KcgLYu1l74x0ZXQ==", "dev": true, "requires": { - "caniuse-lite": "^1.0.30001181", - "colorette": "^1.2.1", - "electron-to-chromium": "^1.3.649", + "caniuse-lite": "^1.0.30001219", + "colorette": "^1.2.2", + "electron-to-chromium": "^1.3.723", "escalade": "^3.1.1", - "node-releases": "^1.1.70" + "node-releases": "^1.1.71" } }, "buffer": { "version": "5.7.1", "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, "requires": { "base64-js": "^1.3.1", "ieee754": "^1.1.13" } }, - "buffer-crc32": { - "version": "0.2.13", - "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", - "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=" - }, "buffer-from": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", @@ -1261,9 +1193,9 @@ "dev": true }, "caniuse-lite": { - "version": "1.0.30001204", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001204.tgz", - "integrity": "sha512-JUdjWpcxfJ9IPamy2f5JaRDCaqJOxDzOSKtbdx4rH9VivMd1vIzoPumsJa9LoMIi4Fx2BV2KZOxWhNkBjaYivQ==", + "version": "1.0.30001230", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001230.tgz", + "integrity": "sha512-5yBd5nWCBS+jWKTcHOzXwo5xzcj4ePE/yjtkZyUV1BTUmrBaA9MRGC+e7mxnqXSA90CmCA8L3eKLaSUkt099IQ==", "dev": true }, "caseless": { @@ -1420,33 +1352,6 @@ "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", "dev": true }, - "compress-commons": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-2.1.1.tgz", - "integrity": "sha512-eVw6n7CnEMFzc3duyFVrQEuY1BlHR3rYsSztyG32ibGMW722i3C6IizEGMFmfMU+A+fALvBIwxN3czffTcdA+Q==", - "requires": { - "buffer-crc32": "^0.2.13", - "crc32-stream": "^3.0.1", - "normalize-path": "^3.0.0", - "readable-stream": "^2.3.6" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - } - } - }, "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -1514,23 +1419,6 @@ "request": "^2.88.2" } }, - "crc": { - "version": "3.8.0", - "resolved": "https://registry.npmjs.org/crc/-/crc-3.8.0.tgz", - "integrity": "sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ==", - "requires": { - "buffer": "^5.1.0" - } - }, - "crc32-stream": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-3.0.1.tgz", - "integrity": "sha512-mctvpXlbzsvK+6z8kJwSJ5crm7yBwrQMTybJzMw1O4lLGJqjlDCXY2Zw7KheiA6XBEcBmfLx1D88mjRGVJtY9w==", - "requires": { - "crc": "^3.4.4", - "readable-stream": "^3.4.0" - } - }, "cross-platform-clear-console": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/cross-platform-clear-console/-/cross-platform-clear-console-2.3.0.tgz", @@ -1858,9 +1746,9 @@ } }, "electron-to-chromium": { - "version": "1.3.702", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.702.tgz", - "integrity": "sha512-qJVUKFWQnF6wP7MmTngDkmm8/KPzaiTXNFOAg5j7DSa6J7kPou7mTBqC8jpUOxauQWwHR3pn4dMRdV8IE1xdtA==", + "version": "1.3.741", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.741.tgz", + "integrity": "sha512-4i3T0cwnHo1O4Mnp9JniEco8bZiXoqbm3PhW5hv7uu8YLg35iajYrRnNyKFaN8/8SSTskU2hYqVTeYVPceSpUA==", "dev": true }, "emoji-regex": { @@ -1868,14 +1756,6 @@ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, - "end-of-stream": { - "version": "1.4.4", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", - "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", - "requires": { - "once": "^1.4.0" - } - }, "enhanced-resolve": { "version": "5.7.0", "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.7.0.tgz", @@ -2712,11 +2592,6 @@ "integrity": "sha512-cHEpEQHUg0f8XdtZCc2ZAhrHzKzT0MrFUTcvx+hfxYu7rGMDc5SKoXFh+n4YigxsHXRzc6OrCshdR1bWH6HHyg==", "dev": true }, - "fs-constants": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", - "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==" - }, "fs-extra": { "version": "7.0.1", "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", @@ -3004,7 +2879,8 @@ "ieee754": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "dev": true }, "ignore": { "version": "5.1.8", @@ -3012,6 +2888,11 @@ "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", "dev": true }, + "immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=" + }, "import-fresh": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.1.tgz", @@ -3453,18 +3334,15 @@ "verror": "1.10.0" } }, - "just-extend": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-4.1.1.tgz", - "integrity": "sha512-aWgeGFW67BP3e5181Ep1Fv2v8z//iBJfrvyTnq8wG86vEESwmonn1zPBJ0VfmT9CJq2FIT0VsETtrNFm2a+SHA==", - "dev": true - }, - "lazystream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.0.tgz", - "integrity": "sha1-9plf4PggOS9hOWvolGJAe7dxaOQ=", + "jszip": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.6.0.tgz", + "integrity": "sha512-jgnQoG9LKnWO3mnVNBnfhkh0QknICd1FGSrXcgrl67zioyJ4wgx25o9ZqwNtrROSflGBCGYnJfjrIyRIby1OoQ==", "requires": { - "readable-stream": "^2.0.5" + "lie": "~3.3.0", + "pako": "~1.0.2", + "readable-stream": "~2.3.6", + "set-immediate-shim": "~1.0.1" }, "dependencies": { "readable-stream": { @@ -3483,6 +3361,12 @@ } } }, + "just-extend": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-4.1.1.tgz", + "integrity": "sha512-aWgeGFW67BP3e5181Ep1Fv2v8z//iBJfrvyTnq8wG86vEESwmonn1zPBJ0VfmT9CJq2FIT0VsETtrNFm2a+SHA==", + "dev": true + }, "lcov-parse": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/lcov-parse/-/lcov-parse-1.0.0.tgz", @@ -3499,6 +3383,14 @@ "type-check": "~0.4.0" } }, + "lie": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", + "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", + "requires": { + "immediate": "~3.0.5" + } + }, "load-json-file": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", @@ -3537,22 +3429,8 @@ "lodash": { "version": "4.17.21", "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", - "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" - }, - "lodash.defaults": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", - "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw=" - }, - "lodash.difference": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", - "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=" - }, - "lodash.flatten": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", - "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8=" + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true }, "lodash.flattendeep": { "version": "4.4.0", @@ -3566,16 +3444,6 @@ "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", "dev": true }, - "lodash.isplainobject": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", - "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" - }, - "lodash.union": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", - "integrity": "sha1-SLtQiECfFvGCFmZkHETdGqrjzYg=" - }, "log-driver": { "version": "1.2.7", "resolved": "https://registry.npmjs.org/log-driver/-/log-driver-1.2.7.tgz", @@ -4142,9 +4010,9 @@ } }, "node-releases": { - "version": "1.1.71", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.71.tgz", - "integrity": "sha512-zR6HoT6LrLCRBwukmrVbHv0EpEQjksO6GmFcZQQuCAy139BEsoVKPYnf3jongYW83fAa1torLGYwxxky/p28sg==", + "version": "1.1.72", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.72.tgz", + "integrity": "sha512-LLUo+PpH3dU6XizX3iVoubUNheF/owjXCZZ5yACDxNnPtgFuludV1ZL3ayK1kVep42Rmm0+R9/Y60NQbZ2bifw==", "dev": true }, "node-run-cmd": { @@ -4589,6 +4457,11 @@ "release-zalgo": "^1.0.0" } }, + "pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" + }, "parent-module": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", @@ -5029,6 +4902,7 @@ "version": "3.6.0", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, "requires": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", @@ -5164,16 +5038,19 @@ } }, "roku-deploy": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.2.4.tgz", - "integrity": "sha512-teZB4sRWS2EVHx8KWSqHW4QS9B2Z5jD5MrZrbdikl3thMljdVkbGLkarxrFw1NZfMR6nf0b0XXygOWjvePIeuw==", + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.0.tgz", + "integrity": "sha512-z5TTj+r4EBQnVfepUEebARB0IXmqq6ZvMRqJoSac2kfni4uofwsBMIWO4UwDaleZAAKxW6GR/YDRl/kgzfpGgw==", "requires": { - "archiver": "^3.0.0", "dateformat": "^3.0.3", + "eventemitter3": "^4.0.7", "fs-extra": "^7.0.1", "glob": "^7.1.6", "jsonc-parser": "^2.3.0", + "jszip": "^3.6.0", "minimatch": "^3.0.4", + "moment": "^2.29.1", + "parse-ms": "^2.1.0", "path": "^0.12.7", "request": "^2.88.0", "xml2js": "^0.4.23" @@ -5263,6 +5140,11 @@ "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", "dev": true }, + "set-immediate-shim": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz", + "integrity": "sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E=" + }, "shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -5657,18 +5539,6 @@ "integrity": "sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw==", "dev": true }, - "tar-stream": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", - "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", - "requires": { - "bl": "^4.0.3", - "end-of-stream": "^1.4.1", - "fs-constants": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^3.1.1" - } - }, "temp": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/temp/-/temp-0.4.0.tgz", @@ -6234,16 +6104,6 @@ "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", "dev": true - }, - "zip-stream": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/zip-stream/-/zip-stream-2.1.3.tgz", - "integrity": "sha512-EkXc2JGcKhO5N5aZ7TmuNo45budRaFGHOmz24wtJR7znbNqDPmdZtUauKX6et8KAVseAMBOyWJqEpXcHTBsh7Q==", - "requires": { - "archiver-utils": "^2.1.0", - "compress-commons": "^2.1.1", - "readable-stream": "^3.4.0" - } } } } diff --git a/package.json b/package.json index bb37835e8..ea03d1dfe 100644 --- a/package.json +++ b/package.json @@ -128,7 +128,7 @@ "moment": "^2.23.0", "p-settle": "^2.1.0", "parse-ms": "^2.1.0", - "roku-deploy": "^3.2.4", + "roku-deploy": "^3.4.0", "serialize-error": "^7.0.1", "source-map": "^0.7.3", "vscode-languageserver": "7.0.0", From 4c66542b114234cd57a9f13cb73e39f5a5c48b5b Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 28 May 2021 16:57:06 -0400 Subject: [PATCH 028/278] update changelog for v0.39.2. Added missing changelog diff links. --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4a5735556..d7f7bb8b6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.39.2] - 2021-05-28 +### Changed + - upgraded to [roku-deploy@3.4.0](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#340---2021-05-28) which brings significant zip speed improvements + + + ## [0.39.1] - 2021-05-24 [0.39.1]: https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1 ### Changed @@ -1136,3 +1142,6 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 [0.37.3]: https://github.com/rokucommunity/brighterscript/compare/v0.37.2...v0.37.3 [0.37.4]: https://github.com/rokucommunity/brighterscript/compare/v0.37.3...v0.37.4 [0.38.0]: https://github.com/rokucommunity/brighterscript/compare/v0.37.4...v0.38.0 +[0.39.0]: https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.39.0 +[0.39.1]: https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1 +[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.1...v0.39.2 From e2cef4dfe28585beaf5458f4dbc59e44be374f1f Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 28 May 2021 17:28:15 -0400 Subject: [PATCH 029/278] fix interface issues from roku-deploy --- src/ProgramBuilder.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/ProgramBuilder.ts b/src/ProgramBuilder.ts index c06b9ec89..05225af04 100644 --- a/src/ProgramBuilder.ts +++ b/src/ProgramBuilder.ts @@ -359,6 +359,7 @@ export class ProgramBuilder { await this.logger.time(LogLevel.log, [`Creating package at ${this.options.outFile}`], async () => { await rokuDeploy.zipPackage({ ...this.options, + logLevel: this.options.logLevel as LogLevel, outDir: util.getOutDir(this.options), outFile: path.basename(this.options.outFile) }); @@ -374,6 +375,7 @@ export class ProgramBuilder { let options = util.cwdWork(this.options.cwd, () => { return rokuDeploy.getOptions({ ...this.options, + logLevel: this.options.logLevel as LogLevel, outDir: util.getOutDir(this.options), outFile: path.basename(this.options.outFile) }); @@ -417,6 +419,7 @@ export class ProgramBuilder { await this.logger.time(LogLevel.log, ['Deploying package to', this.options.host], async () => { await rokuDeploy.publish({ ...this.options, + logLevel: this.options.logLevel as LogLevel, outDir: util.getOutDir(this.options), outFile: path.basename(this.options.outFile) }); From 903296cb9e3a34675b6c3b030fa0179772308e8e Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 28 May 2021 17:29:51 -0400 Subject: [PATCH 030/278] 0.39.2 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index fadbf7821..6f8340639 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.1", + "version": "0.39.2", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index ea03d1dfe..443b5eb88 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.1", + "version": "0.39.2", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From b09f844bb1aae4f0e5b018f1b6edd97be31c2b09 Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 1 Jun 2021 23:41:29 -0400 Subject: [PATCH 031/278] roku-deploy@3.4.1 --- package-lock.json | 7 ++++--- package.json | 2 +- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/package-lock.json b/package-lock.json index 6f8340639..981facf99 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5038,10 +5038,11 @@ } }, "roku-deploy": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.0.tgz", - "integrity": "sha512-z5TTj+r4EBQnVfepUEebARB0IXmqq6ZvMRqJoSac2kfni4uofwsBMIWO4UwDaleZAAKxW6GR/YDRl/kgzfpGgw==", + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.1.tgz", + "integrity": "sha512-KomV/FvWv84f2Lh9V5c9stXCkGbM/L7B5cCLpNddBTlgh3zxyCAjEj2oFPYnzjbG+aNMvOpKatQ7Xx8bMJM03g==", "requires": { + "chalk": "^2.4.2", "dateformat": "^3.0.3", "eventemitter3": "^4.0.7", "fs-extra": "^7.0.1", diff --git a/package.json b/package.json index 443b5eb88..727d531d6 100644 --- a/package.json +++ b/package.json @@ -128,7 +128,7 @@ "moment": "^2.23.0", "p-settle": "^2.1.0", "parse-ms": "^2.1.0", - "roku-deploy": "^3.4.0", + "roku-deploy": "^3.4.1", "serialize-error": "^7.0.1", "source-map": "^0.7.3", "vscode-languageserver": "7.0.0", From 41989c50ce31f4c21dd90e2d4a6c20a47f067f4b Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 1 Jun 2021 23:41:38 -0400 Subject: [PATCH 032/278] update changelog for v0.39.3 --- CHANGELOG.md | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d7f7bb8b6..b6dd5f924 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,7 +6,15 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.39.3] - 2021-06-01 +[0.39.3]: https://github.com/RokuCommunity/roku-debug/compare/v0.8.2...v0.8.3 +### Changed + - upgraded to [roku-deploy@3.4.1](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#341---2021-06-01) which fixes bugs introduced in roku-deploy@3.4.0 + + + ## [0.39.2] - 2021-05-28 +[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.1...v0.39.2 ### Changed - upgraded to [roku-deploy@3.4.0](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#340---2021-05-28) which brings significant zip speed improvements @@ -1142,6 +1150,3 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 [0.37.3]: https://github.com/rokucommunity/brighterscript/compare/v0.37.2...v0.37.3 [0.37.4]: https://github.com/rokucommunity/brighterscript/compare/v0.37.3...v0.37.4 [0.38.0]: https://github.com/rokucommunity/brighterscript/compare/v0.37.4...v0.38.0 -[0.39.0]: https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.39.0 -[0.39.1]: https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1 -[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.1...v0.39.2 From 7a0131da975d9667d65e2c4230b6817bf7202849 Mon Sep 17 00:00:00 2001 From: Bronley Date: Tue, 1 Jun 2021 23:43:06 -0400 Subject: [PATCH 033/278] 0.39.3 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 981facf99..41425a091 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.2", + "version": "0.39.3", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 727d531d6..9386ac73a 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.2", + "version": "0.39.3", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 22b8251294a906046f6da72f2e8b6a76ab485ffd Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Thu, 10 Jun 2021 06:14:38 -0400 Subject: [PATCH 034/278] Bump glob-parent from 5.1.1 to 5.1.2 (#428) Bumps [glob-parent](https://github.com/gulpjs/glob-parent) from 5.1.1 to 5.1.2. - [Release notes](https://github.com/gulpjs/glob-parent/releases) - [Changelog](https://github.com/gulpjs/glob-parent/blob/main/CHANGELOG.md) - [Commits](https://github.com/gulpjs/glob-parent/compare/v5.1.1...v5.1.2) --- updated-dependencies: - dependency-name: glob-parent dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index 41425a091..1ef8c042e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -2702,9 +2702,9 @@ } }, "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", "requires": { "is-glob": "^4.0.1" } From 8782a0dff7de63b4669da7d4e7b97be27cb97e09 Mon Sep 17 00:00:00 2001 From: Philippe Date: Mon, 21 Jun 2021 11:53:31 +0100 Subject: [PATCH 035/278] Fix incorrect Block range for inline if/then branch (#424) * Fix incorrect Block range for inline if/then branch * Added tests Fix other tests Co-authored-by: Bronley Plumb --- src/lexer/Token.ts | 4 ++-- src/parser/Parser.ts | 5 ++-- src/parser/tests/Parser.spec.ts | 21 ++++++++++------- src/parser/tests/controlFlow/For.spec.ts | 4 ++-- src/parser/tests/controlFlow/If.spec.ts | 23 ++++++++++++++++++- src/parser/tests/expression/Call.spec.ts | 8 +++---- .../tests/statement/ReturnStatement.spec.ts | 2 +- 7 files changed, 46 insertions(+), 21 deletions(-) diff --git a/src/lexer/Token.ts b/src/lexer/Token.ts index 37166e03e..42e814123 100644 --- a/src/lexer/Token.ts +++ b/src/lexer/Token.ts @@ -10,13 +10,13 @@ export interface Token { /** The text found in the original BrightScript source, if any. */ text: string; /** True if this token's `text` is a reserved word, otherwise `false`. */ - isReserved: boolean; + isReserved?: boolean; /** Where the token was found. */ range: Range; /** * Any leading whitespace found prior to this token. Excludes newline characters. */ - leadingWhitespace: string; + leadingWhitespace?: string; } /** diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index b9be0de98..ed0d08f14 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -200,8 +200,6 @@ export class Parser { /** * Static wrapper around creating a new parser and parsing a list of tokens */ - public static parse(source: string, options?: ParseOptions): Parser; - public static parse(tokens: Token[], options?: ParseOptions): Parser; public static parse(toParse: Token[] | string, options?: ParseOptions): Parser { let tokens: Token[]; if (typeof toParse === 'string') { @@ -1637,6 +1635,7 @@ export class Parser { return undefined; } statements.push(statement); + const startingRange = statement.range; //look for colon statement separator let foundColon = false; @@ -1662,7 +1661,7 @@ export class Parser { }); } } - return new Block(statements, this.peek().range); + return new Block(statements, startingRange); } private expressionStatement(expr: Expression): ExpressionStatement | IncrementStatement { diff --git a/src/parser/tests/Parser.spec.ts b/src/parser/tests/Parser.spec.ts index d900e5a64..ce2369acb 100644 --- a/src/parser/tests/Parser.spec.ts +++ b/src/parser/tests/Parser.spec.ts @@ -1,15 +1,12 @@ import type { Token } from '../../lexer'; import { TokenKind, ReservedWords } from '../../lexer'; import { interpolatedRange } from '../../astUtils/creators'; +import type { Range } from '../../astUtils'; /* A set of utilities to be used while writing tests for the BRS parser. */ /** * Creates a token with the given `kind` and (optional) `literal` value. - * @param {TokenKind} kind the tokenKind the produced token should represent. - * @param {string} text the text represented by this token. - * @param {*} [literal] the literal value that the produced token should contain, if any - * @returns {object} a token of `kind` representing `text` with value `literal`. */ export function token(kind: TokenKind, text?: string): Token { return { @@ -23,11 +20,19 @@ export function token(kind: TokenKind, text?: string): Token { /** * Creates an Identifier token with the given `text`. - * @param {string} text - * @returns {object} a token with the provided `text`. */ -export function identifier(text) { - return exports.token(TokenKind.Identifier, text); +export function identifier(text: string) { + return token(TokenKind.Identifier, text); +} + +/** + * Test whether a range matches a group of elements with a `range` + */ +export function rangeMatch(range: Range, elements: ({ range: Range })[]): boolean { + return range.start.line === elements[0].range.start.line && + range.start.character === elements[0].range.start.character && + range.end.line === elements[elements.length - 1].range.end.line && + range.end.character === elements[elements.length - 1].range.end.character; } /** An end-of-file token. */ diff --git a/src/parser/tests/controlFlow/For.spec.ts b/src/parser/tests/controlFlow/For.spec.ts index c86305c96..684cbd198 100644 --- a/src/parser/tests/controlFlow/For.spec.ts +++ b/src/parser/tests/controlFlow/For.spec.ts @@ -128,8 +128,8 @@ describe('parser for loops', () => { text: 'to', isReserved: false, range: { - start: { line: 0, column: 10 }, - end: { start: 0, column: 12 } + start: { line: 0, character: 10 }, + end: { line: 0, character: 12 } } }, { diff --git a/src/parser/tests/controlFlow/If.spec.ts b/src/parser/tests/controlFlow/If.spec.ts index 53d249fcf..d3252cafb 100644 --- a/src/parser/tests/controlFlow/If.spec.ts +++ b/src/parser/tests/controlFlow/If.spec.ts @@ -3,8 +3,9 @@ import * as assert from 'assert'; import { Parser } from '../../Parser'; import { TokenKind, Lexer } from '../../../lexer'; -import { EOF, identifier, token } from '../Parser.spec'; +import { EOF, identifier, rangeMatch, token } from '../Parser.spec'; import { isBlock, isCommentStatement, isIfStatement } from '../../../astUtils'; +import type { Block, IfStatement } from '../../Statement'; describe('parser if statements', () => { it('allows empty if blocks', () => { @@ -602,4 +603,24 @@ describe('parser if statements', () => { expect(diagnostics).to.be.lengthOf(0); expect(statements).to.be.length.greaterThan(0); }); + + it('single-line if block statements have correct range', () => { + let { tokens } = Lexer.scan(` + if false then print "true" + if false then print "true": a = 10 + if false then print "true" else print "false" + if false then print "true" else print "false": a = 20 + `); + let { statements, diagnostics } = Parser.parse(tokens); + expect(diagnostics).to.be.lengthOf(0); + + const then1 = (statements[0] as IfStatement).thenBranch; + expect(rangeMatch(then1.range, then1.statements)).to.be.true; + const then2 = (statements[1] as IfStatement).thenBranch; + expect(rangeMatch(then2.range, then2.statements)).to.be.true; + const else1 = (statements[2] as IfStatement).elseBranch as Block; + expect(rangeMatch(else1.range, else1.statements)).to.be.true; + const else2 = (statements[3] as IfStatement).elseBranch as Block; + expect(rangeMatch(else2.range, else2.statements)).to.be.true; + }); }); diff --git a/src/parser/tests/expression/Call.spec.ts b/src/parser/tests/expression/Call.spec.ts index 770a06ab1..136188673 100644 --- a/src/parser/tests/expression/Call.spec.ts +++ b/src/parser/tests/expression/Call.spec.ts @@ -9,7 +9,7 @@ describe('parser call expressions', () => { it('parses named function calls', () => { const { statements, diagnostics } = Parser.parse([ identifier('RebootSystem'), - { kind: TokenKind.LeftParen, text: '(', line: 1 }, + { kind: TokenKind.LeftParen, text: '(', range: null }, token(TokenKind.RightParen, ')'), EOF ]); @@ -57,7 +57,7 @@ describe('parser call expressions', () => { it('allows closing parentheses on separate line', () => { const { statements, diagnostics } = Parser.parse([ identifier('RebootSystem'), - { kind: TokenKind.LeftParen, text: '(', line: 1 }, + { kind: TokenKind.LeftParen, text: '(', range: null }, token(TokenKind.Newline, '\\n'), token(TokenKind.Newline, '\\n'), token(TokenKind.RightParen, ')'), @@ -71,9 +71,9 @@ describe('parser call expressions', () => { it('accepts arguments', () => { const { statements, diagnostics } = Parser.parse([ identifier('add'), - { kind: TokenKind.LeftParen, text: '(', line: 1 }, + { kind: TokenKind.LeftParen, text: '(', range: null }, token(TokenKind.IntegerLiteral, '1'), - { kind: TokenKind.Comma, text: ',', line: 1 }, + { kind: TokenKind.Comma, text: ',', range: null }, token(TokenKind.IntegerLiteral, '2'), token(TokenKind.RightParen, ')'), EOF diff --git a/src/parser/tests/statement/ReturnStatement.spec.ts b/src/parser/tests/statement/ReturnStatement.spec.ts index e6123b45e..9974418b2 100644 --- a/src/parser/tests/statement/ReturnStatement.spec.ts +++ b/src/parser/tests/statement/ReturnStatement.spec.ts @@ -51,7 +51,7 @@ describe('parser return statements', () => { token(TokenKind.Newline, '\\n'), token(TokenKind.Return, 'return'), identifier('RebootSystem'), - { kind: TokenKind.LeftParen, text: '(', line: 2 }, + { kind: TokenKind.LeftParen, text: '(', range: null }, token(TokenKind.RightParen, ')'), token(TokenKind.Newline, '\\n'), token(TokenKind.EndFunction, 'end function'), From e63656f35ff50b038bebfe5ef32e524a0dea732d Mon Sep 17 00:00:00 2001 From: Philippe Date: Tue, 22 Jun 2021 11:38:01 +0100 Subject: [PATCH 036/278] Extract AA comma when parsing (#427) * Extract AA comma when parsing * Missing comma when following by comment * Added a test * linting Co-authored-by: Bronley Plumb --- src/parser/Expression.ts | 1 + src/parser/Parser.ts | 21 ++++++++++----- .../AssociativeArrayLiterals.spec.ts | 27 +++++++++++++++++++ 3 files changed, 43 insertions(+), 6 deletions(-) diff --git a/src/parser/Expression.ts b/src/parser/Expression.ts index 07790e7f7..aef007f9a 100644 --- a/src/parser/Expression.ts +++ b/src/parser/Expression.ts @@ -629,6 +629,7 @@ export class AAMemberExpression extends Expression { } public range: Range; + public commaToken?: Token; transpile(state: BrsTranspileState) { //TODO move the logic from AALiteralExpression loop into this function diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index ed0d08f14..09fcc17e7 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -2317,30 +2317,38 @@ export class Parser { } if (!this.match(TokenKind.RightCurlyBrace)) { + let lastAAMember: AAMemberExpression; if (this.check(TokenKind.Comment)) { + lastAAMember = null; members.push(new CommentStatement([this.advance()])); } else { let k = key(); let expr = this.expression(); - members.push(new AAMemberExpression( + lastAAMember = new AAMemberExpression( k.keyToken, k.colonToken, expr - )); + ); + members.push(lastAAMember); } while (this.matchAny(TokenKind.Comma, TokenKind.Newline, TokenKind.Colon, TokenKind.Comment)) { + // collect comma at end of expression + if (lastAAMember && this.checkPrevious(TokenKind.Comma)) { + lastAAMember.commaToken = this.previous(); + } + //check for comment at the end of the current line if (this.check(TokenKind.Comment) || this.checkPrevious(TokenKind.Comment)) { let token = this.checkPrevious(TokenKind.Comment) ? this.previous() : this.advance(); members.push(new CommentStatement([token])); } else { - while (this.matchAny(TokenKind.Newline, TokenKind.Colon)) { + this.consumeStatementSeparators(true); - } //check for a comment on its own line if (this.check(TokenKind.Comment) || this.checkPrevious(TokenKind.Comment)) { let token = this.checkPrevious(TokenKind.Comment) ? this.previous() : this.advance(); + lastAAMember = null; members.push(new CommentStatement([token])); continue; } @@ -2350,11 +2358,12 @@ export class Parser { } let k = key(); let expr = this.expression(); - members.push(new AAMemberExpression( + lastAAMember = new AAMemberExpression( k.keyToken, k.colonToken, expr - )); + ); + members.push(lastAAMember); } } diff --git a/src/parser/tests/expression/AssociativeArrayLiterals.spec.ts b/src/parser/tests/expression/AssociativeArrayLiterals.spec.ts index b427b5896..738dd67d4 100644 --- a/src/parser/tests/expression/AssociativeArrayLiterals.spec.ts +++ b/src/parser/tests/expression/AssociativeArrayLiterals.spec.ts @@ -4,6 +4,9 @@ import { Parser } from '../../Parser'; import { TokenKind } from '../../../lexer'; import { EOF, identifier, token } from '../Parser.spec'; import { Range } from 'vscode-languageserver'; +import type { AssignmentStatement } from '../../Statement'; +import type { AALiteralExpression } from '../../Expression'; +import { isCommentStatement } from '../../../astUtils'; describe('parser associative array literals', () => { describe('empty associative arrays', () => { @@ -178,6 +181,30 @@ describe('parser associative array literals', () => { expect(statements).to.be.length.greaterThan(0); }); + it('captures commas', () => { + let { statements } = Parser.parse(` + _ = { + p1: 1, + p2: 2, 'comment + p3: 3 + p4: 4 + 'comment + p5: 5, + } + `); + const commas = ((statements[0] as AssignmentStatement).value as AALiteralExpression).elements + .map(s => !isCommentStatement(s) && !!s.commaToken); + expect(commas).to.deep.equal([ + true, // p1 + true, // p2 + false, // comment + false, // p3 + false, // p4 + false, // comment + true // p5 + ]); + }); + it('location tracking', () => { /** * 0 0 0 1 From 5034735213240013c5f4a16e3d0225dbf7842ef4 Mon Sep 17 00:00:00 2001 From: Mark Pearce Date: Fri, 25 Jun 2021 21:41:34 -0300 Subject: [PATCH 037/278] Allow up to 6 arguements to CreateObject to support roRegion - fixes #430 (#432) Co-authored-by: Mark Pearce --- src/files/BrsFile.spec.ts | 10 ++++++++++ src/globalCallables.ts | 8 ++++++++ 2 files changed, 18 insertions(+) diff --git a/src/files/BrsFile.spec.ts b/src/files/BrsFile.spec.ts index bccd7baf4..e033a3f83 100644 --- a/src/files/BrsFile.spec.ts +++ b/src/files/BrsFile.spec.ts @@ -49,6 +49,16 @@ describe('BrsFile', () => { expect(program.getDiagnostics()[0]?.message).to.not.exist; }); + it('supports the 6 params in CreateObject for roRegion', () => { + program.addOrReplaceFile('source/main.brs', ` + sub createRegion(bitmap as object) + region = CreateObject("roRegion", bitmap, 20, 40, 100, 200) + end sub + `); + program.validate(); + expect(program.getDiagnostics()[0]?.message).to.not.exist; + }); + it('sets needsTranspiled to true for .bs files', () => { //BrightScript expect(new BrsFile(`${rootDir}/source/main.brs`, 'source/main.brs', program).needsTranspiled).to.be.false; diff --git a/src/globalCallables.ts b/src/globalCallables.ts index 4c98fafb3..e620b9b68 100644 --- a/src/globalCallables.ts +++ b/src/globalCallables.ts @@ -190,6 +190,14 @@ let runtimeFunctions = [{ name: 'param4', type: new DynamicType(), isOptional: true + }, { + name: 'param5', + type: new DynamicType(), + isOptional: true + }, { + name: 'param6', + type: new DynamicType(), + isOptional: true }] }, { name: 'Type', From c040a906d72283b2899c7996b1f916c74cec841a Mon Sep 17 00:00:00 2001 From: Mark Pearce Date: Fri, 25 Jun 2021 22:30:06 -0300 Subject: [PATCH 038/278] Adds a bunch of functions from v30/bslCore library to the global callables list (#433) Co-authored-by: Mark Pearce --- src/globalCallables.spec.ts | 16 ++++ src/globalCallables.ts | 148 ++++++++++++++++++++++++++++++++++++ 2 files changed, 164 insertions(+) diff --git a/src/globalCallables.spec.ts b/src/globalCallables.spec.ts index f9aeacf30..8d2ab2add 100644 --- a/src/globalCallables.spec.ts +++ b/src/globalCallables.spec.ts @@ -31,6 +31,22 @@ describe('globalCallables', () => { }); }); + describe('bslCore', () => { + it('exists', () => { + program.addOrReplaceFile('source/main.brs', ` + Library "v30/bslCore.brs" + + sub main() + print bslBrightScriptErrorCodes() + print bslUniversalControlEventCodes() + print HexToAscii(AsciiToHex("Hi")) + end sub + `); + program.validate(); + expectZeroDiagnostics(program); + }); + }); + describe('val', () => { it('allows single parameter', () => { program.addOrReplaceFile('source/main.brs', ` diff --git a/src/globalCallables.ts b/src/globalCallables.ts index e620b9b68..cc8ff8991 100644 --- a/src/globalCallables.ts +++ b/src/globalCallables.ts @@ -743,6 +743,154 @@ let programStatementFunctions = [ type: new FunctionType(new ObjectType()), file: globalFile, params: [] + }, { //TODO Same as the Roku_Ads.brs (RAF) library above, the following functions are from the 'v30/bslCore.brs' library + name: 'bslBrightScriptErrorCodes', + shortDescription: 'Returns an roAssociativeArray with name value pairs of the error name and corresponding integer value, for example ERR_OKAY = &hFF.', + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [] + }, { + name: 'bslGeneralConstants', + shortDescription: 'Returns an roAssociativeArray with name value pairs of system constants, for example MAX_INT = 2147483647.', + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [] + }, { + name: 'bslUniversalControlEventCodes', + shortDescription: 'Returns an roAssociativeArray with name value pairs of the remote key code (buttons) constants, for example BUTTON_SELECT_PRESSED = 6.', + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [] + }, + { + name: 'AsciiToHex', + shortDescription: 'Returns the hex encoded string, for example AsciiToHex("Hi!") = "486921".', + type: new FunctionType(new StringType()), + file: globalFile, + params: [{ + name: 'ascii', + type: new StringType() + }] + }, { + name: 'HexToAscii', + shortDescription: 'Returns a string that is the hex decoded string, for example HexToAscii("486921") = "Hi!".', + type: new FunctionType(new StringType()), + file: globalFile, + params: [{ + name: 'hex', + type: new StringType() + }] + }, + { + name: 'HexToInteger', + shortDescription: 'Returns the integer value of the passed in hex string.', + type: new FunctionType(new IntegerType()), + file: globalFile, + params: [{ + name: 'hex', + type: new StringType() + }] + }, { + name: 'HexToInteger', + shortDescription: 'Returns a string that is the hex decoded string, for example HexToAscii("486921") = "Hi!".', + type: new FunctionType(new IntegerType()), + file: globalFile, + params: [{ + name: 'hex', + type: new StringType() + }] + }, { + name: 'dfNewBitmapSet', + shortDescription: `The goal is to enable simple xml descriptions of graphics resources like bitmaps, regions, sprites, animations, and layouts to be used in your games. The library handles parsing, loading, rendering, and animation of sprites sheets (multiple images in a single png file). +Filename is the path to an XML file that contains info about bitmap regions, animation frames, and ExtraInfo metadata (any fields you would like) about resources used in 2d games. +Returns an roAssociativeArray with the following name value pairs: +ExtraInfo: roAssociativeArray +Regions: roAssociativeArray of name, roRegions pairs +Animations: roAssociativeArray of name, roArray of roRegion pairs. +Backgrounds: roAssociativeArray of name, path pairs.`, + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [{ + name: 'filename', + type: new StringType() + }] + }, { + name: 'dfDrawMessage', + shortDescription: 'dest is an roScreen/roBitmap/roRegion and region is an roRegion.\nGreys the entire dest region and draws it the region centered on the drawable dest.', + type: new FunctionType(new VoidType()), + file: globalFile, + params: [{ + name: 'dest', + type: new ObjectType() + }, { + name: 'region', + type: new ObjectType() + }] + }, { + name: 'dfDrawImage', + shortDescription: 'Returns True if successful.\nCreates a bitmap out of the image stored in the filename "path" and draws it at position (x,y) of the drawable dest.', + type: new FunctionType(new BooleanType()), + file: globalFile, + params: [{ + name: 'dest', + type: new ObjectType() + }, { + name: 'path', + type: new StringType() + }, { + name: 'x', + type: new IntegerType() + }, { + name: 'y', + type: new IntegerType() + }] + }, { + name: 'dfSetupDisplayRegions', + shortDescription: `Helper function to setup screen scaling with supplied pillar box or letterbox images to fill the entire screen. +screen is an roScreen +topx and topy are the coordinates of the upper left hand corner of the main drawing region +Width and height is the size of the main drawing region + +Returns an associative array containing the following roRegions +Main: main drawable region +Left: left region if there is pillar box area on the left +Right: right region if there is a pillar box area on the right +Upper: upper region if there is a letterbox area at thetop +Lower: lower region if there is a letterbox area at the bottom +When using these regions as drawables, your graphics will be translated and clipped to these regions.`, + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [{ + name: 'screen', + type: new ObjectType() + }, { + name: 'topx', + type: new IntegerType() + }, { + name: 'topy', + type: new IntegerType() + }, { + name: 'width', + type: new IntegerType() + }, { + name: 'height', + type: new IntegerType() + }] + }, { + name: 'dfSetBackground', + shortDescription: `dfSetBackground helps manage the limited video memory. The video memory does not currently run a defragmenter, and is very limited. These constraints make it important that large bitmaps (like backgrounds that fill the entire screen) are only allocated when needed. It is also helpful if you set your first initial background very early in your program, and then immediately replace the background after it is no longer in use. This helper function supports this background management for your application. +backgroundName is a key for the Backgrounds roAssociative array of backgrounds. +Backgrounds is an roAssociative array of background name keys and file path string values +This function creates an roBitmap out of the background image file and returns a region the size of the entire roBitmap.`, + type: new FunctionType(new ObjectType()), + file: globalFile, + params: [{ + name: 'backgroundName', + type: new StringType() + }, { + name: 'backgrounds', + type: new ObjectType() + }] } ] as Callable[]; From d37447592230a158de9f529ad3e42266f6330a9d Mon Sep 17 00:00:00 2001 From: Bronley Date: Sun, 27 Jun 2021 21:53:36 -0400 Subject: [PATCH 039/278] update changelog for v0.39.4 --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index b6dd5f924..86ff7dda3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,15 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.39.4] - 2021-06-27 +[0.39.4]: https://github.com/RokuCommunity/roku-debug/compare/v0.8.3...v0.8.4 +### Fixed + - incorrect block range for inline if/then branch ([#424](https://github.com/rokucommunity/brighterscript/pull/424)) + - extract associative array comma when parsing ([#427](https://github.com/rokucommunity/brighterscript/pull/424)) + - allow up to 6 arguments in `CreateObject` function signature ([#430](https://github.com/rokucommunity/brighterscript/pull/430)) + - add v30/bslCore library functions to global callables ([#433](https://github.com/rokucommunity/brighterscript/pull/433)) + + ## [0.39.3] - 2021-06-01 [0.39.3]: https://github.com/RokuCommunity/roku-debug/compare/v0.8.2...v0.8.3 ### Changed From d5ae729f95c63a166a1c3d935ddf096cb857c14b Mon Sep 17 00:00:00 2001 From: Bronley Date: Sun, 27 Jun 2021 21:55:28 -0400 Subject: [PATCH 040/278] 0.39.4 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 1ef8c042e..c2dbc4fc0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.3", + "version": "0.39.4", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 9386ac73a..629820c85 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.3", + "version": "0.39.4", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 49cbccdd8e1214043e9eccd4bb750bf7542477ff Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Mon, 28 Jun 2021 21:33:17 -0400 Subject: [PATCH 041/278] Adds language support for Interface statements (#426) * Add InterfaceType for type checking * Add basic support for interface statements. * Fix iface annotation capturing. Add namespace support * fix lint issues. * Add `isInterfaceStatement` reflection check * Support `interface`, `class`, `namespace` as local vars. Fix broken test. --- src/DiagnosticMessages.ts | 12 +- src/astUtils/reflection.ts | 19 +- src/lexer/TokenKind.ts | 9 +- src/parser/Parser.Class.spec.ts | 2 +- src/parser/Parser.spec.ts | 2 +- src/parser/Parser.ts | 219 ++++++++++++- src/parser/Statement.ts | 290 +++++++++++++++++- .../statement/InterfaceStatement.spec.ts | 64 ++++ src/parser/tests/statement/Misc.spec.ts | 38 ++- src/testHelpers.spec.ts | 33 +- src/types/InterfaceType.spec.ts | 208 +++++++++++++ src/types/InterfaceType.ts | 62 +++- 12 files changed, 897 insertions(+), 61 deletions(-) create mode 100644 src/parser/tests/statement/InterfaceStatement.spec.ts create mode 100644 src/types/InterfaceType.spec.ts diff --git a/src/DiagnosticMessages.ts b/src/DiagnosticMessages.ts index 07a5b577e..c547a4d71 100644 --- a/src/DiagnosticMessages.ts +++ b/src/DiagnosticMessages.ts @@ -175,8 +175,8 @@ export let DiagnosticMessages = { code: 1031, severity: DiagnosticSeverity.Error }), - expectedClassKeyword: () => ({ - message: `Expected 'class' keyword`, + expectedKeyword: (kind: TokenKind) => ({ + message: `Expected '${kind}' keyword`, code: 1032, severity: DiagnosticSeverity.Error }), @@ -423,8 +423,8 @@ export let DiagnosticMessages = { code: 1080, severity: DiagnosticSeverity.Error }), - foundUnexpectedToken: (text: string) => ({ - message: `Found unexpected token '${text}'`, + unexpectedToken: (text: string) => ({ + message: `Unexpected token '${text}'`, code: 1081, severity: DiagnosticSeverity.Error }), @@ -566,8 +566,8 @@ export let DiagnosticMessages = { code: 1108, severity: DiagnosticSeverity.Error }), - expectedTokenAButFoundTokenB: (tokenA: string, tokenB: string) => ({ - message: `Expected '${tokenA}' but instead found ${tokenB}`, + expectedToken: (tokenKind: string) => ({ + message: `Expected '${tokenKind}'`, code: 1109, severity: DiagnosticSeverity.Error }), diff --git a/src/astUtils/reflection.ts b/src/astUtils/reflection.ts index d04d4591e..11726fd11 100644 --- a/src/astUtils/reflection.ts +++ b/src/astUtils/reflection.ts @@ -1,4 +1,4 @@ -import type { Body, AssignmentStatement, Block, ExpressionStatement, CommentStatement, ExitForStatement, ExitWhileStatement, FunctionStatement, IfStatement, IncrementStatement, PrintStatement, GotoStatement, LabelStatement, ReturnStatement, EndStatement, StopStatement, ForStatement, ForEachStatement, WhileStatement, DottedSetStatement, IndexedSetStatement, LibraryStatement, NamespaceStatement, ImportStatement, ClassFieldStatement, ClassMethodStatement, ClassStatement, Statement } from '../parser/Statement'; +import type { Body, AssignmentStatement, Block, ExpressionStatement, CommentStatement, ExitForStatement, ExitWhileStatement, FunctionStatement, IfStatement, IncrementStatement, PrintStatement, GotoStatement, LabelStatement, ReturnStatement, EndStatement, StopStatement, ForStatement, ForEachStatement, WhileStatement, DottedSetStatement, IndexedSetStatement, LibraryStatement, NamespaceStatement, ImportStatement, ClassFieldStatement, ClassMethodStatement, ClassStatement, Statement, InterfaceFieldStatement, InterfaceMethodStatement, InterfaceStatement } from '../parser/Statement'; import type { LiteralExpression, Expression, BinaryExpression, CallExpression, FunctionExpression, NamespacedVariableNameExpression, DottedGetExpression, XmlAttributeGetExpression, IndexedGetExpression, GroupingExpression, EscapedCharCodeLiteralExpression, ArrayLiteralExpression, AALiteralExpression, UnaryExpression, VariableExpression, SourceLiteralExpression, NewExpression, CallfuncExpression, TemplateStringQuasiExpression, TemplateStringExpression, TaggedTemplateStringExpression, AnnotationExpression, FunctionParameterExpression } from '../parser/Expression'; import type { BrsFile } from '../files/BrsFile'; import type { XmlFile } from '../files/XmlFile'; @@ -17,6 +17,8 @@ import { CustomType } from '../types/CustomType'; import type { Scope } from '../Scope'; import type { XmlScope } from '../XmlScope'; import { DynamicType } from '../types/DynamicType'; +import type { InterfaceType } from '../types/InterfaceType'; +import type { ObjectType } from '../types/ObjectType'; // File reflection @@ -127,6 +129,15 @@ export function isClassMethodStatement(element: Statement | Expression | undefin export function isClassFieldStatement(element: Statement | Expression | undefined): element is ClassFieldStatement { return element?.constructor.name === 'ClassFieldStatement'; } +export function isInterfaceStatement(element: Statement | Expression | undefined): element is InterfaceStatement { + return element?.constructor.name === 'InterfaceStatement'; +} +export function isInterfaceMethodStatement(element: Statement | Expression | undefined): element is InterfaceMethodStatement { + return element?.constructor.name === 'InterfaceMethodStatement'; +} +export function isInterfaceFieldStatement(element: Statement | Expression | undefined): element is InterfaceFieldStatement { + return element?.constructor.name === 'InterfaceFieldStatement'; +} // Expressions reflection /** @@ -242,6 +253,12 @@ export function isCustomType(e: any): e is CustomType { export function isDynamicType(e: any): e is DynamicType { return e?.constructor.name === DynamicType.name; } +export function isInterfaceType(e: any): e is InterfaceType { + return e?.constructor.name === 'InterfaceType'; +} +export function isObjectType(e: any): e is ObjectType { + return e?.constructor.name === 'ObjectType'; +} const numberConstructorNames = [ IntegerType.name, diff --git a/src/lexer/TokenKind.ts b/src/lexer/TokenKind.ts index 8719ede68..f01bb4b0c 100644 --- a/src/lexer/TokenKind.ts +++ b/src/lexer/TokenKind.ts @@ -153,6 +153,7 @@ export enum TokenKind { New = 'New', Override = 'Override', Import = 'Import', + EndInterface = 'EndInterface', //brighterscript source literals LineNumLiteral = 'LineNumLiteral', @@ -304,7 +305,9 @@ export const Keywords: Record = { catch: TokenKind.Catch, endtry: TokenKind.EndTry, 'end try': TokenKind.EndTry, - throw: TokenKind.Throw + throw: TokenKind.Throw, + 'end interface': TokenKind.EndInterface, + endinterface: TokenKind.EndInterface }; //hide the constructor prototype method because it causes issues Keywords.constructor = undefined; @@ -319,6 +322,7 @@ export type BlockTerminator = | TokenKind.EndSub | TokenKind.EndFunction | TokenKind.EndNamespace + | TokenKind.EndInterface | TokenKind.Catch | TokenKind.EndTry; @@ -423,7 +427,8 @@ export const AllowedProperties = [ TokenKind.Try, TokenKind.Catch, TokenKind.EndTry, - TokenKind.Throw + TokenKind.Throw, + TokenKind.EndInterface ]; /** List of TokenKind that are allowed as local var identifiers. */ diff --git a/src/parser/Parser.Class.spec.ts b/src/parser/Parser.Class.spec.ts index 2abe5d5f7..ac1e78489 100644 --- a/src/parser/Parser.Class.spec.ts +++ b/src/parser/Parser.Class.spec.ts @@ -78,7 +78,7 @@ describe('parser class', () => { end sub `); - expect(parser.diagnostics[0]?.message).to.eql(DiagnosticMessages.foundUnexpectedToken('=').message); + expect(parser.diagnostics[0]?.message).to.eql(DiagnosticMessages.unexpectedToken('=').message); }); it('does not allow function named "throw"', () => { diff --git a/src/parser/Parser.spec.ts b/src/parser/Parser.spec.ts index c4d6fa7dc..90bc1c9d4 100644 --- a/src/parser/Parser.spec.ts +++ b/src/parser/Parser.spec.ts @@ -679,7 +679,7 @@ describe('parser', () => { sub main() end sub `, ParseMode.BrighterScript); - expect(diagnostics[0]?.message).to.equal(DiagnosticMessages.foundUnexpectedToken('@').message); + expect(diagnostics[0]?.message).to.equal(DiagnosticMessages.unexpectedToken('@').message); }); it('properly handles empty annotation above class method', () => { diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index 09fcc17e7..795d7886b 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -15,12 +15,16 @@ import { isToken, DeclarableTypes } from '../lexer'; + import type { Statement, PrintSeparatorTab, PrintSeparatorSpace } from './Statement'; import { + InterfaceStatement, + InterfaceMethodStatement, + InterfaceFieldStatement, AssignmentStatement, Block, Body, @@ -310,10 +314,6 @@ export class Parser { private declaration(): Statement | AnnotationExpression | undefined { try { - if (this.check(TokenKind.Class)) { - return this.classDeclaration(); - } - if (this.checkAny(TokenKind.Sub, TokenKind.Function)) { return this.functionDeclaration(false); } @@ -322,10 +322,6 @@ export class Parser { return this.libraryStatement(); } - if (this.check(TokenKind.Namespace)) { - return this.namespaceStatement(); - } - if (this.check(TokenKind.At) && this.checkNext(TokenKind.Identifier)) { return this.annotationExpression(); } @@ -349,6 +345,156 @@ export class Parser { } } + private identifier(...additionalTokenKinds: TokenKind[]) { + const identifier = this.consume( + DiagnosticMessages.expectedIdentifier(), + TokenKind.Identifier, + ...additionalTokenKinds + ) as Identifier; + // force the name into an identifier so the AST makes some sense + identifier.kind = TokenKind.Identifier; + return identifier; + } + + /** + * Create a new InterfaceMethodStatement. This should only be called from within `interfaceDeclaration` + */ + private interfaceFieldStatement() { + const name = this.identifier(...AllowedProperties); + let asToken = this.consumeToken(TokenKind.As); + let typeToken = this.typeToken(); + const type = util.tokenToBscType(typeToken); + + if (!type) { + this.diagnostics.push({ + ...DiagnosticMessages.functionParameterTypeIsInvalid(name.text, typeToken.text), + range: typeToken.range + }); + throw this.lastDiagnosticAsError(); + } + + return new InterfaceFieldStatement(name, asToken, typeToken, type); + } + + /** + * Create a new InterfaceMethodStatement. This should only be called from within `interfaceDeclaration()` + */ + private interfaceMethodStatement() { + const functionType = this.advance(); + const name = this.identifier(...AllowedProperties); + const leftParen = this.consumeToken(TokenKind.LeftParen); + + const params = []; + const rightParen = this.consumeToken(TokenKind.RightParen); + let asToken = null as Token; + let returnTypeToken = null as Token; + if (this.check(TokenKind.As)) { + asToken = this.advance(); + returnTypeToken = this.typeToken(); + const returnType = util.tokenToBscType(returnTypeToken); + if (!returnType) { + this.diagnostics.push({ + ...DiagnosticMessages.functionParameterTypeIsInvalid(name.text, returnTypeToken.text), + range: returnTypeToken.range + }); + throw this.lastDiagnosticAsError(); + } + } + + return new InterfaceMethodStatement( + functionType, + name, + leftParen, + params, + rightParen, + asToken, + returnTypeToken, + util.tokenToBscType(returnTypeToken) + ); + } + + private interfaceDeclaration(): InterfaceStatement { + this.warnIfNotBrighterScriptMode('interface declarations'); + + const parentAnnotations = this.enterAnnotationBlock(); + + const interfaceToken = this.consume( + DiagnosticMessages.expectedKeyword(TokenKind.Interface), + TokenKind.Interface + ); + const nameToken = this.identifier(); + + let extendsToken: Token; + let parentInterfaceName: NamespacedVariableNameExpression; + + if (this.peek().text.toLowerCase() === 'extends') { + extendsToken = this.advance(); + parentInterfaceName = this.getNamespacedVariableNameExpression(); + } + this.consumeStatementSeparators(); + //gather up all interface members (Fields, Methods) + let body = [] as Statement[]; + while (this.checkAny(TokenKind.Comment, TokenKind.Identifier, TokenKind.At, ...AllowedProperties)) { + try { + let decl: Statement; + + //collect leading annotations + if (this.check(TokenKind.At)) { + this.annotationExpression(); + } + + //fields + if (this.checkAny(TokenKind.Identifier, ...AllowedProperties) && this.checkNext(TokenKind.As)) { + decl = this.interfaceFieldStatement(); + + //methods (function/sub keyword followed by opening paren) + } else if (this.checkAny(TokenKind.Function, TokenKind.Sub) && this.checkAny(TokenKind.Identifier, ...AllowedProperties)) { + decl = this.interfaceMethodStatement(); + + //comments + } else if (this.check(TokenKind.Comment)) { + decl = this.commentStatement(); + } + + if (decl) { + this.consumePendingAnnotations(decl); + body.push(decl); + } else { + //we didn't find a declaration...flag tokens until next line + this.flagUntil(TokenKind.Newline, TokenKind.Colon, TokenKind.Eof); + } + } catch (e) { + //throw out any failed members and move on to the next line + this.flagUntil(TokenKind.Newline, TokenKind.Colon, TokenKind.Eof); + } + + //ensure statement separator + this.consumeStatementSeparators(); + //break out of this loop if we encountered the `EndInterface` token not followed by `as` + if (this.check(TokenKind.EndInterface) && !this.checkNext(TokenKind.As)) { + break; + } + } + + //consume the final `end interface` token + const endInterfaceToken = this.consumeToken(TokenKind.EndInterface); + + this.consumeStatementSeparators(); + + const statement = new InterfaceStatement( + interfaceToken, + nameToken, + extendsToken, + parentInterfaceName, + body, + endInterfaceToken, + this.currentNamespaceName + ); + this._references.interfaceStatements.push(statement); + this.exitAnnotationBlock(parentAnnotations); + return statement; + } + /** * A BrighterScript class declaration */ @@ -358,7 +504,7 @@ export class Parser { const parentAnnotations = this.enterAnnotationBlock(); let classKeyword = this.consume( - DiagnosticMessages.expectedClassKeyword(), + DiagnosticMessages.expectedKeyword(TokenKind.Class), TokenKind.Class ); let extendsKeyword: Token; @@ -879,6 +1025,19 @@ export class Parser { return this.assignment(); } + //some BrighterScript keywords are allowed as a local identifiers, so we need to check for them AFTER the assignment check + if (this.check(TokenKind.Interface)) { + return this.interfaceDeclaration(); + } + + if (this.check(TokenKind.Class)) { + return this.classDeclaration(); + } + + if (this.check(TokenKind.Namespace)) { + return this.namespaceStatement(); + } + // TODO: support multi-statements return this.setStatement(); } @@ -1097,7 +1256,7 @@ export class Parser { //consume multiple dot identifiers (i.e. `Name.Space.Can.Have.Many.Parts`) while (this.check(TokenKind.Dot)) { let dot = this.tryConsume( - DiagnosticMessages.foundUnexpectedToken(this.peek().text), + DiagnosticMessages.unexpectedToken(this.peek().text), TokenKind.Dot ); if (!dot) { @@ -1128,7 +1287,7 @@ export class Parser { while (!this.checkAny(...stopTokens) && !this.isAtEnd()) { let token = this.advance(); this.diagnostics.push({ - ...DiagnosticMessages.foundUnexpectedToken(token.text), + ...DiagnosticMessages.unexpectedToken(token.text), range: token.range }); } @@ -1216,7 +1375,7 @@ export class Parser { this.advance(); } - const colonToken = this.tryConsume(DiagnosticMessages.expectedTokenAButFoundTokenB(TokenKind.Colon, this.peek().text), TokenKind.Colon); + const colonToken = this.tryConsumeToken(TokenKind.Colon); //consume newlines while (this.checkAny(TokenKind.Newline, TokenKind.Comment)) { @@ -1656,7 +1815,7 @@ export class Parser { //error: colon before next keyword const colon = this.previous(); this.diagnostics.push({ - ...DiagnosticMessages.foundUnexpectedToken(colon.text), + ...DiagnosticMessages.unexpectedToken(colon.text), range: colon.range }); } @@ -2068,7 +2227,7 @@ export class Parser { let nameExpr = this.getNamespacedVariableNameExpression(); let leftParen = this.consume( - DiagnosticMessages.foundUnexpectedToken(this.peek().text), + DiagnosticMessages.unexpectedToken(this.peek().text), TokenKind.LeftParen ); let call = this.finishCall(leftParen, nameExpr); @@ -2186,7 +2345,7 @@ export class Parser { /** * Tries to get the next token as a type * Allows for built-in types (double, string, etc.) or namespaced custom types in Brighterscript mode - * Will always return a token of whatever is next to be parsed + * Will return a token of whatever is next to be parsed (unless `advanceIfUnknown` is false, in which case undefined will be returned instead */ private typeToken(): Token { let typeToken: Token; @@ -2395,7 +2554,7 @@ export class Parser { //something went wrong...throw an error so the upstream processor can scrap this line and move on } else { this.diagnostics.push({ - ...DiagnosticMessages.foundUnexpectedToken(this.peek().text), + ...DiagnosticMessages.unexpectedToken(this.peek().text), range: this.peek().range }); throw this.lastDiagnosticAsError(); @@ -2442,6 +2601,13 @@ export class Parser { } } + private consumeToken(tokenKind: TokenKind) { + return this.consume( + DiagnosticMessages.expectedToken(tokenKind), + tokenKind + ); + } + /** * Consume, or add a message if not found. But then continue and return undefined */ @@ -2458,6 +2624,13 @@ export class Parser { }); } + private tryConsumeToken(tokenKind: TokenKind) { + return this.tryConsume( + DiagnosticMessages.expectedToken(tokenKind), + tokenKind + ); + } + private consumeStatementSeparators(optional = false) { //a comment or EOF mark the end of the statement if (this.isAtEnd() || this.check(TokenKind.Comment)) { @@ -2671,6 +2844,20 @@ export class References { return this._functionStatementLookup; } private _functionStatementLookup: Map; + + public interfaceStatements = [] as InterfaceStatement[]; + + public get interfaceStatementLookup() { + if (!this._interfaceStatementLookup) { + this._interfaceStatementLookup = new Map(); + for (const stmt of this.interfaceStatements) { + this._interfaceStatementLookup.set(stmt.fullName.toLowerCase(), stmt); + } + } + return this._interfaceStatementLookup; + } + private _interfaceStatementLookup: Map; + public importStatements = [] as ImportStatement[]; public libraryStatements = [] as LibraryStatement[]; public namespaceStatements = [] as NamespaceStatement[]; diff --git a/src/parser/Statement.ts b/src/parser/Statement.ts index 8e389a7d0..21c1cb503 100644 --- a/src/parser/Statement.ts +++ b/src/parser/Statement.ts @@ -1,7 +1,7 @@ /* eslint-disable no-bitwise */ import type { Token, Identifier } from '../lexer'; import { CompoundAssignmentOperators, TokenKind } from '../lexer'; -import type { BinaryExpression, Expression, NamespacedVariableNameExpression, FunctionExpression, AnnotationExpression } from './Expression'; +import type { BinaryExpression, Expression, NamespacedVariableNameExpression, FunctionExpression, AnnotationExpression, FunctionParameterExpression } from './Expression'; import { CallExpression, VariableExpression } from './Expression'; import { util } from '../util'; import type { Range } from 'vscode-languageserver'; @@ -10,10 +10,13 @@ import type { BrsTranspileState } from './BrsTranspileState'; import { ParseMode, Parser } from './Parser'; import type { WalkVisitor, WalkOptions } from '../astUtils/visitors'; import { InternalWalkMode, walk, createVisitor, WalkMode } from '../astUtils/visitors'; -import { isCallExpression, isClassFieldStatement, isClassMethodStatement, isCommentStatement, isExpression, isExpressionStatement, isFunctionStatement, isIfStatement, isInvalidType, isLiteralExpression, isVoidType } from '../astUtils/reflection'; +import { isCallExpression, isClassFieldStatement, isClassMethodStatement, isCommentStatement, isExpression, isExpressionStatement, isFunctionStatement, isIfStatement, isInterfaceFieldStatement, isInterfaceMethodStatement, isInvalidType, isLiteralExpression, isVoidType } from '../astUtils/reflection'; import type { TranspileResult, TypedefProvider } from '../interfaces'; import { createInvalidLiteral, createToken, interpolatedRange } from '../astUtils/creators'; import { DynamicType } from '../types/DynamicType'; +import type { BscType } from '../types/BscType'; +import type { SourceNode } from 'source-map'; +import type { TranspileState } from './TranspileState'; /** * A BrightScript statement @@ -285,8 +288,8 @@ export class CommentStatement extends Statement implements Expression, TypedefPr return result; } - public getTypedef(state: BrsTranspileState) { - return this.transpile(state); + public getTypedef(state: TranspileState) { + return this.transpile(state as BrsTranspileState); } walk(visitor: WalkVisitor, options: WalkOptions) { @@ -1213,6 +1216,285 @@ export class ImportStatement extends Statement implements TypedefProvider { } } +export class InterfaceStatement extends Statement implements TypedefProvider { + constructor( + interfaceToken: Token, + name: Identifier, + extendsToken: Token, + public parentInterfaceName: NamespacedVariableNameExpression, + public body: Statement[], + endInterfaceToken: Token, + public namespaceName: NamespacedVariableNameExpression + ) { + super(); + this.tokens.interface = interfaceToken; + this.tokens.name = name; + this.tokens.extends = extendsToken; + this.tokens.endInterface = endInterfaceToken; + } + + public tokens = {} as { + interface: Token; + name: Identifier; + extends: Token; + endInterface: Token; + }; + + public range: Range; + + public get fields() { + return this.body.filter(x => isInterfaceFieldStatement(x)); + } + + public get methods() { + return this.body.filter(x => isInterfaceMethodStatement(x)); + } + + /** + * The name of the interface WITH its leading namespace (if applicable) + */ + public get fullName() { + const name = this.tokens.name?.text; + if (name) { + if (this.namespaceName) { + let namespaceName = this.namespaceName.getName(ParseMode.BrighterScript); + return `${namespaceName}.${name}`; + } else { + return name; + } + } else { + //return undefined which will allow outside callers to know that this interface doesn't have a name + return undefined; + } + } + + /** + * The name of the interface (without the namespace prefix) + */ + public get name() { + return this.tokens.name?.text; + } + + public transpile(state: BrsTranspileState): TranspileResult { + //interfaces should completely disappear at runtime + return []; + } + + getTypedef(state: BrsTranspileState) { + const result = [] as TranspileResult; + for (let annotation of this.annotations ?? []) { + result.push( + ...annotation.getTypedef(state), + state.newline, + state.indent() + ); + } + result.push( + this.tokens.interface.text, + ' ', + this.tokens.name.text + ); + const parentInterfaceName = this.parentInterfaceName?.getName(ParseMode.BrighterScript); + if (parentInterfaceName) { + result.push( + ' extends ', + parentInterfaceName + ); + } + const body = this.body ?? []; + if (body.length > 0) { + state.blockDepth++; + } + for (const statement of body) { + if (isInterfaceMethodStatement(statement) || isInterfaceFieldStatement(statement)) { + result.push( + state.newline, + state.indent(), + ...statement.getTypedef(state) + ); + } else { + result.push( + state.newline, + state.indent(), + ...statement.transpile(state) + ); + } + } + if (body.length > 0) { + state.blockDepth--; + } + result.push( + state.newline, + state.indent(), + 'end interface', + state.newline + ); + return result; + } + + walk(visitor: WalkVisitor, options: WalkOptions) { + if (options.walkMode & InternalWalkMode.walkStatements) { + for (let i = 0; i < this.body.length; i++) { + walk(this.body, i, visitor, options, this); + } + } + } +} + +export class InterfaceFieldStatement extends Statement implements TypedefProvider { + public transpile(state: BrsTranspileState): TranspileResult { + throw new Error('Method not implemented.'); + } + constructor( + nameToken: Identifier, + asToken: Token, + typeToken: Token, + public type: BscType + ) { + super(); + this.tokens.name = nameToken; + this.tokens.as = asToken; + this.tokens.type = typeToken; + } + public get range() { + return util.createRangeFromPositions( + this.tokens.name.range.start, + (this.tokens.type ?? this.tokens.as ?? this.tokens.name).range.end + ); + } + + public tokens = {} as { + name: Identifier; + as: Token; + type: Token; + }; + + public get name() { + return this.tokens.name.text; + } + + walk(visitor: WalkVisitor, options: WalkOptions) { + //nothing to walk + } + + getTypedef(state: BrsTranspileState): (string | SourceNode)[] { + const result = [] as TranspileResult; + for (let annotation of this.annotations ?? []) { + result.push( + ...annotation.getTypedef(state), + state.newline, + state.indent() + ); + } + + result.push( + this.tokens.name.text + ); + if (this.tokens.type?.text?.length > 0) { + result.push( + ' as ', + this.tokens.type.text + ); + } + return result; + } + +} + +export class InterfaceMethodStatement extends Statement implements TypedefProvider { + public transpile(state: BrsTranspileState): TranspileResult { + throw new Error('Method not implemented.'); + } + constructor( + functionTypeToken: Token, + nameToken: Identifier, + leftParen: Token, + public params: FunctionParameterExpression[], + rightParen: Token, + asToken?: Token, + returnTypeToken?: Token, + public returnType?: BscType + ) { + super(); + this.tokens.functionType = functionTypeToken; + this.tokens.name = nameToken; + this.tokens.leftParen = leftParen; + this.tokens.rightParen = rightParen; + this.tokens.as = asToken; + this.tokens.returnType = returnTypeToken; + } + + public get range() { + return util.createRangeFromPositions( + this.tokens.name.range.start, + ( + this.tokens.returnType ?? + this.tokens.as ?? + this.tokens.rightParen ?? + this.params?.[this.params?.length - 1] ?? + this.tokens.leftParen ?? + this.tokens.name ?? + this.tokens.functionType + ).range.end + ); + } + + public tokens = {} as { + functionType: Token; + name: Identifier; + leftParen: Token; + rightParen: Token; + as: Token; + returnType: Token; + }; + + walk(visitor: WalkVisitor, options: WalkOptions) { + //nothing to walk + } + + getTypedef(state: BrsTranspileState) { + const result = [] as TranspileResult; + for (let annotation of this.annotations ?? []) { + result.push( + ...annotation.getTypedef(state), + state.newline, + state.indent() + ); + } + + result.push( + this.tokens.functionType.text, + ' ', + this.tokens.name.text, + '(' + ); + const params = this.params ?? []; + for (let i = 0; i < params.length; i++) { + if (i > 0) { + result.push(', '); + } + const param = params[i]; + result.push(param.name.text); + if (param.typeToken?.text?.length > 0) { + result.push( + ' as ', + param.typeToken.text + ); + } + } + result.push( + ')' + ); + if (this.tokens.returnType?.text.length > 0) { + result.push( + ' as ', + this.tokens.returnType.text + ); + } + return result; + } +} + export class ClassStatement extends Statement implements TypedefProvider { constructor( diff --git a/src/parser/tests/statement/InterfaceStatement.spec.ts b/src/parser/tests/statement/InterfaceStatement.spec.ts new file mode 100644 index 000000000..aeca6749c --- /dev/null +++ b/src/parser/tests/statement/InterfaceStatement.spec.ts @@ -0,0 +1,64 @@ +import { getTestGetTypedef } from '../../../testHelpers.spec'; +import { standardizePath as s } from '../../../util'; +import { Program } from '../../../Program'; + +describe('InterfaceStatement', () => { + const rootDir = s`${process.cwd()}/.tmp/rootDir`; + let program: Program; + beforeEach(() => { + program = new Program({ + rootDir: rootDir + }); + }); + + const testGetTypedef = getTestGetTypedef(() => [program, rootDir]); + + it('allows strange keywords as property names', () => { + testGetTypedef(` + interface Person + public as string + protected as string + private as string + sub as string + function as string + interface as string + endInterface as string + end interface + `, undefined, undefined, undefined, true); + }); + + it('allows strange keywords as method names', () => { + testGetTypedef(` + interface Person + sub public() as string + sub protected() as string + sub private() as string + sub sub() as string + sub function() as string + sub interface() as string + sub endInterface() as string + end interface + `, undefined, undefined, undefined, true); + }); + + it('includes comments', () => { + testGetTypedef(` + interface Person + 'some comment + sub someFunc() as string + end interface + `, undefined, undefined, undefined, true); + }); + + it('includes annotations', () => { + testGetTypedef(` + @IFace + interface Person + @Method + sub someFunc() as string + @Field + someField as string + end interface + `, undefined, undefined, undefined, true); + }); +}); diff --git a/src/parser/tests/statement/Misc.spec.ts b/src/parser/tests/statement/Misc.spec.ts index 2471335b0..81574037d 100644 --- a/src/parser/tests/statement/Misc.spec.ts +++ b/src/parser/tests/statement/Misc.spec.ts @@ -3,6 +3,7 @@ import { Parser } from '../../Parser'; import { Lexer, DisallowedLocalIdentifiersText, TokenKind } from '../../../lexer'; import { Range } from 'vscode-languageserver'; import type { AAMemberExpression } from '../../Expression'; +import { expectZeroDiagnostics } from '../../../testHelpers.spec'; describe('parser', () => { describe('`end` keyword', () => { @@ -73,25 +74,28 @@ describe('parser', () => { statementList.push(statements); }); }); + it('allows certain TokenKinds to be treated as local variables', () => { //a few additional keywords that we don't have tokenKinds for - let { tokens } = Lexer.scan(` - sub main() - Void = true - Number = true - Boolean = true - Integer = true - LongInteger = true - Float = true - Double = true - String = true - Object = true - Interface = true - Dynamic = true - end sub - `); - let { diagnostics } = Parser.parse(tokens); - expect(diagnostics).to.be.lengthOf(0); + expectZeroDiagnostics( + Parser.parse(` + sub main() + Void = true + Number = true + Boolean = true + Integer = true + LongInteger = true + Float = true + Double = true + String = true + Object = true + Interface = true + Dynamic = true + Class = true + Namespace = true + end sub + `) + ); }); it('allows certain TokenKinds as object properties', () => { diff --git a/src/testHelpers.spec.ts b/src/testHelpers.spec.ts index ad43f41d4..df3aefa28 100644 --- a/src/testHelpers.spec.ts +++ b/src/testHelpers.spec.ts @@ -1,4 +1,4 @@ -import type { BsDiagnostic } from './interfaces'; +import type { BscFile, BsDiagnostic } from './interfaces'; import * as assert from 'assert'; import type { Diagnostic } from 'vscode-languageserver'; import { createSandbox } from 'sinon'; @@ -9,6 +9,7 @@ import { codeActionUtil } from './CodeActionUtil'; import { BrsFile } from './files/BrsFile'; import type { Program } from './Program'; import { standardizePath as s } from './util'; +import type { CodeWithSourceMap } from 'source-map'; /** * Trim leading whitespace for every line (to make test writing cleaner @@ -108,7 +109,23 @@ export function expectCodeActions(test: () => any, expected: CodeActionShorthand } export function getTestTranspile(scopeGetter: () => [Program, string]) { - return (source: string, expected?: string, formatType: 'trim' | 'none' = 'trim', pkgPath = 'source/main.bs', failOnDiagnostic = true) => { + return getTestFileAction((file) => file.transpile(), scopeGetter); +} + +export function getTestGetTypedef(scopeGetter: () => [Program, string]) { + return getTestFileAction((file) => { + return { + code: (file as BrsFile).getTypedef(), + map: undefined + }; + }, scopeGetter); +} + +function getTestFileAction( + action: (file: BscFile) => CodeWithSourceMap, + scopeGetter: () => [Program, string] +) { + return function testFileAction(source: string, expected?: string, formatType: 'trim' | 'none' = 'trim', pkgPath = 'source/main.bs', failOnDiagnostic = true) { let [program, rootDir] = scopeGetter(); expected = expected ? expected : source; let file = program.addOrReplaceFile({ src: s`${rootDir}/${pkgPath}`, dest: pkgPath }, source); @@ -116,9 +133,9 @@ export function getTestTranspile(scopeGetter: () => [Program, string]) { if (failOnDiagnostic !== false) { expectZeroDiagnostics(program); } - let transpiled = file.transpile(); + let codeWithMap = action(file); - let sources = [transpiled.code, expected]; + let sources = [codeWithMap.code, expected]; for (let i = 0; i < sources.length; i++) { if (formatType === 'trim') { let lines = sources[i].split('\n'); @@ -146,6 +163,12 @@ export function getTestTranspile(scopeGetter: () => [Program, string]) { } } expect(trimMap(sources[0])).to.equal(sources[1]); - return transpiled; + return { + file: file, + source: source, + expected: expected, + actual: codeWithMap.code, + map: codeWithMap.map + }; }; } diff --git a/src/types/InterfaceType.spec.ts b/src/types/InterfaceType.spec.ts new file mode 100644 index 000000000..4185a098d --- /dev/null +++ b/src/types/InterfaceType.spec.ts @@ -0,0 +1,208 @@ +import { expect } from 'chai'; +import { assert } from 'sinon'; +import type { BscType } from './BscType'; +import { DynamicType } from './DynamicType'; +import { IntegerType } from './IntegerType'; +import { InterfaceType } from './InterfaceType'; +import { ObjectType } from './ObjectType'; +import { StringType } from './StringType'; + +describe('InterfaceType', () => { + describe('toString', () => { + it('returns empty curly braces when no members', () => { + expect(iface({}).toString()).to.eql('{}'); + }); + + it('includes member types', () => { + expect(iface({ name: new StringType() }).toString()).to.eql('{ name: string; }'); + }); + + it('includes nested object types', () => { + expect( + iface({ + name: new StringType(), + parent: iface({ + age: new IntegerType() + }) + } + ).toString() + ).to.eql('{ name: string; parent: { age: integer; }; }'); + }); + }); + + describe('isConvertibleTo', () => { + it('works', () => { + expectAssignable({ + name: new StringType() + }, { + name: new StringType() + }); + }); + }); + + describe('equals', () => { + it('matches equal objects', () => { + expect( + iface({ name: new StringType() }).equals(iface({ name: new StringType() })) + ).to.be.true; + }); + + it('does not match inequal objects', () => { + expect( + iface({ name: new StringType() }).equals(iface({ name: new IntegerType() })) + ).to.be.false; + }); + }); + + describe('isAssignableTo', () => { + it('rejects being assignable to other types', () => { + expect( + iface({ + name: new StringType() + }).isAssignableTo(new IntegerType()) + ).to.be.false; + }); + + it('matches exact properties', () => { + expectAssignable({ + name: new StringType() + }, { + name: new StringType() + }); + }); + + it('matches an object with more properties being assigned to an object with less', () => { + expectAssignable({ + name: new StringType() + }, { + name: new StringType(), + age: new IntegerType() + }); + }); + + it('rejects assigning an object with less properties to one with more', () => { + expectNotAssignable({ + name: new StringType(), + age: new IntegerType() + }, { + name: new StringType() + }); + }); + + it('matches properties in mismatched order', () => { + expect( + new InterfaceType(new Map([ + ['name', new StringType()], + ['age', new IntegerType()] + ])).isAssignableTo(new InterfaceType(new Map([ + ['age', new IntegerType()], + ['name', new StringType()] + ]))) + ).to.be.true; + }); + + it('rejects with member having mismatched type', () => { + expectNotAssignable({ + name: new StringType() + }, { + name: new IntegerType() + }); + }); + + it('rejects with object member having mismatched type', () => { + expectNotAssignable({ + parent: iface({ + name: new StringType() + }) + }, { + parent: iface({ + name: new IntegerType() + }) + }); + }); + + it('rejects with object member having missing prop type', () => { + expectNotAssignable({ + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }, { + parent: iface({ + name: new StringType() + }) + }); + }); + + it('accepts with object member having same prop types', () => { + expectAssignable({ + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }, { + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }); + }); + + it('accepts with source member having dyanmic prop type', () => { + expectAssignable({ + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }, { + parent: new DynamicType() + }); + }); + + it('accepts with target member having dyanmic prop type', () => { + expectAssignable({ + parent: new DynamicType() + }, { + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }); + }); + + it('accepts with target member having "object" prop type', () => { + expectAssignable({ + parent: new ObjectType() + }, { + parent: iface({ + name: new StringType(), + age: new IntegerType() + }) + }); + }); + }); +}); + +function iface(members: Record) { + return new InterfaceType( + new Map( + Object.entries(members) + ) + ); +} + +function expectAssignable(targetMembers: Record, sourceMembers: Record) { + const targetIface = iface(targetMembers); + const sourceIface = iface(sourceMembers); + if (!sourceIface.isAssignableTo(targetIface)) { + assert.fail(`expected type ${targetIface.toString()} to be assignable to type ${sourceIface.toString()}`); + } +} + +function expectNotAssignable(targetMembers: Record, sourceMembers: Record) { + const targetIface = iface(targetMembers); + const sourceIface = iface(sourceMembers); + if (sourceIface.isAssignableTo(targetIface)) { + assert.fail(`expected type ${targetIface.toString()} to not be assignable to type ${sourceIface.toString()}`); + } +} diff --git a/src/types/InterfaceType.ts b/src/types/InterfaceType.ts index f3f5398eb..dfbafb007 100644 --- a/src/types/InterfaceType.ts +++ b/src/types/InterfaceType.ts @@ -1,12 +1,42 @@ +import { isDynamicType, isInterfaceType, isObjectType } from '../astUtils/reflection'; import type { BscType } from './BscType'; -import { DynamicType } from './DynamicType'; export class InterfaceType implements BscType { + public constructor( + public members: Map + ) { + + } + + /** + * The name of the interface. Can be null. + */ + public name: string; + public isAssignableTo(targetType: BscType) { - return ( - targetType instanceof InterfaceType || - targetType instanceof DynamicType - ); + //we must have all of the members of the target type, and they must be equivalent types + if (isInterfaceType(targetType)) { + for (const [targetMemberName, targetMemberType] of targetType.members) { + //we don't have the target member + if (!this.members.has(targetMemberName)) { + return false; + } + //our member's type is not assignable to the target member type + if (!this.members.get(targetMemberName).isAssignableTo(targetMemberType)) { + return false; + } + } + //we have all of the target member's types. we are assignable! + return true; + + //we are always assignable to dynamic or object + } else if (isDynamicType(targetType) || isObjectType(targetType)) { + return true; + + //not assignable to any other object types + } else { + return false; + } } public isConvertibleTo(targetType: BscType) { @@ -14,11 +44,27 @@ export class InterfaceType implements BscType { } public toString() { - //TODO make this match the actual interface of the object - return 'interface'; + let result = '{'; + for (const [key, type] of this.members.entries()) { + result += ' ' + key + ': ' + type.toString() + ';'; + } + if (this.members.size > 0) { + result += ' '; + } + return result + '}'; } public toTypeString(): string { - return this.toString(); + return 'object'; + } + + public equals(targetType: BscType): boolean { + if (isInterfaceType(targetType)) { + if (targetType.members.size !== this.members.size) { + return false; + } + return targetType.isAssignableTo(this); + } + return false; } } From 986eb11f4c89ddc118e836e2ced88af78600d557 Mon Sep 17 00:00:00 2001 From: Mark Pearce Date: Mon, 28 Jun 2021 22:36:57 -0300 Subject: [PATCH 042/278] CLI outputs the path of bsconfig.json on startup (#434) Co-authored-by: Mark Pearce Co-authored-by: Bronley Plumb --- src/ProgramBuilder.ts | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/ProgramBuilder.ts b/src/ProgramBuilder.ts index 05225af04..41ad59e72 100644 --- a/src/ProgramBuilder.ts +++ b/src/ProgramBuilder.ts @@ -95,6 +95,11 @@ export class ProgramBuilder { this.isRunning = true; try { this.options = util.normalizeAndResolveConfig(options); + if (this.options.project) { + this.logger.log(`Using config file: "${this.options.project}"`); + } else { + this.logger.log(`No bsconfig.json file found, using default options`); + } this.loadPlugins(); } catch (e) { if (e?.file && e.message && e.code) { From 795a1e656d5670ae3b0cf6d0b057171d353ebae5 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 2 Aug 2021 15:27:35 -0400 Subject: [PATCH 043/278] update changelog for v0.40.0 --- CHANGELOG.md | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 86ff7dda3..86c221f19 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,8 +6,17 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.40.0] - 2021-08-02 +[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0 +### Added + - language support for Interface statements ([#426](https://github.com/rokucommunity/brighterscript/pull/426)) +### Changed + - cli prints the path of any loaded bsconfig.json on startup ([#434](https://github.com/rokucommunity/brighterscript/pull/434)) + + + ## [0.39.4] - 2021-06-27 -[0.39.4]: https://github.com/RokuCommunity/roku-debug/compare/v0.8.3...v0.8.4 +[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.3...v0.39.4 ### Fixed - incorrect block range for inline if/then branch ([#424](https://github.com/rokucommunity/brighterscript/pull/424)) - extract associative array comma when parsing ([#427](https://github.com/rokucommunity/brighterscript/pull/424)) @@ -16,7 +25,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [0.39.3] - 2021-06-01 -[0.39.3]: https://github.com/RokuCommunity/roku-debug/compare/v0.8.2...v0.8.3 +[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.2...v0.39.3 ### Changed - upgraded to [roku-deploy@3.4.1](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#341---2021-06-01) which fixes bugs introduced in roku-deploy@3.4.0 From 961539358db3f473e5b5789ce95a6bb0ad829beb Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 2 Aug 2021 15:29:31 -0400 Subject: [PATCH 044/278] 0.40.0 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index c2dbc4fc0..2884e0434 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.4", + "version": "0.40.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 629820c85..baa3a0981 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.39.4", + "version": "0.40.0", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 485a77db6e1be217a019e159d5eab4eb391cfdce Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 2 Aug 2021 21:37:55 -0400 Subject: [PATCH 045/278] fix changelog links. --- CHANGELOG.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 86c221f19..d40f766e7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,7 +7,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [0.40.0] - 2021-08-02 -[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0 +[0.40.0]: https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0 ### Added - language support for Interface statements ([#426](https://github.com/rokucommunity/brighterscript/pull/426)) ### Changed @@ -16,7 +16,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [0.39.4] - 2021-06-27 -[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.3...v0.39.4 +[0.39.4]: https://github.com/rokucommunity/brighterscript/compare/v0.39.3...v0.39.4 ### Fixed - incorrect block range for inline if/then branch ([#424](https://github.com/rokucommunity/brighterscript/pull/424)) - extract associative array comma when parsing ([#427](https://github.com/rokucommunity/brighterscript/pull/424)) @@ -25,7 +25,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [0.39.3] - 2021-06-01 -[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.2...v0.39.3 +[0.39.3]: https://github.com/rokucommunity/brighterscript/compare/v0.39.2...v0.39.3 ### Changed - upgraded to [roku-deploy@3.4.1](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#341---2021-06-01) which fixes bugs introduced in roku-deploy@3.4.0 From 47cde1d9dcd1441fa3f1c5d7c18220567e3c52a1 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 8 Sep 2021 15:27:21 -0400 Subject: [PATCH 046/278] Bump path-parse from 1.0.6 to 1.0.7 (#436) Bumps [path-parse](https://github.com/jbgutierrez/path-parse) from 1.0.6 to 1.0.7. - [Release notes](https://github.com/jbgutierrez/path-parse/releases) - [Commits](https://github.com/jbgutierrez/path-parse/commits/v1.0.7) --- updated-dependencies: - dependency-name: path-parse dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index 2884e0434..72d36a9eb 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4518,9 +4518,9 @@ "dev": true }, "path-parse": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", - "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", "dev": true }, "path-to-regexp": { From 3ec5e56147f04626407ba08eefe43f3c08a8c8ab Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 8 Sep 2021 15:31:59 -0400 Subject: [PATCH 047/278] Bump jszip from 3.6.0 to 3.7.1 (#438) Bumps [jszip](https://github.com/Stuk/jszip) from 3.6.0 to 3.7.1. - [Release notes](https://github.com/Stuk/jszip/releases) - [Changelog](https://github.com/Stuk/jszip/blob/master/CHANGES.md) - [Commits](https://github.com/Stuk/jszip/compare/v3.6.0...v3.7.1) --- updated-dependencies: - dependency-name: jszip dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index 72d36a9eb..9ba8b9094 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3335,9 +3335,9 @@ } }, "jszip": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.6.0.tgz", - "integrity": "sha512-jgnQoG9LKnWO3mnVNBnfhkh0QknICd1FGSrXcgrl67zioyJ4wgx25o9ZqwNtrROSflGBCGYnJfjrIyRIby1OoQ==", + "version": "3.7.1", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.7.1.tgz", + "integrity": "sha512-ghL0tz1XG9ZEmRMcEN2vt7xabrDdqHHeykgARpmZ0BiIctWxM47Vt63ZO2dnp4QYt/xJVLLy5Zv1l/xRdh2byg==", "requires": { "lie": "~3.3.0", "pako": "~1.0.2", From 7aa1847d18095f4a400358d1dd970b60dacff821 Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 17 Sep 2021 06:30:21 -0400 Subject: [PATCH 048/278] roku-deploy@3.4.2 --- package-lock.json | 6 +++--- package.json | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/package-lock.json b/package-lock.json index 9ba8b9094..0748ebdec 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5038,9 +5038,9 @@ } }, "roku-deploy": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.1.tgz", - "integrity": "sha512-KomV/FvWv84f2Lh9V5c9stXCkGbM/L7B5cCLpNddBTlgh3zxyCAjEj2oFPYnzjbG+aNMvOpKatQ7Xx8bMJM03g==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.2.tgz", + "integrity": "sha512-Tg/K5M9/gvH5bzzs3g2bUxrcPiwKigKdnpbq5yiTjMnAtDJM6u7Ug4PbcnnoH3qJ3IJqODCW7Kq1iIH7sWmqBg==", "requires": { "chalk": "^2.4.2", "dateformat": "^3.0.3", diff --git a/package.json b/package.json index baa3a0981..0127bc29d 100644 --- a/package.json +++ b/package.json @@ -128,7 +128,7 @@ "moment": "^2.23.0", "p-settle": "^2.1.0", "parse-ms": "^2.1.0", - "roku-deploy": "^3.4.1", + "roku-deploy": "^3.4.2", "serialize-error": "^7.0.1", "source-map": "^0.7.3", "vscode-languageserver": "7.0.0", From 95d4676d844cdbeb53676a66a4c713545115a973 Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 17 Sep 2021 06:32:43 -0400 Subject: [PATCH 049/278] update changelog for v0.40.1 --- CHANGELOG.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index d40f766e7..f90ce28ef 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.40.1](https://github.com/rokucommunity/brighterscript/compare/v0.40.0...v0.40.1) - 2021-09-17 +### Changed + - install roku-deploy@3.4.2 which prevents deploy crashes when target Roku doesn't have an installed channel ([roku-deploy#65](https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0)) + + + ## [0.40.0] - 2021-08-02 [0.40.0]: https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0 ### Added From 1818c361b3efa1120212cde8e4969b745587880e Mon Sep 17 00:00:00 2001 From: Bronley Date: Fri, 17 Sep 2021 06:34:21 -0400 Subject: [PATCH 050/278] 0.40.1 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 0748ebdec..99ae83b54 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.40.0", + "version": "0.40.1", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 0127bc29d..af5bd1221 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.40.0", + "version": "0.40.1", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 22c8a30a8d6c387fe630cadd62e5c4e46dfbed5d Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Mon, 27 Sep 2021 07:35:03 -0400 Subject: [PATCH 051/278] Add regex literal support (#452) * Basic regex literal lexer support. * Add lexer support for escaped regexp chars * Add parser and transpile functionality * Add very basic docs about regex literals * Verify lexer handles quotemark properly * Escape quotemarks --- docs/readme.md | 1 + docs/regex-literals.md | 13 ++++ src/lexer/Lexer.spec.ts | 38 ++++++++++ src/lexer/Lexer.ts | 71 ++++++++++++++++--- src/lexer/TokenKind.ts | 1 + src/parser/Expression.ts | 43 +++++++++++ src/parser/Parser.ts | 9 +++ .../expression/RegexLiteralExpression.spec.ts | 67 +++++++++++++++++ 8 files changed, 235 insertions(+), 8 deletions(-) create mode 100644 docs/regex-literals.md create mode 100644 src/parser/tests/expression/RegexLiteralExpression.spec.ts diff --git a/docs/readme.md b/docs/readme.md index 65da5e877..f3e1a04f1 100644 --- a/docs/readme.md +++ b/docs/readme.md @@ -10,6 +10,7 @@ See the following pages for more information - [Namespaces](namespaces.md) - [Null-coalescing operator](null-coalescing-operator.md) - [Plugins](plugins.md) + - [Regular Expression Literals](regex-literals.md) - [Source Literals](source-literals.md) - [Template Strings (Template Literals)](template-strings.md) - [Ternary (Conditional) Operator](ternary-operator.md) diff --git a/docs/regex-literals.md b/docs/regex-literals.md new file mode 100644 index 000000000..a3feae1b2 --- /dev/null +++ b/docs/regex-literals.md @@ -0,0 +1,13 @@ +# Regular Expression Literals +You can create a regular expression literal in brighterscript. This simplifies pattern writing and improves readability. + +Example: +```BrighterScript +print /hello world/ig +``` + +transpiles to: + +```BrightScript +print CreateObject("roRegex","hello world","ig") +``` diff --git a/src/lexer/Lexer.spec.ts b/src/lexer/Lexer.spec.ts index c12a6ba30..765c92c49 100644 --- a/src/lexer/Lexer.spec.ts +++ b/src/lexer/Lexer.spec.ts @@ -1222,4 +1222,42 @@ describe('lexer', () => { TokenKind.Eof ]); }); + + describe('regular expression literals', () => { + function testRegex(...regexps: Array) { + regexps = regexps.map(x => x.toString()); + const results = [] as string[]; + for (const regexp of regexps) { + const { tokens } = Lexer.scan(regexp as string); + results.push(tokens[0].text); + } + expect(results).to.eql(regexps); + } + + it('recognizes regex literals', () => { + testRegex( + /simple/, + /SimpleWithValidFlags/g, + /UnknownFlags/gi, + /with spaces/s, + /with(parens)and[squarebraces]/, + //lots of special characters + /.*()^$@/, + //captures quote char + /"/ + ); + }); + + it('handles escape characters properly', () => { + testRegex( + //an escaped forward slash right next to the end-regexp forwardslash + /\//, + /\r/, + /\n/, + /\r\n/, + //a literal backslash in front of an escape backslash + /\\\n/ + ); + }); + }); }); diff --git a/src/lexer/Lexer.ts b/src/lexer/Lexer.ts index 962f4ab1d..ae159303f 100644 --- a/src/lexer/Lexer.ts +++ b/src/lexer/Lexer.ts @@ -199,14 +199,17 @@ export class Lexer { } }, '/': function (this: Lexer) { - switch (this.peek()) { - case '=': - this.advance(); - this.addToken(TokenKind.ForwardslashEqual); - break; - default: - this.addToken(TokenKind.Forwardslash); - break; + //try capturing a regex literal. If that doesn't work, fall back to normal handling + if (!this.regexLiteral()) { + switch (this.peek()) { + case '=': + this.advance(); + this.addToken(TokenKind.ForwardslashEqual); + break; + default: + this.addToken(TokenKind.Forwardslash); + break; + } } }, '\\': function (this: Lexer) { @@ -384,6 +387,19 @@ export class Lexer { this.columnEnd++; } + private lookaheadStack = [] as Array<{ current: number; columnEnd: number }>; + private pushLookahead() { + this.lookaheadStack.push({ + current: this.current, + columnEnd: this.columnEnd + }); + } + private popLookahead() { + const { current, columnEnd } = this.lookaheadStack.pop(); + this.current = current; + this.columnEnd = columnEnd; + } + /** * Returns the character at position `current` or a null character if we've reached the end of * input. @@ -927,6 +943,45 @@ export class Lexer { } } + /** + * Capture a regex literal token. Returns false if not found. + * This is lookahead lexing which might techincally belong in the parser, + * but it's easy enough to do here in the lexer + */ + private regexLiteral() { + this.pushLookahead(); + + let nextCharNeedsEscaped = false; + + //finite loop to prevent infinite loop if something went wrong + for (let i = this.current; i < this.source.length; i++) { + + //if we reached the end of the regex, consume any flags + if (this.check('/') && !nextCharNeedsEscaped) { + this.advance(); + //consume all flag-like chars (let the parser validate the actual values) + while (/[a-z]/i.exec(this.peek())) { + this.advance(); + } + //finalize the regex literal and EXIT + this.addToken(TokenKind.RegexLiteral); + return true; + + //if we found a non-escaped newline, there's a syntax error with this regex (or it's not a regex), so quit + } else if (this.check('\n') || this.isAtEnd()) { + break; + } else if (this.check('\\')) { + this.advance(); + nextCharNeedsEscaped = true; + } else { + this.advance(); + nextCharNeedsEscaped = false; + } + } + this.popLookahead(); + return false; + } + /** * Creates a `Token` and adds it to the `tokens` array. * @param kind the type of token to produce. diff --git a/src/lexer/TokenKind.ts b/src/lexer/TokenKind.ts index f01bb4b0c..70ed8d7df 100644 --- a/src/lexer/TokenKind.ts +++ b/src/lexer/TokenKind.ts @@ -52,6 +52,7 @@ export enum TokenKind { DoubleLiteral = 'DoubleLiteral', LongIntegerLiteral = 'LongIntegerLiteral', EscapedCharCodeLiteral = 'EscapedCharCodeLiteral', //this is used to capture things like `\n`, `\r\n` in template strings + RegexLiteral = 'RegexLiteral', //types Void = 'Void', diff --git a/src/parser/Expression.ts b/src/parser/Expression.ts index aef007f9a..279263be5 100644 --- a/src/parser/Expression.ts +++ b/src/parser/Expression.ts @@ -1406,6 +1406,49 @@ export class NullCoalescingExpression extends Expression { } } +export class RegexLiteralExpression extends Expression { + public constructor( + public tokens: { + regexLiteral: Token; + } + ) { + super(); + } + + public get range() { + return this.tokens.regexLiteral.range; + } + + public transpile(state: BrsTranspileState): TranspileResult { + let text = this.tokens.regexLiteral?.text ?? ''; + let flags = ''; + //get any flags from the end + const flagMatch = /\/([a-z]+)$/i.exec(text); + if (flagMatch) { + text = text.substring(0, flagMatch.index + 1); + flags = flagMatch[1]; + } + let pattern = text + //remove leading and trailing slashes + .substring(1, text.length - 1) + //escape quotemarks + .split('"').join('" + chr(34) + "'); + + return [ + state.sourceNode(this.tokens.regexLiteral, [ + 'CreateObject("roRegex", ', + `"${pattern}", `, + `"${flags}"`, + ')' + ]) + ]; + } + + walk(visitor: WalkVisitor, options: WalkOptions) { + //nothing to walk + } +} + // eslint-disable-next-line @typescript-eslint/consistent-indexed-object-style type ExpressionValue = string | number | boolean | Expression | ExpressionValue[] | { [key: string]: ExpressionValue }; diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index 795d7886b..54faa418f 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -93,6 +93,7 @@ import { Logger } from '../Logger'; import { isAnnotationExpression, isCallExpression, isCallfuncExpression, isClassMethodStatement, isCommentStatement, isDottedGetExpression, isIfStatement, isIndexedGetExpression, isVariableExpression } from '../astUtils/reflection'; import { createVisitor, WalkMode } from '../astUtils/visitors'; import { createStringLiteral, createToken } from '../astUtils/creators'; +import { RegexLiteralExpression } from '.'; export class Parser { /** @@ -1396,6 +1397,12 @@ export class Parser { return new NullCoalescingExpression(test, questionQuestionToken, alternate); } + private regexLiteralExpression() { + return new RegexLiteralExpression({ + regexLiteral: this.advance() + }); + } + private templateString(isTagged: boolean): TemplateStringExpression | TaggedTemplateStringExpression { this.warnIfNotBrighterScriptMode('template string'); @@ -2544,6 +2551,8 @@ export class Parser { return new VariableExpression(token, this.currentNamespaceName); case this.checkAny(TokenKind.Function, TokenKind.Sub): return this.anonymousFunction(); + case this.check(TokenKind.RegexLiteral): + return this.regexLiteralExpression(); case this.check(TokenKind.Comment): return new CommentStatement([this.advance()]); default: diff --git a/src/parser/tests/expression/RegexLiteralExpression.spec.ts b/src/parser/tests/expression/RegexLiteralExpression.spec.ts new file mode 100644 index 000000000..be5542ad8 --- /dev/null +++ b/src/parser/tests/expression/RegexLiteralExpression.spec.ts @@ -0,0 +1,67 @@ +import { Program } from '../../../Program'; +import { standardizePath as s } from '../../../util'; +import { getTestTranspile } from '../../../testHelpers.spec'; + +describe('RegexLiteralExpression', () => { + let rootDir = s`${process.cwd()}/rootDir`; + let program: Program; + let testTranspile = getTestTranspile(() => [program, rootDir]); + + beforeEach(() => { + program = new Program({ rootDir: rootDir }); + }); + afterEach(() => { + program.dispose(); + }); + + describe('transpile', () => { + it('captures flags', () => { + testTranspile(` + sub main() + print /hello/gi + end sub + `, ` + sub main() + print CreateObject("roRegex", "hello", "gi") + end sub + `); + }); + + it('handles when no flags', () => { + testTranspile(` + sub main() + print /hello/ + end sub + `, ` + sub main() + print CreateObject("roRegex", "hello", "") + end sub + `); + }); + + it('handles weird escapes', () => { + testTranspile(` + sub main() + print /\\r\\n\\// + end sub + `, ` + sub main() + print CreateObject("roRegex", "\\r\\n\\/", "") + end sub + `); + }); + + it('escapes quotemark', () => { + testTranspile(` + sub main() + print /"/ + end sub + `, ` + sub main() + print CreateObject("roRegex", "" + chr(34) + "", "") + end sub + `); + }); + + }); +}); From 8bc9744337dddedc8ff05647cdfc2b4c81b94f58 Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 27 Sep 2021 07:42:14 -0400 Subject: [PATCH 052/278] fix out-of-sync compliled doc code blocks --- docs/regex-literals.md | 2 +- docs/ternary-operator.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/regex-literals.md b/docs/regex-literals.md index a3feae1b2..b0763732c 100644 --- a/docs/regex-literals.md +++ b/docs/regex-literals.md @@ -9,5 +9,5 @@ print /hello world/ig transpiles to: ```BrightScript -print CreateObject("roRegex","hello world","ig") +print CreateObject("roRegex", "hello world", "ig") ``` diff --git a/docs/ternary-operator.md b/docs/ternary-operator.md index 5ad90e382..17809a02c 100644 --- a/docs/ternary-operator.md +++ b/docs/ternary-operator.md @@ -10,7 +10,7 @@ authStatus = user <> invalid ? "logged in" : "not logged in" transpiles to: ```BrightScript -a = bslib_ternary(user = invalid, "no user", "logged in") +authStatus = bslib_ternary(user <> invalid, "logged in", "not logged in") ``` The `bslib_ternary` function checks the condition, and returns either the consequent or alternate. From c5828b433a81c5970797a99dec53dd814667a99a Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 27 Sep 2021 08:54:26 -0400 Subject: [PATCH 053/278] update changelog for v0.41.0 --- CHANGELOG.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index f90ce28ef..c263e6939 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.41.0](https://github.com/rokucommunity/brighterscript/compare/v0.40.1...v0.41.0) - 2021-09-27 +### Added + - Regex literal support in brighterscript ([#452](https://github.com/rokucommunity/brighterscript/pull/452)). + + + ## [0.40.1](https://github.com/rokucommunity/brighterscript/compare/v0.40.0...v0.40.1) - 2021-09-17 ### Changed - install roku-deploy@3.4.2 which prevents deploy crashes when target Roku doesn't have an installed channel ([roku-deploy#65](https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0)) From 859af826650b34fe8107d53dbc943fd5c023168b Mon Sep 17 00:00:00 2001 From: Bronley Date: Mon, 27 Sep 2021 08:57:20 -0400 Subject: [PATCH 054/278] 0.41.0 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 99ae83b54..89820d4f3 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.40.1", + "version": "0.41.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index af5bd1221..acbf8eacf 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.40.1", + "version": "0.41.0", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 67c544b427cf58ee87e535cf79d430a40c6a1481 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Mon, 4 Oct 2021 12:15:22 -0400 Subject: [PATCH 055/278] Upgrade typescript and eslint (#456) * typescript@4.4.3 * Upgrade eslint. * Fix lint issues. --- .eslintrc.js | 16 +- package-lock.json | 1431 ++++++++++++++++++++++-------------- package.json | 12 +- src/BsConfig.ts | 4 +- src/LanguageServer.ts | 12 +- src/ProgramBuilder.ts | 2 +- src/astUtils/reflection.ts | 5 +- src/files/BrsFile.ts | 4 +- src/lexer/Lexer.ts | 2 +- src/parser/Parser.ts | 2 +- src/parser/Statement.ts | 10 +- src/util.ts | 2 +- 12 files changed, 931 insertions(+), 571 deletions(-) diff --git a/.eslintrc.js b/.eslintrc.js index 32e6cfbd2..d7257c074 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -31,6 +31,7 @@ module.exports = { '@typescript-eslint/method-signature-style': 'off', '@typescript-eslint/naming-convention': 'off', '@typescript-eslint/no-base-to-string': 'off', + '@typescript-eslint/no-confusing-void-expression': 'off', '@typescript-eslint/no-dynamic-delete': 'off', '@typescript-eslint/no-empty-function': 'off', '@typescript-eslint/no-explicit-any': 'off', @@ -40,6 +41,15 @@ module.exports = { '@typescript-eslint/no-invalid-this': 'off', '@typescript-eslint/no-magic-numbers': 'off', '@typescript-eslint/no-parameter-properties': 'off', + //had to add this rule to prevent eslint from crashing + '@typescript-eslint/no-restricted-imports': ['off', {}], + //mitigating this sometimes results in undesirably verbose code. Should investigate enabling again in the future. + '@typescript-eslint/no-unsafe-argument': 'off', + 'object-curly-spacing': 'off', + '@typescript-eslint/object-curly-spacing': [ + 'error', + 'always' + ], '@typescript-eslint/no-shadow': 'off', '@typescript-eslint/no-this-alias': 'off', //possibly disable this once we have converted all throw statements to actual errors @@ -70,6 +80,7 @@ module.exports = { '@typescript-eslint/require-array-sort-compare': 'off', '@typescript-eslint/restrict-plus-operands': 'off', '@typescript-eslint/restrict-template-expressions': 'off', + '@typescript-eslint/sort-type-union-intersection-members': 'off', '@typescript-eslint/space-before-function-paren': 'off', '@typescript-eslint/strict-boolean-expressions': 'off', '@typescript-eslint/typedef': 'off', @@ -140,10 +151,6 @@ module.exports = { 'no-unneeded-ternary': 'off', 'no-useless-escape': 'off', 'no-warning-comments': 'off', - 'object-curly-spacing': [ - 'error', - 'always' - ], 'object-property-newline': 'off', 'object-shorthand': [ 'error', @@ -190,6 +197,7 @@ module.exports = { '@typescript-eslint/no-unused-vars': 'off', '@typescript-eslint/no-unused-vars-experimental': 'off', '@typescript-eslint/dot-notation': 'off', + '@typescript-eslint/no-unsafe-argument': 'off', 'github/array-foreach': 'off', 'new-cap': 'off', 'no-shadow': 'off', diff --git a/package-lock.json b/package-lock.json index 89820d4f3..06900d891 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5,9 +5,9 @@ "requires": true, "dependencies": { "@babel/code-frame": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", - "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.12.11.tgz", + "integrity": "sha512-Zt1yodBx1UcyiePMSkWnU4hPqhwq7hGi2nFL1LeA3EUl+q2LQx16MISgJ0+z7dnmgvP9QtIleuETGOiOH1RcIw==", "dev": true, "requires": { "@babel/highlight": "^7.10.4" @@ -229,9 +229,9 @@ } }, "@babel/helper-validator-identifier": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", - "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==", + "version": "7.15.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.15.7.tgz", + "integrity": "sha512-K4JvCtQqad9OY2+yTU8w+E82ywk/fe+ELNlt1G8z3bVGlZfn/hOcQQsUhGhW/N+tb3fxK800wLtKOE/aM0m72w==", "dev": true }, "@babel/helper-validator-option": { @@ -252,12 +252,12 @@ } }, "@babel/highlight": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", - "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", + "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", "dev": true, "requires": { - "@babel/helper-validator-identifier": "^7.10.4", + "@babel/helper-validator-identifier": "^7.14.5", "chalk": "^2.0.0", "js-tokens": "^4.0.0" } @@ -389,19 +389,18 @@ } }, "@eslint/eslintrc": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.1.3.tgz", - "integrity": "sha512-4YVwPkANLeNtRjMekzux1ci8hIaH5eGKktGqR0d3LWsKNn5B2X/1Z6Trxy7jQXl9EBGE6Yj02O+t09FMeRllaA==", + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.3.tgz", + "integrity": "sha512-J6KFFz5QCYUJq3pf0mjEcCJVERbzv71PUIDczuh9JkwGEzced6CO5ADLHB1rbf/+oPBtoPfMYNOpGDzCANlbXw==", "dev": true, "requires": { "ajv": "^6.12.4", "debug": "^4.1.1", "espree": "^7.3.0", - "globals": "^12.1.0", + "globals": "^13.9.0", "ignore": "^4.0.6", "import-fresh": "^3.2.1", "js-yaml": "^3.13.1", - "lodash": "^4.17.19", "minimatch": "^3.0.4", "strip-json-comments": "^3.1.1" }, @@ -414,6 +413,23 @@ } } }, + "@humanwhocodes/config-array": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.5.0.tgz", + "integrity": "sha512-FagtKFz74XrTl7y6HCzQpwDfXP0yhxe9lHLD1UZxjvZIcbyRz8zTFF/yYNfSfzU414eDwZ1SrO0Qvtyf+wFMQg==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^1.2.0", + "debug": "^4.1.1", + "minimatch": "^3.0.4" + } + }, + "@humanwhocodes/object-schema": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.0.tgz", + "integrity": "sha512-wdppn25U8z/2yiaT6YGquE6X8sSv7hNMWSXYSSU1jGv/yd6XqjXgTDJ8KP4NgjTXfJ3GbRjeeb8RTV7a/VpM+w==", + "dev": true + }, "@istanbuljs/load-nyc-config": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", @@ -600,9 +616,9 @@ } }, "@types/json-schema": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.6.tgz", - "integrity": "sha512-3c+yGKvVP5Y9TYBEibGNR+kLtijnj7mYrXRg+WpFb2X9xm04g/DXYkfg4hmzJQosc9snFNUPkbYIhu+KAm6jJw==", + "version": "7.0.9", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.9.tgz", + "integrity": "sha512-qcUXuemtEu+E5wZSJHNxUXeCZhAfXKQ41D+duX+VYPde7xyEVZci+/oXKJL13tnRs9lR2pr4fod59GT6/X1/yQ==", "dev": true }, "@types/json5": { @@ -697,85 +713,198 @@ "dev": true }, "@typescript-eslint/eslint-plugin": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.4.1.tgz", - "integrity": "sha512-O+8Utz8pb4OmcA+Nfi5THQnQpHSD2sDUNw9AxNHpuYOo326HZTtG8gsfT+EAYuVrFNaLyNb2QnUNkmTRDskuRA==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.32.0.tgz", + "integrity": "sha512-+OWTuWRSbWI1KDK8iEyG/6uK2rTm3kpS38wuVifGUTDB6kjEuNrzBI1MUtxnkneuWG/23QehABe2zHHrj+4yuA==", "dev": true, "requires": { - "@typescript-eslint/experimental-utils": "4.4.1", - "@typescript-eslint/scope-manager": "4.4.1", - "debug": "^4.1.1", + "@typescript-eslint/experimental-utils": "4.32.0", + "@typescript-eslint/scope-manager": "4.32.0", + "debug": "^4.3.1", "functional-red-black-tree": "^1.0.1", - "regexpp": "^3.0.0", - "semver": "^7.3.2", - "tsutils": "^3.17.1" + "ignore": "^5.1.8", + "regexpp": "^3.1.0", + "semver": "^7.3.5", + "tsutils": "^3.21.0" + }, + "dependencies": { + "debug": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "tsutils": { + "version": "3.21.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + } + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + } } }, "@typescript-eslint/experimental-utils": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.4.1.tgz", - "integrity": "sha512-Nt4EVlb1mqExW9cWhpV6pd1a3DkUbX9DeyYsdoeziKOpIJ04S2KMVDO+SEidsXRH/XHDpbzXykKcMTLdTXH6cQ==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.32.0.tgz", + "integrity": "sha512-WLoXcc+cQufxRYjTWr4kFt0DyEv6hDgSaFqYhIzQZ05cF+kXfqXdUh+//kgquPJVUBbL3oQGKQxwPbLxHRqm6A==", "dev": true, "requires": { - "@types/json-schema": "^7.0.3", - "@typescript-eslint/scope-manager": "4.4.1", - "@typescript-eslint/types": "4.4.1", - "@typescript-eslint/typescript-estree": "4.4.1", - "eslint-scope": "^5.0.0", - "eslint-utils": "^2.0.0" + "@types/json-schema": "^7.0.7", + "@typescript-eslint/scope-manager": "4.32.0", + "@typescript-eslint/types": "4.32.0", + "@typescript-eslint/typescript-estree": "4.32.0", + "eslint-scope": "^5.1.1", + "eslint-utils": "^3.0.0" } }, "@typescript-eslint/parser": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.4.1.tgz", - "integrity": "sha512-S0fuX5lDku28Au9REYUsV+hdJpW/rNW0gWlc4SXzF/kdrRaAVX9YCxKpziH7djeWT/HFAjLZcnY7NJD8xTeUEg==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.32.0.tgz", + "integrity": "sha512-lhtYqQ2iEPV5JqV7K+uOVlPePjClj4dOw7K4/Z1F2yvjIUvyr13yJnDzkK6uon4BjHYuHy3EG0c2Z9jEhFk56w==", "dev": true, "requires": { - "@typescript-eslint/scope-manager": "4.4.1", - "@typescript-eslint/types": "4.4.1", - "@typescript-eslint/typescript-estree": "4.4.1", - "debug": "^4.1.1" + "@typescript-eslint/scope-manager": "4.32.0", + "@typescript-eslint/types": "4.32.0", + "@typescript-eslint/typescript-estree": "4.32.0", + "debug": "^4.3.1" + }, + "dependencies": { + "debug": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + } } }, "@typescript-eslint/scope-manager": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.4.1.tgz", - "integrity": "sha512-2oD/ZqD4Gj41UdFeWZxegH3cVEEH/Z6Bhr/XvwTtGv66737XkR4C9IqEkebCuqArqBJQSj4AgNHHiN1okzD/wQ==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.32.0.tgz", + "integrity": "sha512-DK+fMSHdM216C0OM/KR1lHXjP1CNtVIhJ54kQxfOE6x8UGFAjha8cXgDMBEIYS2XCYjjCtvTkjQYwL3uvGOo0w==", "dev": true, "requires": { - "@typescript-eslint/types": "4.4.1", - "@typescript-eslint/visitor-keys": "4.4.1" + "@typescript-eslint/types": "4.32.0", + "@typescript-eslint/visitor-keys": "4.32.0" } }, "@typescript-eslint/types": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.4.1.tgz", - "integrity": "sha512-KNDfH2bCyax5db+KKIZT4rfA8rEk5N0EJ8P0T5AJjo5xrV26UAzaiqoJCxeaibqc0c/IvZxp7v2g3difn2Pn3w==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.32.0.tgz", + "integrity": "sha512-LE7Z7BAv0E2UvqzogssGf1x7GPpUalgG07nGCBYb1oK4mFsOiFC/VrSMKbZQzFJdN2JL5XYmsx7C7FX9p9ns0w==", "dev": true }, "@typescript-eslint/typescript-estree": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.4.1.tgz", - "integrity": "sha512-wP/V7ScKzgSdtcY1a0pZYBoCxrCstLrgRQ2O9MmCUZDtmgxCO/TCqOTGRVwpP4/2hVfqMz/Vw1ZYrG8cVxvN3g==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.32.0.tgz", + "integrity": "sha512-tRYCgJ3g1UjMw1cGG8Yn1KzOzNlQ6u1h9AmEtPhb5V5a1TmiHWcRyF/Ic+91M4f43QeChyYlVTcf3DvDTZR9vw==", "dev": true, "requires": { - "@typescript-eslint/types": "4.4.1", - "@typescript-eslint/visitor-keys": "4.4.1", - "debug": "^4.1.1", - "globby": "^11.0.1", + "@typescript-eslint/types": "4.32.0", + "@typescript-eslint/visitor-keys": "4.32.0", + "debug": "^4.3.1", + "globby": "^11.0.3", "is-glob": "^4.0.1", - "lodash": "^4.17.15", - "semver": "^7.3.2", - "tsutils": "^3.17.1" + "semver": "^7.3.5", + "tsutils": "^3.21.0" + }, + "dependencies": { + "debug": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "globby": { + "version": "11.0.4", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.4.tgz", + "integrity": "sha512-9O4MVG9ioZJ08ffbcyVYyLOJLk5JQ688pJ4eMGLpdWLHq/Wr1D9BlriLQyL0E+jbkuePVZXYFj47QM/v093wHg==", + "dev": true, + "requires": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.1.1", + "ignore": "^5.1.4", + "merge2": "^1.3.0", + "slash": "^3.0.0" + } + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "tsutils": { + "version": "3.21.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + } + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + } } }, "@typescript-eslint/visitor-keys": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.4.1.tgz", - "integrity": "sha512-H2JMWhLaJNeaylSnMSQFEhT/S/FsJbebQALmoJxMPMxLtlVAMy2uJP/Z543n9IizhjRayLSqoInehCeNW9rWcw==", + "version": "4.32.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.32.0.tgz", + "integrity": "sha512-e7NE0qz8W+atzv3Cy9qaQ7BTLwWsm084Z0c4nIO2l3Bp6u9WIgdqCgyPyV5oSPDMIW3b20H59OOCmVk3jw3Ptw==", "dev": true, "requires": { - "@typescript-eslint/types": "4.4.1", + "@typescript-eslint/types": "4.32.0", "eslint-visitor-keys": "^2.0.0" } }, @@ -810,9 +939,9 @@ "dev": true }, "acorn-jsx": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", - "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", "dev": true }, "aggregate-error": { @@ -906,50 +1035,16 @@ "integrity": "sha512-hfJmKupmQN0lwi0xG6FQ5U8Rd97RnIERplymOv/qpq8AoNKPPAnxJadjFA23FNWm88wykh9HmpLJUUwUtNU/iw==" }, "array-includes": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.2.tgz", - "integrity": "sha512-w2GspexNQpx+PutG3QpT437/BenZBj0M/MZGn5mzv/MofYqo0xmRHzn4lFsoDlWJ+THYsGJmFlW68WlDFx7VRw==", + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.3.tgz", + "integrity": "sha512-gcem1KlBU7c9rB+Rq8/3PPKsK2kjqeEBa3bD5kkQo4nYlOHQCJqIJFqBXDEfwaRuYTT4E+FxA9xez7Gf/e3Q7A==", "dev": true, "requires": { - "call-bind": "^1.0.0", + "call-bind": "^1.0.2", "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1", - "get-intrinsic": "^1.0.1", + "es-abstract": "^1.18.0-next.2", + "get-intrinsic": "^1.1.1", "is-string": "^1.0.5" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "object.assign": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", - "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", - "dev": true, - "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "has-symbols": "^1.0.1", - "object-keys": "^1.1.1" - } - } } }, "array-union": { @@ -959,48 +1054,14 @@ "dev": true }, "array.prototype.flat": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.4.tgz", - "integrity": "sha512-4470Xi3GAPAjZqFcljX2xzckv1qeKPizoNkiS0+O4IoPR2ZNpcjE0pkhdihlDouK+x6QOast26B4Q/O9DJnwSg==", + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.5.tgz", + "integrity": "sha512-KaYU+S+ndVqyUnignHftkwc58o3uVU1jzczILJ1tN2YaIZpFIKBiP/x/j97E5MVPsaCloPbqWLB/8qCTVvT2qg==", "dev": true, "requires": { - "call-bind": "^1.0.0", + "call-bind": "^1.0.2", "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "object.assign": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", - "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", - "dev": true, - "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "has-symbols": "^1.0.1", - "object-keys": "^1.1.1" - } - } + "es-abstract": "^1.19.0" } }, "asap": { @@ -1035,9 +1096,9 @@ "dev": true }, "astral-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", - "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", + "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", "dev": true }, "asynckit": { @@ -1171,9 +1232,9 @@ } }, "call-bind": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.1.tgz", - "integrity": "sha512-tvAvUwNcRikl3RVF20X9lsYmmepsovzTWeJiXjO0PkJp15uy/6xKFZOQtuiSULwYW+6ToZBprphCgWXC2dSgcQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", "dev": true, "requires": { "function-bind": "^1.1.1", @@ -1386,12 +1447,6 @@ } } }, - "contains-path": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", - "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", - "dev": true - }, "convert-source-map": { "version": "1.7.0", "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", @@ -1520,6 +1575,7 @@ "version": "1.1.3", "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "dev": true, "requires": { "object-keys": "^1.0.12" } @@ -1556,6 +1612,12 @@ "requires": { "ms": "2.1.2" } + }, + "typescript": { + "version": "3.9.10", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", + "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", + "dev": true } } }, @@ -1683,6 +1745,12 @@ "@typescript-eslint/types": "4.20.0", "eslint-visitor-keys": "^2.0.0" } + }, + "typescript": { + "version": "3.9.10", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", + "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", + "dev": true } } }, @@ -1785,80 +1853,38 @@ } }, "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.19.1.tgz", + "integrity": "sha512-2vJ6tjA/UfqLm2MPs7jxVybLoB8i1t1Jd9R3kISld20sIxPcTbLuggQOUxeWeAvIUkduv/CfMjuh4WmiXr2v9w==", + "dev": true, "requires": { + "call-bind": "^1.0.2", "es-to-primitive": "^1.2.1", "function-bind": "^1.1.1", + "get-intrinsic": "^1.1.1", + "get-symbol-description": "^1.0.0", "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", + "has-symbols": "^1.0.2", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.4", + "is-negative-zero": "^2.0.1", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.1", + "is-string": "^1.0.7", + "is-weakref": "^1.0.1", + "object-inspect": "^1.11.0", "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "object.assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.1.tgz", - "integrity": "sha512-VT/cxmx5yaoHSOTSyrCygIDFco+RsibY2NM0a4RdEeY/4KgqezwFtK1yr3U67xYhqJSlASm2pKhLVzPj2lr4bA==", - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.0", - "has-symbols": "^1.0.1", - "object-keys": "^1.1.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - } + "object.assign": "^4.1.2", + "string.prototype.trimend": "^1.0.4", + "string.prototype.trimstart": "^1.0.4", + "unbox-primitive": "^1.0.1" } }, "es-to-primitive": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, "requires": { "is-callable": "^1.1.4", "is-date-object": "^1.0.1", @@ -1949,29 +1975,32 @@ } }, "eslint": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.11.0.tgz", - "integrity": "sha512-G9+qtYVCHaDi1ZuWzBsOWo2wSwd70TXnU6UHA3cTYHp7gCTXZcpggWFoUVAMRarg68qtPoNfFbzPh+VdOgmwmw==", + "version": "7.32.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.32.0.tgz", + "integrity": "sha512-VHZ8gX+EDfz+97jGcgyGCyRia/dPOd6Xh9yPv8Bl1+SoaIwD+a/vlrOmGRUyOYu7MwUhc7CxqeaDZU13S4+EpA==", "dev": true, "requires": { - "@babel/code-frame": "^7.0.0", - "@eslint/eslintrc": "^0.1.3", + "@babel/code-frame": "7.12.11", + "@eslint/eslintrc": "^0.4.3", + "@humanwhocodes/config-array": "^0.5.0", "ajv": "^6.10.0", "chalk": "^4.0.0", "cross-spawn": "^7.0.2", "debug": "^4.0.1", "doctrine": "^3.0.0", "enquirer": "^2.3.5", + "escape-string-regexp": "^4.0.0", "eslint-scope": "^5.1.1", "eslint-utils": "^2.1.0", "eslint-visitor-keys": "^2.0.0", - "espree": "^7.3.0", - "esquery": "^1.2.0", + "espree": "^7.3.1", + "esquery": "^1.4.0", "esutils": "^2.0.2", - "file-entry-cache": "^5.0.1", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", "functional-red-black-tree": "^1.0.1", - "glob-parent": "^5.0.0", - "globals": "^12.1.0", + "glob-parent": "^5.1.2", + "globals": "^13.6.0", "ignore": "^4.0.6", "import-fresh": "^3.0.0", "imurmurhash": "^0.1.4", @@ -1979,7 +2008,7 @@ "js-yaml": "^3.13.1", "json-stable-stringify-without-jsonify": "^1.0.1", "levn": "^0.4.1", - "lodash": "^4.17.19", + "lodash.merge": "^4.6.2", "minimatch": "^3.0.4", "natural-compare": "^1.4.0", "optionator": "^0.9.1", @@ -1988,7 +2017,7 @@ "semver": "^7.2.1", "strip-ansi": "^6.0.0", "strip-json-comments": "^3.1.0", - "table": "^5.2.3", + "table": "^6.0.9", "text-table": "^0.2.0", "v8-compile-cache": "^2.0.3" }, @@ -2003,9 +2032,9 @@ } }, "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, "requires": { "ansi-styles": "^4.1.0", @@ -2027,6 +2056,29 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "eslint-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", + "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } + } + }, "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -2051,55 +2103,59 @@ } }, "eslint-config-prettier": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-7.1.0.tgz", - "integrity": "sha512-9sm5/PxaFG7qNJvJzTROMM1Bk1ozXVTKI0buKOyb0Bsr1hrwi0H/TzxF/COtf1uxikIK8SwhX7K6zg78jAzbeA==", + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.3.0.tgz", + "integrity": "sha512-BgZuLUSeKzvlL/VUjx/Yb787VQ26RU3gGjA3iiFvdsp/2bMfVIWUVP7tjxtjS0e+HP409cPlPvNkQloz8C91ew==", "dev": true }, "eslint-import-resolver-node": { - "version": "0.3.4", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.4.tgz", - "integrity": "sha512-ogtf+5AB/O+nM6DIeBUNr2fuT7ot9Qg/1harBfBtaP13ekEWFQEEMP94BCB7zaNW3gyY+8SHYF00rnqYwXKWOA==", + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.6.tgz", + "integrity": "sha512-0En0w03NRVMn9Uiyn8YRPDKvWjxCWkslUEhGNTdGx15RvPJYQ+lbOlqrlNI2vEAs4pDYK4f/HN2TbDmk5TP0iw==", "dev": true, "requires": { - "debug": "^2.6.9", - "resolve": "^1.13.1" + "debug": "^3.2.7", + "resolve": "^1.20.0" }, "dependencies": { "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { - "ms": "2.0.0" + "ms": "^2.1.1" } }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true + "resolve": { + "version": "1.20.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz", + "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==", + "dev": true, + "requires": { + "is-core-module": "^2.2.0", + "path-parse": "^1.0.6" + } } } }, "eslint-module-utils": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.0.tgz", - "integrity": "sha512-6j9xxegbqe8/kZY8cYpcp0xhbK0EgJlg3g9mib3/miLaExuuwc3n5UEfSnU6hWMbT0FAYVvDbL9RrRgpUeQIvA==", + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.2.tgz", + "integrity": "sha512-QG8pcgThYOuqxupd06oYTZoNOGaUdTY1PqK+oS6ElF6vs4pBdk/aYxFVQQXzcrAqp9m7cl7lb2ubazX+g16k2Q==", "dev": true, "requires": { - "debug": "^2.6.9", + "debug": "^3.2.7", "pkg-dir": "^2.0.0" }, "dependencies": { "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { - "ms": "2.0.0" + "ms": "^2.1.1" } }, "find-up": { @@ -2121,12 +2177,6 @@ "path-exists": "^3.0.0" } }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, "p-locate": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", @@ -2163,42 +2213,65 @@ "ignore": "^5.0.5" } }, - "eslint-plugin-github": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-github/-/eslint-plugin-github-4.1.1.tgz", - "integrity": "sha512-MzCh4P4zVvR/13AHtumzZ3znq0cbUE7lXehyBEpFURD/EHdx/+7qW+0c+ySTrteImpX9LGLJFTYNtu10BifkbQ==", + "eslint-plugin-filenames": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-filenames/-/eslint-plugin-filenames-1.3.2.tgz", + "integrity": "sha512-tqxJTiEM5a0JmRCUYQmxw23vtTxrb2+a3Q2mMOPhFxvt7ZQQJmdiuMby9B/vUAuVMghyP7oET+nIf6EO6CBd/w==", "dev": true, "requires": { - "@typescript-eslint/eslint-plugin": ">=2.25.0", - "@typescript-eslint/parser": ">=2.25.0", - "eslint-config-prettier": ">=6.10.1", - "eslint-plugin-eslint-comments": ">=3.0.1", - "eslint-plugin-import": ">=2.20.1", - "eslint-plugin-prettier": ">=3.1.2", + "lodash.camelcase": "4.3.0", + "lodash.kebabcase": "4.1.1", + "lodash.snakecase": "4.1.1", + "lodash.upperfirst": "4.3.1" + } + }, + "eslint-plugin-github": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-github/-/eslint-plugin-github-4.3.0.tgz", + "integrity": "sha512-WZ3RCtxSYzF5sIvNykX+SnNlJ+r8He7mjr8EoXKF+01MP+/PsOTrrmW0LrqluVTtYEslakjenQdItw4RNLD0XQ==", + "dev": true, + "requires": { + "@typescript-eslint/eslint-plugin": "^4.20.0", + "@typescript-eslint/parser": "^4.20.0", + "eslint-config-prettier": ">=8.0.0", + "eslint-plugin-eslint-comments": "^3.2.0", + "eslint-plugin-filenames": "^1.3.2", + "eslint-plugin-i18n-text": "^1.0.1", + "eslint-plugin-import": "^2.22.1", + "eslint-plugin-no-only-tests": "^2.6.0", + "eslint-plugin-prettier": "^3.3.1", "eslint-rule-documentation": ">=1.0.0", - "prettier": ">=1.12.0", - "svg-element-attributes": ">=1.3.1" + "prettier": "^2.2.1", + "svg-element-attributes": "^1.3.1" } }, + "eslint-plugin-i18n-text": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-i18n-text/-/eslint-plugin-i18n-text-1.0.1.tgz", + "integrity": "sha512-3G3UetST6rdqhqW9SfcfzNYMpQXS7wNkJvp6dsXnjzGiku6Iu5hl3B0kmk6lIcFPwYjhQIY+tXVRtK9TlGT7RA==", + "dev": true + }, "eslint-plugin-import": { - "version": "2.22.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.22.1.tgz", - "integrity": "sha512-8K7JjINHOpH64ozkAhpT3sd+FswIZTfMZTjdx052pnWrgRCVfp8op9tbjpAk3DdUeI/Ba4C8OjdC0r90erHEOw==", + "version": "2.24.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.24.2.tgz", + "integrity": "sha512-hNVtyhiEtZmpsabL4neEj+6M5DCLgpYyG9nzJY8lZQeQXEn5UPW1DpUdsMHMXsq98dbNm7nt1w9ZMSVpfJdi8Q==", "dev": true, "requires": { - "array-includes": "^3.1.1", - "array.prototype.flat": "^1.2.3", - "contains-path": "^0.1.0", + "array-includes": "^3.1.3", + "array.prototype.flat": "^1.2.4", "debug": "^2.6.9", - "doctrine": "1.5.0", - "eslint-import-resolver-node": "^0.3.4", - "eslint-module-utils": "^2.6.0", + "doctrine": "^2.1.0", + "eslint-import-resolver-node": "^0.3.6", + "eslint-module-utils": "^2.6.2", + "find-up": "^2.0.0", "has": "^1.0.3", + "is-core-module": "^2.6.0", "minimatch": "^3.0.4", - "object.values": "^1.1.1", - "read-pkg-up": "^2.0.0", - "resolve": "^1.17.0", - "tsconfig-paths": "^3.9.0" + "object.values": "^1.1.4", + "pkg-up": "^2.0.0", + "read-pkg-up": "^3.0.0", + "resolve": "^1.20.0", + "tsconfig-paths": "^3.11.0" }, "dependencies": { "debug": { @@ -2211,13 +2284,40 @@ } }, "doctrine": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", - "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", "dev": true, "requires": { - "esutils": "^2.0.2", - "isarray": "^1.0.0" + "esutils": "^2.0.2" + } + }, + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "is-core-module": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.7.0.tgz", + "integrity": "sha512-ByY+tjCciCr+9nLryBYcSD50EOGWt95c7tIsKTG1J2ixKKXPvF7Ej3AVd+UfDydAJom3biBGDBALaO79ktwgEQ==", + "dev": true, + "requires": { + "has": "^1.0.3" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" } }, "ms": { @@ -2225,19 +2325,44 @@ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", "dev": true + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "resolve": { + "version": "1.20.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz", + "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==", + "dev": true, + "requires": { + "is-core-module": "^2.2.0", + "path-parse": "^1.0.6" + } } } }, "eslint-plugin-no-only-tests": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-no-only-tests/-/eslint-plugin-no-only-tests-2.4.0.tgz", - "integrity": "sha512-azP9PwQYfGtXJjW273nIxQH9Ygr+5/UyeW2wEjYoDtVYPI+WPKwbj0+qcAKYUXFZLRumq4HKkFaoDBAwBoXImQ==", + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-no-only-tests/-/eslint-plugin-no-only-tests-2.6.0.tgz", + "integrity": "sha512-T9SmE/g6UV1uZo1oHAqOvL86XWl7Pl2EpRpnLI8g/bkJu+h7XBCB+1LnubRZ2CUQXj805vh4/CYZdnqtVaEo2Q==", "dev": true }, "eslint-plugin-prettier": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.3.1.tgz", - "integrity": "sha512-Rq3jkcFY8RYeQLgk2cCwuc0P7SEFwDravPhsJZOQ5N4YI4DSg50NyqJ/9gdZHzQlHf8MvafSesbNJCcP/FF6pQ==", + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.4.1.tgz", + "integrity": "sha512-htg25EUYUeIhKHXjOinK4BgCcDwtLHjqaxCDsMy5nbnUMkKFvIhMVCp+5GFUXQ4Nr8lBsPqtGAqBenbpFqAA2g==", "dev": true, "requires": { "prettier-linter-helpers": "^1.0.0" @@ -2260,20 +2385,12 @@ } }, "eslint-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", - "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", + "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", "dev": true, "requires": { - "eslint-visitor-keys": "^1.1.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true - } + "eslint-visitor-keys": "^2.0.0" } }, "eslint-visitor-keys": { @@ -2283,13 +2400,13 @@ "dev": true }, "espree": { - "version": "7.3.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-7.3.0.tgz", - "integrity": "sha512-dksIWsvKCixn1yrEXO8UosNSxaDoSYpq9reEjZSbHLpT5hpaCAKTLBwq0RHtLrIr+c0ByiYzWT8KTMRzoRCNlw==", + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-7.3.1.tgz", + "integrity": "sha512-v3JCNCE64umkFpmkFGqzVKsOT0tN1Zr+ueqLZfpV1Ob8e+CEgPWa+OxCoGH3tnhimMKIaBm4m/vaRpJ/krRz2g==", "dev": true, "requires": { "acorn": "^7.4.0", - "acorn-jsx": "^5.2.0", + "acorn-jsx": "^5.3.1", "eslint-visitor-keys": "^1.3.0" }, "dependencies": { @@ -2308,9 +2425,9 @@ "dev": true }, "esquery": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", - "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", + "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", "dev": true, "requires": { "estraverse": "^5.1.0" @@ -2414,12 +2531,12 @@ } }, "file-entry-cache": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", - "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", "dev": true, "requires": { - "flat-cache": "^2.0.1" + "flat-cache": "^3.0.4" } }, "file-url": { @@ -2472,6 +2589,12 @@ "is-core-module": "^2.2.0", "path-parse": "^1.0.6" } + }, + "typescript": { + "version": "3.9.10", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", + "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", + "dev": true } } }, @@ -2528,20 +2651,19 @@ "dev": true }, "flat-cache": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", - "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", + "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", "dev": true, "requires": { - "flatted": "^2.0.0", - "rimraf": "2.6.3", - "write": "1.0.3" + "flatted": "^3.1.0", + "rimraf": "^3.0.2" }, "dependencies": { "rimraf": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", - "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", "dev": true, "requires": { "glob": "^7.1.3" @@ -2550,9 +2672,9 @@ } }, "flatted": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", - "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.2.tgz", + "integrity": "sha512-JaTY/wtrcSyvXJl4IMFHPKyFur1sE9AUqc0QnhOaJ0CxHtAoIV8pYDzeEfAaNEtGkOfq4gr3LBFmdXW5mOQFnA==", "dev": true }, "flatten": { @@ -2616,7 +2738,8 @@ "function-bind": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true }, "functional-red-black-tree": { "version": "1.0.1", @@ -2652,9 +2775,9 @@ "dev": true }, "get-intrinsic": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.0.2.tgz", - "integrity": "sha512-aeX0vrFm21ILl3+JpFFRNe9aUvp6VFZb2/CTbgLb8j75kOhvoNYjt9d8KA/tJG4gSo8nzEDedRl0h7vDmBYRVg==", + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", "dev": true, "requires": { "function-bind": "^1.1.1", @@ -2680,6 +2803,16 @@ "integrity": "sha1-3Xzn3hh8Bsi/NTeWrHHgmfCYDrw=", "dev": true }, + "get-symbol-description": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", + "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + } + }, "getpass": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", @@ -2710,18 +2843,18 @@ } }, "globals": { - "version": "12.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", - "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "version": "13.11.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.11.0.tgz", + "integrity": "sha512-08/xrJ7wQjK9kkkRoI3OFUBbLx4f+6x3SGwcPvQ0QH6goFDrOU2oyAWrmh3dJezu65buo+HBMzAMQy6rovVC3g==", "dev": true, "requires": { - "type-fest": "^0.8.1" + "type-fest": "^0.20.2" }, "dependencies": { "type-fest": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", - "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", "dev": true } } @@ -2787,19 +2920,36 @@ "version": "1.0.3", "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, "requires": { "function-bind": "^1.1.1" } }, + "has-bigints": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.1.tgz", + "integrity": "sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA==", + "dev": true + }, "has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" }, "has-symbols": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", - "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==" + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", + "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==", + "dev": true + }, + "has-tostringtag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", + "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } }, "hasha": { "version": "5.2.2", @@ -2894,9 +3044,9 @@ "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=" }, "import-fresh": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.1.tgz", - "integrity": "sha512-6e1q1cnWP2RXD9/keSkxHScg508CdXqXWgWBaETNhyuBFz+kUZlKboh+ISK+bU++DmbHimVBrOz/zzPe0sZ3sQ==", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", "dev": true, "requires": { "parent-module": "^1.0.0", @@ -2941,12 +3091,32 @@ "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", "dev": true }, + "internal-slot": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.3.tgz", + "integrity": "sha512-O0DB1JC/sPyZl7cIo78n5dR7eUSwwpYPiXRhTzNxZVAMUuB8vlnRFyLxdrVToks6XPLVnFfbzaVd5WLjhgg+vA==", + "dev": true, + "requires": { + "get-intrinsic": "^1.1.0", + "has": "^1.0.3", + "side-channel": "^1.0.4" + } + }, "is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", "dev": true }, + "is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dev": true, + "requires": { + "has-bigints": "^1.0.1" + } + }, "is-binary-path": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", @@ -2955,10 +3125,21 @@ "binary-extensions": "^2.0.0" } }, + "is-boolean-object": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", + "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, "is-callable": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.2.tgz", - "integrity": "sha512-dnMqspv5nU3LoewK2N/y7KLtxtakvTuaCsU9FU50/QDmdbHNy/4/JuRtMHqRU22o3q+W89YQndQEeCVwK+3qrA==" + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.4.tgz", + "integrity": "sha512-nsuwtxZfMX67Oryl9LCQ+upnC0Z0BgpwntpS89m1H/TLF0zNfzfLMV/9Wa/6MZsj0acpEjAO0KF1xT6ZdLl95w==", + "dev": true }, "is-core-module": { "version": "2.2.0", @@ -2970,9 +3151,13 @@ } }, "is-date-object": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", - "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==" + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } }, "is-extglob": { "version": "2.1.1", @@ -2999,15 +3184,25 @@ "dev": true }, "is-negative-zero": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.0.tgz", - "integrity": "sha1-lVOxIbD6wohp2p7UWeIMdUN4hGE=" + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.1.tgz", + "integrity": "sha512-2z6JzQvZRa9A2Y7xC6dQQm4FSTSTNWjKIYYTt4246eMTJmIo0Q+ZyOsU66X8lxK1AbB92dFeglPLrhwpeRKO6w==", + "dev": true }, "is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" }, + "is-number-object": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.6.tgz", + "integrity": "sha512-bEVOqiRcvo3zO1+G2lVMy+gkkEm9Yh7cDMRusKKu5ZJKPUYSJwICTKZrNKHA2EbSP0Tu0+6B/emsYNHZyn6K8g==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, "is-obj": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", @@ -3021,11 +3216,13 @@ "dev": true }, "is-regex": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz", - "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==", + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, "requires": { - "has-symbols": "^1.0.1" + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" } }, "is-regexp": { @@ -3040,6 +3237,12 @@ "integrity": "sha1-CRtGoNZ8HtD+hfH4z93gBrslHUY=", "dev": true }, + "is-shared-array-buffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.1.tgz", + "integrity": "sha512-IU0NmyknYZN0rChcKhRO1X8LYz5Isj/Fsqh8NJOSf+N/hCOTwy29F32Ik7a+QszE63IdvmwdTPDd6cZ5pg4cwA==", + "dev": true + }, "is-stream": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", @@ -3047,17 +3250,21 @@ "dev": true }, "is-string": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz", - "integrity": "sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ==", - "dev": true + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } }, "is-symbol": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", - "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, "requires": { - "has-symbols": "^1.0.1" + "has-symbols": "^1.0.2" } }, "is-typedarray": { @@ -3077,6 +3284,15 @@ "integrity": "sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww==", "dev": true }, + "is-weakref": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.1.tgz", + "integrity": "sha512-b2jKc2pQZjaeFYWEf7ScFj+Be1I+PXmlu572Q8coTXZ+LD/QQZ7ShPMst8h16riVgyXTQwUsFEl74mDvc/3MHQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.0" + } + }, "is-windows": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", @@ -3280,6 +3496,12 @@ "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", "dev": true }, + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "dev": true + }, "json-schema": { "version": "0.2.3", "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", @@ -3392,23 +3614,17 @@ } }, "load-json-file": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", - "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-4.0.0.tgz", + "integrity": "sha1-L19Fq5HjMhYjT9U62rZo607AmTs=", "dev": true, "requires": { "graceful-fs": "^4.1.2", - "parse-json": "^2.2.0", - "pify": "^2.0.0", + "parse-json": "^4.0.0", + "pify": "^3.0.0", "strip-bom": "^3.0.0" }, "dependencies": { - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - }, "strip-bom": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", @@ -3432,6 +3648,18 @@ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", "dev": true }, + "lodash.camelcase": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz", + "integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY=", + "dev": true + }, + "lodash.clonedeep": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", + "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", + "dev": true + }, "lodash.flattendeep": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz", @@ -3444,6 +3672,36 @@ "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", "dev": true }, + "lodash.kebabcase": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz", + "integrity": "sha1-hImxyw0p/4gZXM7KRI/21swpXDY=", + "dev": true + }, + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "lodash.snakecase": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.snakecase/-/lodash.snakecase-4.1.1.tgz", + "integrity": "sha1-OdcUo1NXFHg3rv1ktdy7Fr7Nj40=", + "dev": true + }, + "lodash.truncate": { + "version": "4.4.2", + "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz", + "integrity": "sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM=", + "dev": true + }, + "lodash.upperfirst": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/lodash.upperfirst/-/lodash.upperfirst-4.3.1.tgz", + "integrity": "sha1-E2Xt9DFIBIHvDRxolXpe2Z1J984=", + "dev": true + }, "log-driver": { "version": "1.2.7", "resolved": "https://registry.npmjs.org/log-driver/-/log-driver-1.2.7.tgz", @@ -3609,6 +3867,12 @@ "requires": { "has-flag": "^4.0.0" } + }, + "typescript": { + "version": "3.9.10", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", + "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", + "dev": true } } }, @@ -3667,15 +3931,6 @@ "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", "dev": true }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, "mocha": { "version": "8.3.2", "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.3.2.tgz", @@ -4209,14 +4464,28 @@ "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" }, "object-inspect": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", - "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==" + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.11.0.tgz", + "integrity": "sha512-jp7ikS6Sd3GxQfZJPyH3cjcbJF6GZPClgdV+EFygjFLQ5FmW/dRUnTd9PQ9k0JhoNDabWFbpF1yCdSWCC6gexg==", + "dev": true }, "object-keys": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==" + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true + }, + "object.assign": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + } }, "object.pick": { "version": "1.3.0", @@ -4228,49 +4497,14 @@ } }, "object.values": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.2.tgz", - "integrity": "sha512-MYC0jvJopr8EK6dPBiO8Nb9mvjdypOachO5REGk6MXzujbBrAisKo3HmdEI6kZDL6fC31Mwee/5YbtMebixeag==", + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.5.tgz", + "integrity": "sha512-QUZRW0ilQ3PnPpbNtgdNV1PDbEqLIiSFB3l+EnGtBQ/8SUTLj1PZwtQHABZtLgwpJZTSZhuGLOGk57Drx2IvYg==", "dev": true, "requires": { - "call-bind": "^1.0.0", + "call-bind": "^1.0.2", "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1", - "has": "^1.0.3" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "object.assign": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", - "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", - "dev": true, - "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "has-symbols": "^1.0.1", - "object-keys": "^1.1.1" - } - } + "es-abstract": "^1.19.1" } }, "once": { @@ -4478,12 +4712,13 @@ "dev": true }, "parse-json": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", - "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", "dev": true, "requires": { - "error-ex": "^1.2.0" + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" } }, "parse-ms": { @@ -4562,6 +4797,12 @@ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==" }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, "pkg-dir": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", @@ -4571,6 +4812,51 @@ "find-up": "^4.0.0" } }, + "pkg-up": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-2.0.0.tgz", + "integrity": "sha1-yBmscoBZpGHKscOImivjxJoATX8=", + "dev": true, + "requires": { + "find-up": "^2.1.0" + }, + "dependencies": { + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + } + } + }, "platform": { "version": "1.3.6", "resolved": "https://registry.npmjs.org/platform/-/platform-1.3.6.tgz", @@ -4696,6 +4982,14 @@ "ast-module-types": "^2.7.1", "node-source-walk": "^4.2.0", "typescript": "^3.9.7" + }, + "dependencies": { + "typescript": { + "version": "3.9.10", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", + "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", + "dev": true + } } } } @@ -4707,9 +5001,9 @@ "dev": true }, "prettier": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.2.1.tgz", - "integrity": "sha512-PqyhM2yCjg/oKkFPtTGUojv7gnZAoG80ttl45O6x2Ug/rMJw4wcc9k6aaf2hibP7BGVCCM33gZoGjyvt9mm16Q==", + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.4.1.tgz", + "integrity": "sha512-9fbDAXSBcc6Bs1mZrDYb3XKzDLm4EXXL9sC1LqKP5rZkT6KRr/rf9amVUcODVXgguK/isJz0d0hP72WeaKWsvA==", "dev": true }, "prettier-linter-helpers": { @@ -4825,41 +5119,35 @@ } }, "read-pkg": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz", - "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-3.0.0.tgz", + "integrity": "sha1-nLxoaXj+5l0WwA4rGcI3/Pbjg4k=", "dev": true, "requires": { - "load-json-file": "^2.0.0", + "load-json-file": "^4.0.0", "normalize-package-data": "^2.3.2", - "path-type": "^2.0.0" + "path-type": "^3.0.0" }, "dependencies": { "path-type": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz", - "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", "dev": true, "requires": { - "pify": "^2.0.0" + "pify": "^3.0.0" } - }, - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true } } }, "read-pkg-up": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz", - "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4=", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-3.0.0.tgz", + "integrity": "sha1-PtSWaF26D4/hGNBpHcUfSh/5bwc=", "dev": true, "requires": { "find-up": "^2.0.0", - "read-pkg": "^2.0.0" + "read-pkg": "^3.0.0" }, "dependencies": { "find-up": { @@ -4923,9 +5211,9 @@ "integrity": "sha512-tlbJqcMHnPKI9zSrystikWKwHkBqu2a/Sgw01h3zFjvYrMxEDYHzzoMZnUrbIfpTFEsoRnnviOXNCzFiSc54Qw==" }, "regexpp": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", - "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", + "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", "dev": true }, "release-zalgo": { @@ -4969,6 +5257,12 @@ "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" }, + "require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dev": true + }, "require-main-filename": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", @@ -5161,6 +5455,17 @@ "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true }, + "side-channel": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", + "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "get-intrinsic": "^1.0.2", + "object-inspect": "^1.9.0" + } + }, "sigmund": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/sigmund/-/sigmund-1.0.1.tgz", @@ -5218,20 +5523,38 @@ "dev": true }, "slice-ansi": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", - "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", + "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", "dev": true, "requires": { - "ansi-styles": "^3.2.0", - "astral-regex": "^1.0.0", - "is-fullwidth-code-point": "^2.0.0" + "ansi-styles": "^4.0.0", + "astral-regex": "^2.0.0", + "is-fullwidth-code-point": "^3.0.0" }, "dependencies": { - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true } } @@ -5326,9 +5649,9 @@ } }, "spdx-license-ids": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.7.tgz", - "integrity": "sha512-U+MTEOO0AiDzxwFvoa4JVnMV6mZlJKk2sBLt90s7G0Gd0Mlknc7kxEn3nuDPNZRta7O2uy8oLcZLVT+4sqNZHQ==", + "version": "3.0.10", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.10.tgz", + "integrity": "sha512-oie3/+gKf7QtpitB0LYLETe+k8SifzsX4KixvpOsbI6S0kRiRQ5MKOio8eMSAKQ17N06+wdEOXRiId+zOxo0hA==", "dev": true }, "sprintf-js": { @@ -5364,21 +5687,23 @@ } }, "string.prototype.trimend": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", - "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.4.tgz", + "integrity": "sha512-y9xCjw1P23Awk8EvTpcyL2NIr1j7wJ39f+k6lvRnSMz+mz9CGz9NYPelDk42kOz6+ql8xjfK8oYzy3jAP5QU5A==", + "dev": true, "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" } }, "string.prototype.trimstart": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", - "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.4.tgz", + "integrity": "sha512-jh6e984OBfvxS50tdY2nRZnoC5/mLFKOREQfw8t5yytkoUsJRNxvI/E39qu1sD0OtWI3OC0XgKSmcWwziwYuZw==", + "dev": true, "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" } }, "string_decoder": { @@ -5483,53 +5808,61 @@ } }, "table": { - "version": "5.4.6", - "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", - "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "version": "6.7.2", + "resolved": "https://registry.npmjs.org/table/-/table-6.7.2.tgz", + "integrity": "sha512-UFZK67uvyNivLeQbVtkiUs8Uuuxv24aSL4/Vil2PJVtMgU8Lx0CYkP12uCGa3kjyQzOSgV1+z9Wkb82fCGsO0g==", "dev": true, "requires": { - "ajv": "^6.10.2", - "lodash": "^4.17.14", - "slice-ansi": "^2.1.0", - "string-width": "^3.0.0" + "ajv": "^8.0.1", + "lodash.clonedeep": "^4.5.0", + "lodash.truncate": "^4.4.2", + "slice-ansi": "^4.0.0", + "string-width": "^4.2.3", + "strip-ansi": "^6.0.1" }, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true + "ajv": { + "version": "8.6.3", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.6.3.tgz", + "integrity": "sha512-SMJOdDP6LqTkD0Uq8qLi+gMwSt0imXLSV080qFVwJCpH9U6Mb+SUGHAXM0KNbcBPguytWyvFxcHgMLe2D2XSpw==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + } }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", "dev": true }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", "dev": true }, "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "dev": true, "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" } }, "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "dev": true, "requires": { - "ansi-regex": "^4.1.0" + "ansi-regex": "^5.0.1" } } } @@ -5653,9 +5986,9 @@ } }, "tsconfig-paths": { - "version": "3.9.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.9.0.tgz", - "integrity": "sha512-dRcuzokWhajtZWkQsDVKbWyY+jgcLC5sqJhg2PSgf4ZkH2aHPvaOY8YWGhmjb68b5qqTfasSsDO9k7RUiEmZAw==", + "version": "3.11.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.11.0.tgz", + "integrity": "sha512-7ecdYDnIdmv639mmDwslG6KQg1Z9STTz1j7Gcz0xa+nshh/gKDAHcPxRbWOsA3SPp0tXP2leTcY9Kw+NAkfZzA==", "dev": true, "requires": { "@types/json5": "^0.0.29", @@ -5745,9 +6078,9 @@ } }, "typescript": { - "version": "3.9.7", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.7.tgz", - "integrity": "sha512-BLbiRkiBzAwsjut4x/dsibSTB6yWpwT5qWmC2OfuCg3GgVQCSgMs4vEctYPhsaGtd0AeuuHMkjZ2h2WG8MSzRw==", + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.4.3.tgz", + "integrity": "sha512-4xfscpisVgqqDfPaJo5vkd+Qd/ItkoagnHpufr+i2QCHBsNYp+G7UAoyFl8aPtx879u38wPV65rZ8qbGZijalA==", "dev": true }, "typescript-formatter": { @@ -5760,6 +6093,18 @@ "editorconfig": "^0.15.0" } }, + "unbox-primitive": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.1.tgz", + "integrity": "sha512-tZU/3NqK3dA5gpE1KtyiJUrEB0lxnGkMFHptJ7q6ewdZ8s12QrODwNbhIJStmJkd1QDXa1NRA8aF2A1zk/Ypyw==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has-bigints": "^1.0.1", + "has-symbols": "^1.0.2", + "which-boxed-primitive": "^1.0.2" + } + }, "uniq": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", @@ -5805,9 +6150,9 @@ "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" }, "v8-compile-cache": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.1.tgz", - "integrity": "sha512-8OQ9CL+VWyt3JStj7HX7/ciTL2V3Rl1Wf5OL+SNTm0yK1KvtReVulksyeRnCANHHuUxHlQig+JJDlUhBt1NQDQ==", + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", + "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==", "dev": true }, "validate-npm-package-license": { @@ -5899,6 +6244,19 @@ "isexe": "^2.0.0" } }, + "which-boxed-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", + "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", + "dev": true, + "requires": { + "is-bigint": "^1.0.1", + "is-boolean-object": "^1.1.0", + "is-number-object": "^1.0.4", + "is-string": "^1.0.5", + "is-symbol": "^1.0.3" + } + }, "which-module": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", @@ -5997,15 +6355,6 @@ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, - "write": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", - "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", - "dev": true, - "requires": { - "mkdirp": "^0.5.1" - } - }, "write-file-atomic": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", diff --git a/package.json b/package.json index acbf8eacf..31c3762ea 100644 --- a/package.json +++ b/package.json @@ -82,15 +82,15 @@ "@types/request": "^2.47.0", "@types/sinon": "^9.0.4", "@types/yargs": "^15.0.5", - "@typescript-eslint/eslint-plugin": "^4.4.1", - "@typescript-eslint/parser": "^4.4.1", + "@typescript-eslint/eslint-plugin": "4.32.0", + "@typescript-eslint/parser": "4.32.0", "benchmark": "^2.1.4", "chai": "^4.2.0", "chai-files": "^1.4.0", "coveralls": "^3.0.0", - "eslint": "^7.11.0", - "eslint-plugin-github": "^4.1.1", - "eslint-plugin-no-only-tests": "^2.4.0", + "eslint": "^7.32.0", + "eslint-plugin-github": "4.3.0", + "eslint-plugin-no-only-tests": "2.6.0", "madge": "^4.0.2", "mocha": "^8.3.2", "node-run-cmd": "^1.0.1", @@ -103,7 +103,7 @@ "sync-request": "^6.1.0", "testdouble": "^3.5.2", "ts-node": "8.9.1", - "typescript": "^3.9.2", + "typescript": "^4.4.3", "typescript-formatter": "^7.2.2", "vscode-jsonrpc": "^5.0.1" }, diff --git a/src/BsConfig.ts b/src/BsConfig.ts index d719152b8..53b7e0174 100644 --- a/src/BsConfig.ts +++ b/src/BsConfig.ts @@ -91,7 +91,7 @@ export interface BsConfig { /** * A list of error codes the compiler should NOT emit, even if encountered. */ - ignoreErrorCodes?: (number|string)[]; + ignoreErrorCodes?: (number | string)[]; /** * Emit full paths to files when printing diagnostics to the console. Defaults to false @@ -107,7 +107,7 @@ export interface BsConfig { /** * A list of filters used to exclude diagnostics from the output */ - diagnosticFilters?: Array; + diagnosticFilters?: Array; /** * Specify what diagnostic types should be printed to the console. Defaults to 'warn' diff --git a/src/LanguageServer.ts b/src/LanguageServer.ts index 422c66ae5..1aa2e0b8e 100644 --- a/src/LanguageServer.ts +++ b/src/LanguageServer.ts @@ -272,7 +272,7 @@ export class LanguageServer { await this.waitAllProgramFirstRuns(false); workspaceCreatedDeferred.resolve(); await this.sendDiagnostics(); - } catch (e) { + } catch (e: any) { this.sendCriticalFailure( `Critical failure during BrighterScript language server startup. Please file a github issue and include the contents of the 'BrighterScript Language Server' output channel. @@ -303,7 +303,7 @@ export class LanguageServer { for (let workspace of workspaces) { try { await workspace.firstRunPromise; - } catch (e) { + } catch (e: any) { status = 'critical-error'; //the first run failed...that won't change unless we reload the workspace, so replace with resolved promise //so we don't show this error again @@ -949,7 +949,7 @@ export class LanguageServer { }); // validate all workspaces await this.validateAllThrottled(); - } catch (e) { + } catch (e: any) { this.sendCriticalFailure(`Critical error parsing/ validating ${filePath}: ${e.message}`); } } @@ -967,7 +967,7 @@ export class LanguageServer { ); await this.sendDiagnostics(); - } catch (e) { + } catch (e: any) { this.connection.console.error(e); this.sendCriticalFailure(`Critical error validating workspace: ${e.message}${e.stack ?? ''}`); } @@ -1049,7 +1049,7 @@ export class LanguageServer { activeParameter: activeParameter }; return results; - } catch (e) { + } catch (e: any) { this.connection.console.error(`error in onSignatureHelp: ${e.stack ?? e.message ?? e}`); return { signatures: [], @@ -1170,7 +1170,7 @@ function AddStackToErrorMessage(target: any, propertyKey: string, descriptor: Pr } else { return result; } - } catch (e) { + } catch (e: any) { if (e?.stack) { e.message = e.stack; } diff --git a/src/ProgramBuilder.ts b/src/ProgramBuilder.ts index 41ad59e72..3d7bb2906 100644 --- a/src/ProgramBuilder.ts +++ b/src/ProgramBuilder.ts @@ -101,7 +101,7 @@ export class ProgramBuilder { this.logger.log(`No bsconfig.json file found, using default options`); } this.loadPlugins(); - } catch (e) { + } catch (e: any) { if (e?.file && e.message && e.code) { let err = e as BsDiagnostic; this.staticDiagnostics.push(err); diff --git a/src/astUtils/reflection.ts b/src/astUtils/reflection.ts index 11726fd11..bea4229d0 100644 --- a/src/astUtils/reflection.ts +++ b/src/astUtils/reflection.ts @@ -2,7 +2,7 @@ import type { Body, AssignmentStatement, Block, ExpressionStatement, CommentStat import type { LiteralExpression, Expression, BinaryExpression, CallExpression, FunctionExpression, NamespacedVariableNameExpression, DottedGetExpression, XmlAttributeGetExpression, IndexedGetExpression, GroupingExpression, EscapedCharCodeLiteralExpression, ArrayLiteralExpression, AALiteralExpression, UnaryExpression, VariableExpression, SourceLiteralExpression, NewExpression, CallfuncExpression, TemplateStringQuasiExpression, TemplateStringExpression, TaggedTemplateStringExpression, AnnotationExpression, FunctionParameterExpression } from '../parser/Expression'; import type { BrsFile } from '../files/BrsFile'; import type { XmlFile } from '../files/XmlFile'; -import type { BscFile, File } from '../interfaces'; +import type { BscFile, File, TypedefProvider } from '../interfaces'; import { InvalidType } from '../types/InvalidType'; import { VoidType } from '../types/VoidType'; import { InternalWalkMode } from './visitors'; @@ -218,6 +218,9 @@ export function isFunctionParameterExpression(element: Statement | Expression | export function isAnnotationExpression(element: Statement | Expression | undefined): element is AnnotationExpression { return element?.constructor.name === 'AnnotationExpression'; } +export function isTypedefProvider(element: any): element is TypedefProvider { + return 'getTypedef' in element; +} // BscType reflection export function isStringType(value: any): value is StringType { diff --git a/src/files/BrsFile.ts b/src/files/BrsFile.ts index 6cc9ed503..474810381 100644 --- a/src/files/BrsFile.ts +++ b/src/files/BrsFile.ts @@ -243,7 +243,7 @@ export class BrsFile { this.program.logger.time(LogLevel.debug, ['preprocessor.process', chalk.green(this.pathAbsolute)], () => { preprocessor.process(lexer.tokens, this.program.getManifest()); }); - } catch (error) { + } catch (error: any) { //if the thrown error is DIFFERENT than any errors from the preprocessor, add that error to the list as well if (this.diagnostics.find((x) => x === error) === undefined) { this.diagnostics.push(error); @@ -857,7 +857,7 @@ export class BrsFile { if (classStatement) { let classes = scope.getClassHierarchy(classStatement.item.getName(ParseMode.BrighterScript).toLowerCase()); for (let cs of classes) { - for (let member of [...cs?.item?.fields, ...cs?.item?.methods]) { + for (let member of [...cs?.item?.fields ?? [], ...cs?.item?.methods ?? []]) { if (!results.has(member.name.text.toLowerCase())) { results.set(member.name.text.toLowerCase(), { label: member.name.text, diff --git a/src/lexer/Lexer.ts b/src/lexer/Lexer.ts index ae159303f..e487abf3e 100644 --- a/src/lexer/Lexer.ts +++ b/src/lexer/Lexer.ts @@ -83,7 +83,7 @@ export class Lexer { * @param options options used to customize the scan process * @returns an object containing an array of `errors` and an array of `tokens` to be passed to a parser. */ - public scan(toScan: string, options?: ScanOptions): Lexer { + public scan(toScan: string, options?: ScanOptions): this { this.source = toScan; this.options = this.sanitizeOptions(options); this.start = 0; diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index 54faa418f..356c54524 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -337,7 +337,7 @@ export class Parser { } return this.statement(); - } catch (error) { + } catch (error: any) { //if the error is not a diagnostic, then log the error for debugging purposes if (!error.isDiagnostic) { this.logger.error(error); diff --git a/src/parser/Statement.ts b/src/parser/Statement.ts index 21c1cb503..3c121bca4 100644 --- a/src/parser/Statement.ts +++ b/src/parser/Statement.ts @@ -10,7 +10,7 @@ import type { BrsTranspileState } from './BrsTranspileState'; import { ParseMode, Parser } from './Parser'; import type { WalkVisitor, WalkOptions } from '../astUtils/visitors'; import { InternalWalkMode, walk, createVisitor, WalkMode } from '../astUtils/visitors'; -import { isCallExpression, isClassFieldStatement, isClassMethodStatement, isCommentStatement, isExpression, isExpressionStatement, isFunctionStatement, isIfStatement, isInterfaceFieldStatement, isInterfaceMethodStatement, isInvalidType, isLiteralExpression, isVoidType } from '../astUtils/reflection'; +import { isCallExpression, isClassFieldStatement, isClassMethodStatement, isCommentStatement, isExpression, isExpressionStatement, isFunctionStatement, isIfStatement, isInterfaceFieldStatement, isInterfaceMethodStatement, isInvalidType, isLiteralExpression, isTypedefProvider, isVoidType } from '../astUtils/reflection'; import type { TranspileResult, TypedefProvider } from '../interfaces'; import { createInvalidLiteral, createToken, interpolatedRange } from '../astUtils/creators'; import { DynamicType } from '../types/DynamicType'; @@ -115,10 +115,10 @@ export class Body extends Statement implements TypedefProvider { let result = []; for (const statement of this.statements) { //if the current statement supports generating typedef, call it - if ('getTypedef' in statement) { + if (isTypedefProvider(statement)) { result.push( state.indent(), - ...(statement as TypedefProvider).getTypedef(state), + ...statement.getTypedef(state), state.newline ); } @@ -1585,10 +1585,10 @@ export class ClassStatement extends Statement implements TypedefProvider { result.push(state.newline); state.blockDepth++; for (const member of this.body) { - if ('getTypedef' in member) { + if (isTypedefProvider(member)) { result.push( state.indent(), - ...(member as TypedefProvider).getTypedef(state), + ...member.getTypedef(state), state.newline ); } diff --git a/src/util.ts b/src/util.ts index b13b0d3c5..488d58d63 100644 --- a/src/util.ts +++ b/src/util.ts @@ -1004,7 +1004,7 @@ export class Util { plugin.name = pathOrModule; } acc.push(plugin); - } catch (err) { + } catch (err: any) { if (onError) { onError(pathOrModule, err); } else { From 1b0584332bc5756876e58cf9648191f573328c0c Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 6 Oct 2021 15:51:38 -0400 Subject: [PATCH 056/278] Fix bugs when parsing regex (#458) * Added unit tests for various ways of parsing regex. * Fix regexp lexing bugs * Tweak docs * Prevent regex literal in .brs mode --- docs/regex-literals.md | 2 +- src/diagnosticUtils.ts | 18 ++- src/lexer/Lexer.spec.ts | 34 ++++++ src/lexer/Lexer.ts | 60 ++++++---- src/lexer/TokenKind.ts | 44 +++++++ src/parser/Parser.ts | 1 + .../expression/RegexLiteralExpression.spec.ts | 109 ++++++++++++++++++ src/testHelpers.spec.ts | 7 ++ 8 files changed, 249 insertions(+), 26 deletions(-) diff --git a/docs/regex-literals.md b/docs/regex-literals.md index b0763732c..102c8d7b9 100644 --- a/docs/regex-literals.md +++ b/docs/regex-literals.md @@ -1,5 +1,5 @@ # Regular Expression Literals -You can create a regular expression literal in brighterscript. This simplifies pattern writing and improves readability. +You can create a regular expression literal in brighterscript. This simplifies pattern writing and improves readability. You can use these regular expression literals anywhere you would normally use `CreateObject("roRegex", "the-pattern", "g")`. Example: ```BrighterScript diff --git a/src/diagnosticUtils.ts b/src/diagnosticUtils.ts index b44fefe5a..58e21d257 100644 --- a/src/diagnosticUtils.ts +++ b/src/diagnosticUtils.ts @@ -88,17 +88,25 @@ export function printDiagnostic( //Get the line referenced by the diagnostic. if we couldn't find a line, // default to an empty string so it doesn't crash the error printing below let diagnosticLine = lines[diagnostic.range?.start?.line ?? -1] ?? ''; + console.log( + getDiagnosticLine(diagnostic, diagnosticLine, typeColor[severity]) + ); + console.log(''); +} - let squigglyText = getDiagnosticSquigglyText(diagnostic, diagnosticLine); +export function getDiagnosticLine(diagnostic: BsDiagnostic, diagnosticLine: string, colorFunction: Chalk) { + let result = ''; //only print the line information if we have some if (diagnostic.range && diagnosticLine) { let lineNumberText = chalk.bgWhite(' ' + chalk.black((diagnostic.range.start.line + 1).toString()) + ' ') + ' '; - let blankLineNumberText = chalk.bgWhite(' ' + chalk.bgWhite((diagnostic.range.start.line + 1).toString()) + ' ') + ' '; - console.log(lineNumberText + diagnosticLine); - console.log(blankLineNumberText + typeColor[severity](squigglyText)); + let blankLineNumberText = chalk.bgWhite(' ' + chalk.white('_') + ' ') + ' '; + let squigglyText = getDiagnosticSquigglyText(diagnostic, diagnosticLine); + result += + lineNumberText + diagnosticLine + '\n' + + blankLineNumberText + colorFunction(squigglyText); } - console.log(''); + return result; } /** diff --git a/src/lexer/Lexer.spec.ts b/src/lexer/Lexer.spec.ts index 765c92c49..8572a9518 100644 --- a/src/lexer/Lexer.spec.ts +++ b/src/lexer/Lexer.spec.ts @@ -1248,6 +1248,40 @@ describe('lexer', () => { ); }); + it('does not capture multiple divisions on one line as regex', () => { + const { tokens } = Lexer.scan(`one = 1/2 + 1/4 + 1/4`, { + includeWhitespace: false + }); + expect(tokens.map(x => x.kind)).to.eql([ + TokenKind.Identifier, + TokenKind.Equal, + TokenKind.IntegerLiteral, + TokenKind.Forwardslash, + TokenKind.IntegerLiteral, + TokenKind.Plus, + TokenKind.IntegerLiteral, + TokenKind.Forwardslash, + TokenKind.IntegerLiteral, + TokenKind.Plus, + TokenKind.IntegerLiteral, + TokenKind.Forwardslash, + TokenKind.IntegerLiteral, + TokenKind.Eof + ]); + }); + + it('only captures alphanumeric flags', () => { + expect( + Lexer.scan('speak(/a/)').tokens.map(x => x.kind) + ).to.eql([ + TokenKind.Identifier, + TokenKind.LeftParen, + TokenKind.RegexLiteral, + TokenKind.RightParen, + TokenKind.Eof + ]); + }); + it('handles escape characters properly', () => { testRegex( //an escaped forward slash right next to the end-regexp forwardslash diff --git a/src/lexer/Lexer.ts b/src/lexer/Lexer.ts index e487abf3e..95725d346 100644 --- a/src/lexer/Lexer.ts +++ b/src/lexer/Lexer.ts @@ -5,6 +5,7 @@ import { isAlpha, isDecimalDigit, isAlphaNumeric, isHexDigit } from './Character import type { Range, Diagnostic } from 'vscode-languageserver'; import { DiagnosticMessages } from '../DiagnosticMessages'; import util from '../util'; +import { PreceedingRegexTypes } from '.'; export class Lexer { /** @@ -943,6 +944,18 @@ export class Lexer { } } + /** + * Find the closest previous non-whtespace token + */ + private getPreviousNonWhitespaceToken() { + for (let i = this.tokens.length - 1; i >= 0; i--) { + let token = this.tokens[i]; + if (token && token.kind !== TokenKind.Whitespace) { + return this.tokens[i]; + } + } + } + /** * Capture a regex literal token. Returns false if not found. * This is lookahead lexing which might techincally belong in the parser, @@ -953,29 +966,36 @@ export class Lexer { let nextCharNeedsEscaped = false; - //finite loop to prevent infinite loop if something went wrong - for (let i = this.current; i < this.source.length; i++) { + //regexps can only occur when preceeded by exactly one of these tokens: + const previousKind = this.getPreviousNonWhitespaceToken()?.kind; - //if we reached the end of the regex, consume any flags - if (this.check('/') && !nextCharNeedsEscaped) { - this.advance(); - //consume all flag-like chars (let the parser validate the actual values) - while (/[a-z]/i.exec(this.peek())) { + //preceeded by an allowed token, or if there are no previous tokens (i.e. this is the first token in the file). + if (PreceedingRegexTypes.has(previousKind) || !previousKind) { + + //finite loop to prevent infinite loop if something went wrong + for (let i = this.current; i < this.source.length; i++) { + + //if we reached the end of the regex, consume any flags + if (this.check('/') && !nextCharNeedsEscaped) { this.advance(); - } - //finalize the regex literal and EXIT - this.addToken(TokenKind.RegexLiteral); - return true; + //consume all flag-like chars (let the parser validate the actual values) + while (/[a-z]/i.exec(this.peek())) { + this.advance(); + } + //finalize the regex literal and EXIT + this.addToken(TokenKind.RegexLiteral); + return true; - //if we found a non-escaped newline, there's a syntax error with this regex (or it's not a regex), so quit - } else if (this.check('\n') || this.isAtEnd()) { - break; - } else if (this.check('\\')) { - this.advance(); - nextCharNeedsEscaped = true; - } else { - this.advance(); - nextCharNeedsEscaped = false; + //if we found a non-escaped newline, there's a syntax error with this regex (or it's not a regex), so quit + } else if (this.check('\n') || this.isAtEnd()) { + break; + } else if (this.check('\\')) { + this.advance(); + nextCharNeedsEscaped = true; + } else { + this.advance(); + nextCharNeedsEscaped = false; + } } } this.popLookahead(); diff --git a/src/lexer/TokenKind.ts b/src/lexer/TokenKind.ts index 70ed8d7df..6c58c416c 100644 --- a/src/lexer/TokenKind.ts +++ b/src/lexer/TokenKind.ts @@ -598,3 +598,47 @@ export const DeclarableTypes = [ TokenKind.Void, TokenKind.Function ]; + +/** + * The tokens that might preceed a regex literal + */ +export const PreceedingRegexTypes = new Set([ + TokenKind.Print, + TokenKind.Question, + TokenKind.QuestionQuestion, + TokenKind.LeftSquareBracket, + TokenKind.LeftParen, + TokenKind.LeftCurlyBrace, + TokenKind.Caret, + TokenKind.Minus, + TokenKind.Plus, + TokenKind.Star, + TokenKind.Forwardslash, + TokenKind.Mod, + TokenKind.Backslash, + TokenKind.LeftShift, + TokenKind.RightShift, + TokenKind.MinusEqual, + TokenKind.PlusEqual, + TokenKind.StarEqual, + TokenKind.ForwardslashEqual, + TokenKind.BackslashEqual, + TokenKind.LeftShiftEqual, + TokenKind.RightShiftEqual, + TokenKind.Less, + TokenKind.LessEqual, + TokenKind.Greater, + TokenKind.GreaterEqual, + TokenKind.Equal, + TokenKind.LessGreater, + TokenKind.And, + TokenKind.Or, + TokenKind.If, + TokenKind.Not, + TokenKind.To, + TokenKind.Newline, + TokenKind.Throw, + TokenKind.Throw, + TokenKind.Colon, + TokenKind.Semicolon +]); diff --git a/src/parser/Parser.ts b/src/parser/Parser.ts index 356c54524..7af59b49d 100644 --- a/src/parser/Parser.ts +++ b/src/parser/Parser.ts @@ -1398,6 +1398,7 @@ export class Parser { } private regexLiteralExpression() { + this.warnIfNotBrighterScriptMode('regular expression literal'); return new RegexLiteralExpression({ regexLiteral: this.advance() }); diff --git a/src/parser/tests/expression/RegexLiteralExpression.spec.ts b/src/parser/tests/expression/RegexLiteralExpression.spec.ts index be5542ad8..ca28e1b21 100644 --- a/src/parser/tests/expression/RegexLiteralExpression.spec.ts +++ b/src/parser/tests/expression/RegexLiteralExpression.spec.ts @@ -1,6 +1,8 @@ import { Program } from '../../../Program'; import { standardizePath as s } from '../../../util'; import { getTestTranspile } from '../../../testHelpers.spec'; +import { expect } from 'chai'; +import { DiagnosticMessages } from '../../../DiagnosticMessages'; describe('RegexLiteralExpression', () => { let rootDir = s`${process.cwd()}/rootDir`; @@ -63,5 +65,112 @@ describe('RegexLiteralExpression', () => { `); }); + it('warns when in non-brighterscript mode', () => { + program.addOrReplaceFile('source/main.brs', ` + sub main() + print /"/ + end sub + `); + expect(program.getDiagnostics()[0].code).to.eql(DiagnosticMessages.bsFeatureNotSupportedInBrsFiles('').code); + }); + + it('handles edge cases', () => { + testTranspile(` + sub main() + print /1/ + ? /1/ + thing = false ?? /1/ + v = /1/.Match("1")[0].ToInt() + v = [ + 0 + ] + v[/0/.Match("0")[0].ToInt()] = true + type(/1/.Match("1")[0].ToInt()) + v = 1 ^ /1/.Match("1")[0].ToInt() + v = 1 - /1/.Match("1")[0].ToInt() + v = 1 + /1/.Match("1")[0].ToInt() + v = 1 * /1/.Match("1")[0].ToInt() + v = 1 / /1/.Match("1")[0].ToInt() + v = 1 mod /1/.Match("1")[0].ToInt() + v = 1 \\ /1/.Match("1")[0].ToInt() + v = 1 >> /1/.Match("1")[0].ToInt() + v = 1 << /1/.Match("1")[0].ToInt() + v -= /1/.Match("1")[0].ToInt() + v += /1/.Match("1")[0].ToInt() + v *= /1/.Match("1")[0].ToInt() + v \\= /1/.Match("1")[0].ToInt() + v /= /1/.Match("1")[0].ToInt() + v <<= /1/.Match("1")[0].ToInt() + v >>= /1/.Match("1")[0].ToInt() + v = 1 < /1/.Match("1")[0].ToInt() + v = 1 <= /1/.Match("1")[0].ToInt() + v = 1 > /1/.Match("1")[0].ToInt() + v = 1 >= /1/.Match("1")[0].ToInt() + v = 1 = /1/.Match("1")[0].ToInt() + v = 1 <> /1/.Match("1")[0].ToInt() + v = 1 and /1/.Match("1")[0].ToInt() + v = 1 or /1/.Match("1")[0].ToInt() + if /1/.Match("1")[0].ToInt() > 0 then + end if + v = not /1/.Match("1")[0].ToInt() > 0 + for i = 0 to /1/.Match("1")[0].ToInt() + print "for!" + end for + v = /1/ + v = { name: /1/.Match("1")[0].ToInt() } + print 1; /1/.Match("1")[0].ToInt() + throw /1/.Match("1")[0] + end sub + `, ` + sub main() + print CreateObject("roRegex", "1", "") + ? CreateObject("roRegex", "1", "") + thing = bslib_coalesce(false, CreateObject("roRegex", "1", "")) + v = CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = [ + 0 + ] + v[CreateObject("roRegex", "0", "").Match("0")[0].ToInt()] = true + type(CreateObject("roRegex", "1", "").Match("1")[0].ToInt()) + v = 1 ^ CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 - CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 + CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 * CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 / CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 mod CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 \\ CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 >> CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 << CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v -= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v += CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v *= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v \\= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v /= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v <<= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v >>= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 < CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 <= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 > CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 >= CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 = CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 <> CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 and CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + v = 1 or CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + if CreateObject("roRegex", "1", "").Match("1")[0].ToInt() > 0 then + end if + v = not CreateObject("roRegex", "1", "").Match("1")[0].ToInt() > 0 + for i = 0 to CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + print "for!" + end for + v = CreateObject("roRegex", "1", "") + v = { + name: CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + } + print 1; CreateObject("roRegex", "1", "").Match("1")[0].ToInt() + throw CreateObject("roRegex", "1", "").Match("1")[0] + end sub + `); + }); + }); }); diff --git a/src/testHelpers.spec.ts b/src/testHelpers.spec.ts index df3aefa28..9a9723337 100644 --- a/src/testHelpers.spec.ts +++ b/src/testHelpers.spec.ts @@ -1,5 +1,6 @@ import type { BscFile, BsDiagnostic } from './interfaces'; import * as assert from 'assert'; +import chalk from 'chalk'; import type { Diagnostic } from 'vscode-languageserver'; import { createSandbox } from 'sinon'; import { expect } from 'chai'; @@ -10,6 +11,7 @@ import { BrsFile } from './files/BrsFile'; import type { Program } from './Program'; import { standardizePath as s } from './util'; import type { CodeWithSourceMap } from 'source-map'; +import { getDiagnosticLine } from './diagnosticUtils'; /** * Trim leading whitespace for every line (to make test writing cleaner @@ -79,6 +81,11 @@ export function expectZeroDiagnostics(arg: { getDiagnostics(): Array //escape any newlines diagnostic.message = diagnostic.message.replace(/\r/g, '\\r').replace(/\n/g, '\\n'); message += `\n • bs${diagnostic.code} "${diagnostic.message}" at ${diagnostic.file?.pathAbsolute ?? ''}#(${diagnostic.range.start.line}:${diagnostic.range.start.character})-(${diagnostic.range.end.line}:${diagnostic.range.end.character})`; + //print the line containing the error (if we can find it) + const line = diagnostic.file?.fileContents?.split(/\r?\n/g)?.[diagnostic.range.start.line]; + if (line) { + message += '\n' + getDiagnosticLine(diagnostic, line, chalk.red); + } } assert.fail(message); } From 909f9b6239faf09975ae0ebcc02a675ca1867314 Mon Sep 17 00:00:00 2001 From: Bronley Date: Thu, 7 Oct 2021 07:27:42 -0400 Subject: [PATCH 057/278] update changelog for v0.41.1 --- CHANGELOG.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index c263e6939..862f60440 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.41.1](https://github.com/rokucommunity/brighterscript/compare/v0.41.0...v0.41.1) - 2021-10-07 +### Fixed + - parse issue with regex literals ([#458](https://github.com/rokucommunity/brighterscript/pull/458)). + + + ## [0.41.0](https://github.com/rokucommunity/brighterscript/compare/v0.40.1...v0.41.0) - 2021-09-27 ### Added - Regex literal support in brighterscript ([#452](https://github.com/rokucommunity/brighterscript/pull/452)). From 538db24ab172a53a185b17683b0498d57e02c90a Mon Sep 17 00:00:00 2001 From: Bronley Date: Thu, 7 Oct 2021 07:29:12 -0400 Subject: [PATCH 058/278] 0.41.1 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 06900d891..208e27522 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.0", + "version": "0.41.1", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 31c3762ea..a57f73e34 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.0", + "version": "0.41.1", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From 4bdc99b3df344fd235099f65faa7f0990ab3b385 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Sun, 17 Oct 2021 07:47:51 -0400 Subject: [PATCH 059/278] Fix crash with mismatched class members (#461) --- src/files/BrsFile.Class.spec.ts | 26 ++++++++++++++++++++++++++ src/parser/Expression.ts | 10 ++++++++++ src/validators/ClassValidator.ts | 14 ++++++++++---- 3 files changed, 46 insertions(+), 4 deletions(-) diff --git a/src/files/BrsFile.Class.spec.ts b/src/files/BrsFile.Class.spec.ts index 20b1b1fab..7aa0b70f4 100644 --- a/src/files/BrsFile.Class.spec.ts +++ b/src/files/BrsFile.Class.spec.ts @@ -1178,4 +1178,30 @@ describe('BrsFile BrighterScript classes', () => { file.parser.references.classStatements[0].getParentClassIndex(new BrsTranspileState(file)); }); }); + + it('does not crash when child has field with same name as sub in parent', () => { + program.addOrReplaceFile('source/main.bs', ` + class Parent + public function helloWorld() + end function + end class + class Child extends Parent + public helloWorld as string + end class + `); + program.validate(); + }); + + it('does not crash when child has method with same name as field in parent', () => { + program.addOrReplaceFile('source/main.bs', ` + class Parent + public helloWorld as string + end class + class Child extends Parent + public function helloWorld() + end function + end class + `); + program.validate(); + }); }); diff --git a/src/parser/Expression.ts b/src/parser/Expression.ts index 279263be5..d7c663a47 100644 --- a/src/parser/Expression.ts +++ b/src/parser/Expression.ts @@ -14,6 +14,7 @@ import type { TranspileResult, TypedefProvider } from '../interfaces'; import { VoidType } from '../types/VoidType'; import { DynamicType } from '../types/DynamicType'; import type { BscType } from '../types/BscType'; +import { FunctionType } from '../types/FunctionType'; export type ExpressionVisitor = (expression: Expression, parent: Expression) => void; @@ -250,6 +251,15 @@ export class FunctionExpression extends Expression implements TypedefProvider { } } } + + getFunctionType(): FunctionType { + let functionType = new FunctionType(this.returnType); + functionType.isSub = this.functionType.text === 'sub'; + for (let param of this.parameters) { + functionType.addParameter(param.name.text, param.type, !!param.typeToken); + } + return functionType; + } } export class FunctionParameterExpression extends Expression { diff --git a/src/validators/ClassValidator.ts b/src/validators/ClassValidator.ts index a23c970b5..131488c12 100644 --- a/src/validators/ClassValidator.ts +++ b/src/validators/ClassValidator.ts @@ -2,7 +2,7 @@ import type { Scope } from '../Scope'; import { DiagnosticMessages } from '../DiagnosticMessages'; import type { CallExpression } from '../parser/Expression'; import { ParseMode } from '../parser/Parser'; -import type { ClassFieldStatement, ClassMethodStatement, ClassStatement } from '../parser/Statement'; +import type { ClassMethodStatement, ClassStatement } from '../parser/Statement'; import { CancellationTokenSource, Location } from 'vscode-languageserver'; import { URI } from 'vscode-uri'; import util from '../util'; @@ -11,6 +11,7 @@ import type { BscFile, BsDiagnostic } from '../interfaces'; import { createVisitor, WalkMode } from '../astUtils'; import type { BrsFile } from '../files/BrsFile'; import { TokenKind } from '../lexer'; +import { DynamicType } from '../types/DynamicType'; export class BsClassValidator { private scope: Scope; @@ -197,9 +198,14 @@ export class BsClassValidator { //child field has same name as parent if (isClassFieldStatement(member)) { - const ancestorFieldType = (ancestorAndMember.member as ClassFieldStatement).getType(); + let ancestorMemberType = new DynamicType(); + if (isClassFieldStatement(ancestorAndMember.member)) { + ancestorMemberType = ancestorAndMember.member.getType(); + } else if (isClassMethodStatement(ancestorAndMember.member)) { + ancestorMemberType = ancestorAndMember.member.func.getFunctionType(); + } const childFieldType = member.getType(); - if (!childFieldType.isAssignableTo(ancestorFieldType)) { + if (!childFieldType.isAssignableTo(ancestorMemberType)) { //flag incompatible child field type to ancestor field type this.diagnostics.push({ ...DiagnosticMessages.childFieldTypeNotAssignableToBaseProperty( @@ -207,7 +213,7 @@ export class BsClassValidator { ancestorAndMember.classStatement.getName(ParseMode.BrighterScript), member.name.text, childFieldType.toString(), - ancestorFieldType.toString() + ancestorMemberType.toString() ), file: classStatement.file, range: member.range From 80903bce1e68e9c41e23e2e304e477233dc1d532 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Sun, 17 Oct 2021 07:49:26 -0400 Subject: [PATCH 060/278] update changelog for v0.41.2 --- CHANGELOG.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 862f60440..02c9ac537 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.41.2](https://github.com/rokucommunity/brighterscript/compare/v0.41.1...v0.41.2) - 2021-10-17 +### Fixed + - crash when subclass field has same name as ancestor method ([#460](https://github.com/rokucommunity/brighterscript/pull/460)). + + + ## [0.41.1](https://github.com/rokucommunity/brighterscript/compare/v0.41.0...v0.41.1) - 2021-10-07 ### Fixed - parse issue with regex literals ([#458](https://github.com/rokucommunity/brighterscript/pull/458)). From 6689d9339a48539449be929b575cc705ebfdd637 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Sun, 17 Oct 2021 07:50:13 -0400 Subject: [PATCH 061/278] 0.41.2 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 208e27522..83d1a3dcb 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.1", + "version": "0.41.2", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index a57f73e34..68f709b53 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.1", + "version": "0.41.2", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From bde1741d2a59800a8700eff8e910e99bffe5af57 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Tue, 26 Oct 2021 16:08:27 -0400 Subject: [PATCH 062/278] Allow diagnostic non-numeric disable code comments (#463) --- src/CommentFlagProcessor.spec.ts | 13 +++++++++++++ src/CommentFlagProcessor.ts | 11 +++++++---- src/files/BrsFile.spec.ts | 22 ++++++++++++++++++++-- src/util.ts | 3 ++- 4 files changed, 42 insertions(+), 7 deletions(-) diff --git a/src/CommentFlagProcessor.spec.ts b/src/CommentFlagProcessor.spec.ts index b3d539fc4..39c7dea93 100644 --- a/src/CommentFlagProcessor.spec.ts +++ b/src/CommentFlagProcessor.spec.ts @@ -153,4 +153,17 @@ describe('CommentFlagProcessor', () => { }); }); }); + + describe('tryAdd', () => { + it('supports non-numeric codes', () => { + processor.tryAdd(`'bs:disable-line lint123 LINT2 some-textual-diagnostic`, Range.create(0, 0, 0, 54)); + expect( + processor.commentFlags.flatMap(x => x.codes) + ).to.eql([ + 'lint123', + 'lint2', + 'some-textual-diagnostic' + ]); + }); + }); }); diff --git a/src/CommentFlagProcessor.ts b/src/CommentFlagProcessor.ts index 67b4e658c..52ea412d0 100644 --- a/src/CommentFlagProcessor.ts +++ b/src/CommentFlagProcessor.ts @@ -69,15 +69,18 @@ export class CommentFlagProcessor { //disable specific diagnostic codes } else { - let codes = [] as number[]; + let codes = [] as DiagnosticCode[]; for (let codeToken of tokenized.codes) { let codeInt = parseInt(codeToken.code); + //is a plugin-contributed or non-numeric code if (isNaN(codeInt)) { - //don't validate non-numeric codes - continue; - //add a warning for unknown codes + codes.push(codeToken.code?.toString()?.toLowerCase()); + + //validate numeric codes against the list of known bsc codes } else if (this.diagnosticCodes.includes(codeInt)) { codes.push(codeInt); + + //add a warning for unknown codes } else { this.diagnostics.push({ ...DiagnosticMessages.unknownDiagnosticCode(codeInt), diff --git a/src/files/BrsFile.spec.ts b/src/files/BrsFile.spec.ts index e033a3f83..319240ab3 100644 --- a/src/files/BrsFile.spec.ts +++ b/src/files/BrsFile.spec.ts @@ -357,17 +357,35 @@ describe('BrsFile', () => { expect(program.getDiagnostics()[0]?.message).to.not.exist; }); - it('ignores non-numeric codes', () => { + it('recognizes non-numeric codes', () => { let file = program.addOrReplaceFile({ src: `${rootDir}/source/main.brs`, dest: 'source/main.brs' }, ` sub Main() 'bs:disable-next-line: LINT9999 name = "bob end sub `); - expect(file.commentFlags[0]).to.not.exist; + expect(file.commentFlags[0]).to.exist; expect(program.getDiagnostics()[0]?.message).to.exist; }); + it('supports disabling non-numeric error codes', () => { + const program = new Program({}); + const file = program.addOrReplaceFile('source/main.brs', ` + sub main() + something = true 'bs:disable-line: LINT1005 + end sub + `); + file.addDiagnostics([{ + code: 'LINT1005', + file: file, + message: 'Something is not right', + range: util.createRange(2, 16, 2, 26) + }]); + expect( + program.getScopesForFile(file)[0].getDiagnostics() + ).to.be.empty; + }); + it('adds diagnostics for unknown numeric diagnostic codes', () => { program.addOrReplaceFile({ src: `${rootDir}/source/main.brs`, dest: 'source/main.brs' }, ` sub main() diff --git a/src/util.ts b/src/util.ts index 488d58d63..8f9d30c42 100644 --- a/src/util.ts +++ b/src/util.ts @@ -642,11 +642,12 @@ export class Util { * @param diagnostic */ public diagnosticIsSuppressed(diagnostic: BsDiagnostic) { + const diagnosticCode = typeof diagnostic.code === 'string' ? diagnostic.code.toLowerCase() : diagnostic.code; for (let flag of diagnostic.file?.commentFlags ?? []) { //this diagnostic is affected by this flag if (this.rangeContains(flag.affectedRange, diagnostic.range.start)) { //if the flag acts upon this diagnostic's code - if (flag.codes === null || flag.codes.includes(diagnostic.code as number)) { + if (flag.codes === null || flag.codes.includes(diagnosticCode)) { return true; } } From 6df1e9eda8f08c1a0f9258803420d3f28d0bc1a5 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 27 Oct 2021 12:44:09 -0400 Subject: [PATCH 063/278] roku-deploy@3.5.0 --- package-lock.json | 247 +++++++++++++--------------------------------- package.json | 4 +- 2 files changed, 68 insertions(+), 183 deletions(-) diff --git a/package-lock.json b/package-lock.json index 83d1a3dcb..9ff1b4f52 100644 --- a/package-lock.json +++ b/package-lock.json @@ -972,9 +972,9 @@ "dev": true }, "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, "ansi-styles": { "version": "3.2.1", @@ -3932,45 +3932,45 @@ "dev": true }, "mocha": { - "version": "8.3.2", - "resolved": "https://registry.npmjs.org/mocha/-/mocha-8.3.2.tgz", - "integrity": "sha512-UdmISwr/5w+uXLPKspgoV7/RXZwKRTiTjJ2/AC5ZiEztIoOYdfKb19+9jNmEInzx5pBsCyJQzarAxqIGBNYJhg==", + "version": "9.1.3", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-9.1.3.tgz", + "integrity": "sha512-Xcpl9FqXOAYqI3j79pEtHBBnQgVXIhpULjGQa7DVb0Po+VzmSIK9kanAiWLHoRR/dbZ2qpdPshuXr8l1VaHCzw==", "dev": true, "requires": { "@ungap/promise-all-settled": "1.1.2", "ansi-colors": "4.1.1", "browser-stdout": "1.3.1", - "chokidar": "3.5.1", - "debug": "4.3.1", + "chokidar": "3.5.2", + "debug": "4.3.2", "diff": "5.0.0", "escape-string-regexp": "4.0.0", "find-up": "5.0.0", - "glob": "7.1.6", + "glob": "7.1.7", "growl": "1.10.5", "he": "1.2.0", - "js-yaml": "4.0.0", - "log-symbols": "4.0.0", + "js-yaml": "4.1.0", + "log-symbols": "4.1.0", "minimatch": "3.0.4", "ms": "2.1.3", - "nanoid": "3.1.20", - "serialize-javascript": "5.0.1", + "nanoid": "3.1.25", + "serialize-javascript": "6.0.0", "strip-json-comments": "3.1.1", "supports-color": "8.1.1", "which": "2.0.2", - "wide-align": "1.1.3", - "workerpool": "6.1.0", + "workerpool": "6.1.5", "yargs": "16.2.0", "yargs-parser": "20.2.4", "yargs-unparser": "2.0.0" }, "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", "dev": true, "requires": { - "color-convert": "^2.0.1" + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" } }, "argparse": { @@ -3979,73 +3979,26 @@ "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", "dev": true }, - "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", - "dev": true, - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "dependencies": { - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } - } - }, "chokidar": { - "version": "3.5.1", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.1.tgz", - "integrity": "sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw==", + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", + "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", "dev": true, "requires": { - "anymatch": "~3.1.1", + "anymatch": "~3.1.2", "braces": "~3.0.2", - "fsevents": "~2.3.1", - "glob-parent": "~5.1.0", + "fsevents": "~2.3.2", + "glob-parent": "~5.1.2", "is-binary-path": "~2.1.0", "is-glob": "~4.0.1", "normalize-path": "~3.0.0", - "readdirp": "~3.5.0" - } - }, - "cliui": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", - "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", - "dev": true, - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^7.0.0" + "readdirp": "~3.6.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "requires": { - "color-name": "~1.1.4" - } - }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", "dev": true, "requires": { "ms": "2.1.2" @@ -4075,12 +4028,19 @@ "path-exists": "^4.0.0" } }, - "fsevents": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", - "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "glob": { + "version": "7.1.7", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", "dev": true, - "optional": true + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } }, "has-flag": { "version": "4.0.0", @@ -4089,9 +4049,9 @@ "dev": true }, "js-yaml": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.0.0.tgz", - "integrity": "sha512-pqon0s+4ScYUvX30wxQi3PogGFAlUyH0awepWvwkj4jD4v+ova3RiYw8bmA6x2rDrEaj8i/oWKoRxpVNW+Re8Q==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "dev": true, "requires": { "argparse": "^2.0.1" @@ -4106,15 +4066,6 @@ "p-locate": "^5.0.0" } }, - "log-symbols": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.0.0.tgz", - "integrity": "sha512-FN8JBzLx6CzeMrB0tg6pqlGU1wCrXW+ZXGH481kfsBqer0hToTIiHdjH4Mq8xJUbvATujKCvaREGWpGUionraA==", - "dev": true, - "requires": { - "chalk": "^4.0.0" - } - }, "ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", @@ -4139,6 +4090,15 @@ "p-limit": "^3.0.2" } }, + "readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "requires": { + "picomatch": "^2.2.1" + } + }, "supports-color": { "version": "8.1.1", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", @@ -4148,38 +4108,6 @@ "has-flag": "^4.0.0" } }, - "wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "dev": true, - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "y18n": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.5.tgz", - "integrity": "sha512-hsRUr4FFrvhhRH12wOdfs38Gy7k2FFzB9qgN9v3aLykRq0dRcdcpz5C9FxdS2NuhOrI/628b/KSTJ3rwHysYSg==", - "dev": true - }, - "yargs": { - "version": "16.2.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", - "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", - "dev": true, - "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" - } - }, "yargs-parser": { "version": "20.2.4", "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", @@ -4231,9 +4159,9 @@ "dev": true }, "nanoid": { - "version": "3.1.20", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.20.tgz", - "integrity": "sha512-a1cQNyczgKbLX9jwbS/+d7W8fX/RfgYR7lVWwWOGIPNgK2m0MWvrGF6/m4kk6U3QcFMnZf3RIhL0v2Jgh/0Uxw==", + "version": "3.1.25", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.25.tgz", + "integrity": "sha512-rdwtIXaXCLFAQbnfqDRnI6jaRHp9fTcYBjtFKE8eezcZ7LuLjhUaQGNeMXf1HmRoCH32CLz6XwX0TtxEOS/A3Q==", "dev": true }, "natural-compare": { @@ -5332,13 +5260,12 @@ } }, "roku-deploy": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.4.2.tgz", - "integrity": "sha512-Tg/K5M9/gvH5bzzs3g2bUxrcPiwKigKdnpbq5yiTjMnAtDJM6u7Ug4PbcnnoH3qJ3IJqODCW7Kq1iIH7sWmqBg==", + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/roku-deploy/-/roku-deploy-3.5.0.tgz", + "integrity": "sha512-F8Y+0g+xkXr54I0i/GKwLrUxXD3FgSPbxcQmGOQ0oSM7w3lY0BacdncDtuL3FUw8VkRWa1jiHtGHc77oSy5jaw==", "requires": { "chalk": "^2.4.2", "dateformat": "^3.0.3", - "eventemitter3": "^4.0.7", "fs-extra": "^7.0.1", "glob": "^7.1.6", "jsonc-parser": "^2.3.0", @@ -5421,9 +5348,9 @@ } }, "serialize-javascript": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz", - "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==", + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz", + "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==", "dev": true, "requires": { "randombytes": "^2.1.0" @@ -6263,48 +6190,6 @@ "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", "dev": true }, - "wide-align": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", - "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", - "dev": true, - "requires": { - "string-width": "^1.0.2 || 2" - }, - "dependencies": { - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - }, - "string-width": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", - "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", - "dev": true, - "requires": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" - } - }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", - "dev": true, - "requires": { - "ansi-regex": "^3.0.0" - } - } - } - }, "word-wrap": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", @@ -6312,9 +6197,9 @@ "dev": true }, "workerpool": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.1.0.tgz", - "integrity": "sha512-toV7q9rWNYha963Pl/qyeZ6wG+3nnsyvolaNUS8+R5Wtw6qJPTxIlOP1ZSvcGhEJw+l3HMMmtiNo9Gl61G4GVg==", + "version": "6.1.5", + "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.1.5.tgz", + "integrity": "sha512-XdKkCK0Zqc6w3iTxLckiuJ81tiD/o5rBE/m+nXpRCB+/Sq4DqkfXZ/x0jW02DG1tGsfUGXbTJyZDP+eu67haSw==", "dev": true }, "wrap-ansi": { diff --git a/package.json b/package.json index 68f709b53..64d7a90df 100644 --- a/package.json +++ b/package.json @@ -92,7 +92,7 @@ "eslint-plugin-github": "4.3.0", "eslint-plugin-no-only-tests": "2.6.0", "madge": "^4.0.2", - "mocha": "^8.3.2", + "mocha": "^9.1.3", "node-run-cmd": "^1.0.1", "nyc": "^15.1.0", "object.pick": "^1.3.0", @@ -128,7 +128,7 @@ "moment": "^2.23.0", "p-settle": "^2.1.0", "parse-ms": "^2.1.0", - "roku-deploy": "^3.4.2", + "roku-deploy": "^3.5.0", "serialize-error": "^7.0.1", "source-map": "^0.7.3", "vscode-languageserver": "7.0.0", From 424239f727fc16e5b547540415649ca76f45ec14 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 27 Oct 2021 12:46:06 -0400 Subject: [PATCH 064/278] update changelog for v0.41.3 --- CHANGELOG.md | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 02c9ac537..823a55009 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.41.3](https://github.com/rokucommunity/brighterscript/compare/v0.41.2...v0.41.3) - 2021-10-27 +### Changed + - upgrade to [roku-deploy@3.5.0](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#350---2021-10-27) which adds the ability to use negated non-rootDir top-level patterns in the files array ([#78](https://github.com/rokucommunity/roku-deploy/pull/78)) +### Fixed + - Allow diagnostic non-numeric disable code comments ([#463](https://github.com/rokucommunity/brighterscript/pull/463)). + + ## [0.41.2](https://github.com/rokucommunity/brighterscript/compare/v0.41.1...v0.41.2) - 2021-10-17 ### Fixed - crash when subclass field has same name as ancestor method ([#460](https://github.com/rokucommunity/brighterscript/pull/460)). From 47ef7b5e041bbfa0046dda35085b61ebbfb47800 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 27 Oct 2021 12:46:56 -0400 Subject: [PATCH 065/278] 0.41.3 --- package-lock.json | 2 +- package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 9ff1b4f52..5879c2208 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.2", + "version": "0.41.3", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 64d7a90df..cf74e67fe 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.41.2", + "version": "0.41.3", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From d1554c7f887c1a0ef69a6342192faf3cadb5ac12 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 27 Oct 2021 14:12:46 -0400 Subject: [PATCH 066/278] inline all changelog links --- CHANGELOG.md | 406 ++++++++++++++++++--------------------------------- 1 file changed, 139 insertions(+), 267 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 823a55009..4a943c75f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -37,8 +37,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.40.0] - 2021-08-02 -[0.40.0]: https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0 +## [0.40.0](https://github.com/rokucommunity/brighterscript/compare/v0.39.4...v0.40.0) - 2021-08-02 ### Added - language support for Interface statements ([#426](https://github.com/rokucommunity/brighterscript/pull/426)) ### Changed @@ -46,8 +45,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.39.4] - 2021-06-27 -[0.39.4]: https://github.com/rokucommunity/brighterscript/compare/v0.39.3...v0.39.4 +## [0.39.4](https://github.com/rokucommunity/brighterscript/compare/v0.39.3...v0.39.4) - 2021-06-27 ### Fixed - incorrect block range for inline if/then branch ([#424](https://github.com/rokucommunity/brighterscript/pull/424)) - extract associative array comma when parsing ([#427](https://github.com/rokucommunity/brighterscript/pull/424)) @@ -55,22 +53,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - add v30/bslCore library functions to global callables ([#433](https://github.com/rokucommunity/brighterscript/pull/433)) -## [0.39.3] - 2021-06-01 -[0.39.3]: https://github.com/rokucommunity/brighterscript/compare/v0.39.2...v0.39.3 +## [0.39.3](https://github.com/rokucommunity/brighterscript/compare/v0.39.2...v0.39.3) - 2021-06-01 ### Changed - upgraded to [roku-deploy@3.4.1](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#341---2021-06-01) which fixes bugs introduced in roku-deploy@3.4.0 -## [0.39.2] - 2021-05-28 -[0.39.2]: https://github.com/rokucommunity/brighterscript/compare/v0.39.1...v0.39.2 +## [0.39.2](https://github.com/rokucommunity/brighterscript/compare/v0.39.1...v0.39.2) - 2021-05-28 ### Changed - upgraded to [roku-deploy@3.4.0](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#340---2021-05-28) which brings significant zip speed improvements -## [0.39.1] - 2021-05-24 -[0.39.1]: https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1 +## [0.39.1](https://github.com/rokucommunity/brighterscript/compare/v0.39.0...v0.39.1) - 2021-05-24 ### Changed - re-export `CodeActionKind` so plugins don't need to import from vscode-brightscript-language directly. ### Fixed @@ -79,23 +74,20 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.39.0] - 2021-05-18 -[0.39.0]: https://github.com/rokucommunity/brighterscript/compare/v0.38.2...v0.39.0 +## [0.39.0](https://github.com/rokucommunity/brighterscript/compare/v0.38.2...v0.39.0) - 2021-05-18 ### Added - semantic token support for plugins - basic semantic tokens for `new` statements and namespace usage -## [0.38.2] - 2021-05-17 -[0.38.2]: https://github.com/rokucommunity/brighterscript/compare/v0.38.1...v0.38.2 +## [0.38.2](https://github.com/rokucommunity/brighterscript/compare/v0.38.1...v0.38.2) - 2021-05-17 ### Fixed - language server crash when namespaced function or class didn't have a name ([#419](https://github.com/rokucommunity/brighterscript/pull/419)) -## [0.38.1] - 2021-05-14 -[0.38.1]: https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.38.1 +## [0.38.1](https://github.com/rokucommunity/brighterscript/compare/v0.38.0...v0.38.1) - 2021-05-14 ### Changed - SOURCE_FILE_PATH and SOURCE_LOCATION source literals are now string concatenations in order to avoid triggering Roku's static analysis rule against `file:/` protocol strings ([#415](https://github.com/rokucommunity/brighterscript/pull/415)) ### Fixed @@ -106,46 +98,46 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.38.0] - 2021-05-04 +## [0.38.0](https://github.com/rokucommunity/brighterscript/compare/v0.37.4...v0.38.0) - 2021-05-04 ### Added - warning for mismatched class method accessibility ([#402](https://github.com/rokucommunity/brighterscript/pull/402)) - allow class field overrides in child classes as long as they are the same type ([#394](https://github.com/rokucommunity/brighterscript/pull/394)) -## [0.37.4] - 2021-04-20 +## [0.37.4](https://github.com/rokucommunity/brighterscript/compare/v0.37.3...v0.37.4) - 2021-04-20 ### Fixed - bug validating namespace function calls ([#390](https://github.com/rokucommunity/brighterscript/pull/390)) -## [0.37.3] - 2021-04-12 +## [0.37.3](https://github.com/rokucommunity/brighterscript/compare/v0.37.2...v0.37.3) - 2021-04-12 ### Fixed - bug where having multiple components with the same name would cause issues in the program, normally requiring a language server or watcher restart. ([#353](https://github.com/rokucommunity/brighterscript/pull/353)) - bug in xml file ignoring `needsTranspiled` flag when set by plugins ([#384](https://github.com/rokucommunity/brighterscript/pull/384)) -## [0.37.2] - 2021-04-08 +## [0.37.2](https://github.com/rokucommunity/brighterscript/compare/v0.37.1...v0.37.2) - 2021-04-08 ### Fixed - erraneous syntax issue when concatenating a template string and a regular string. ([#383](https://github.com/rokucommunity/brighterscript/pull/383)) - prevent circular import causing stack overflow crash. ([#381](https://github.com/rokucommunity/brighterscript/pull/381)) -## [0.37.1] - 2021-03-30 +## [0.37.1](https://github.com/rokucommunity/brighterscript/compare/v0.37.0...v0.37.1) - 2021-03-30 ### Fixed - bug when transpiling print statements that wouldn't retain the existing separators (semicolon, comma, no separator) which all have unique uses ([#373](https://github.com/rokucommunity/brighterscript/pull/373)) -## [0.37.0] - 2021-03-18 +## [0.37.0](https://github.com/rokucommunity/brighterscript/compare/v0.36.0...v0.37.0) - 2021-03-18 ### Added - support for `bs:disable` comments in xml files ([#363](https://github.com/rokucommunity/brighterscript/pull/363)) -## [0.36.0] - 2021-03-15 +## [0.36.0](https://github.com/rokucommunity/brighterscript/compare/v0.35.0...v0.36.0) - 2021-03-15 ### Added - class import code actions ([#365](https://github.com/rokucommunity/brighterscript/pull/365)) ### Changed @@ -158,7 +150,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.35.0] - 2021-03-09 +## [0.35.0](https://github.com/rokucommunity/brighterscript/compare/v0.34.3...v0.35.0) - 2021-03-09 ### Added - code actions for suggesting import statements in brighterscript files ([#347](https://github.com/rokucommunity/brighterscript/pull/347)) ### Fixed @@ -166,13 +158,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.34.3] -2021-03-05 +## [0.34.3](https://github.com/rokucommunity/brighterscript/compare/v0.34.2...v0.34.3) -2021-03-05 ### Fixed - bug when transpiling bsc with custom function and parameter return types -## [0.34.2] - 2021-03-04 +## [0.34.2](https://github.com/rokucommunity/brighterscript/compare/v0.34.1...v0.34.2) - 2021-03-04 ### Added - support for loading bslib without alias (i.e. `@rokucommunity/bslib`). ### Fixed @@ -182,19 +174,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.34.1] - 2021-03-02 +## [0.34.1](https://github.com/rokucommunity/brighterscript/compare/v0.34.0...v0.34.1) - 2021-03-02 ### Fixed - syntax parsing bugs within single-line if statements -## [0.34.0] - 2021-02-28 +## [0.34.0](https://github.com/rokucommunity/brighterscript/compare/v0.33.0...v0.34.0) - 2021-02-28 ### Added - language server file path completions inside strings that start with `pkg:` or `libpkg:` -## [0.33.0] - 2021-02-27 +## [0.33.0](https://github.com/rokucommunity/brighterscript/compare/v0.32.3...v0.33.0) - 2021-02-27 ### Added - support for ropm version of bslib.([#334](https://github.com/rokucommunity/brighterscript/pull/334)) ### Fixed @@ -204,7 +196,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.32.3] - 2021-02-25 +## [0.32.3](https://github.com/rokucommunity/brighterscript/compare/v0.32.2...v0.32.3) - 2021-02-25 ### Fixed - fix significant performance bug in diagnostic filtering - tweaks to the logging system to make `logLevel=verbose` less chatty @@ -213,25 +205,21 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.32.2] - 2021-02-20 +## [0.31.2](https://github.com/rokucommunity/brighterscript/compare/v0.31.1...v0.31.2) - 2021-02-20 ### Fixed - transpile bug when a template string starts with an expression ([#327](https://github.com/rokucommunity/brighterscript/pull/327)) - - - -## [0.31.2] - 2021-02-18 ### Changed - - parent class names in typedefs for class `extends` expressions now always include their full namespace name. + - when generating type definition files, include the namespace for every extends statement ([#324](https://github.com/rokucommunity/brighterscript/pull/324)) -## [0.31.1] - 2021-02-18 +## [0.31.1](https://github.com/rokucommunity/brighterscript/compare/v0.31.0...v0.31.1) - 2021-02-18 ### Fixed - prevent exception in codeAction functionality when file cannot be found in a `Program` -## [0.31.0] - 2021-02-17 +## [0.31.0](https://github.com/rokucommunity/brighterscript/compare/v0.30.9...v0.31.0) - 2021-02-17 ### Added - plugin and language server support for [codeActions](https://microsoft.github.io/language-server-protocol/specifications/specification-current/#textDocument_codeAction) - codeAction to add missing `extends` attribute in components @@ -241,7 +229,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.30.9] - 2021-02-15 +## [0.30.9](https://github.com/rokucommunity/brighterscript/compare/v0.30.8...v0.30.9) - 2021-02-15 ### Fixed - prevent excess validations when non-workspace files are changed ([#315](https://github.com/rokucommunity/brighterscript/pull/315)) - catch errors when getting signatures ([#285](https://github.com/rokucommunity/brighterscript/pull/285)) @@ -249,7 +237,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.30.8] - 2021-02-12 +## [0.30.8](https://github.com/rokucommunity/brighterscript/compare/v0.30.7...v0.30.8) - 2021-02-12 ### Changed - add additional logging in `Program.removeFile` ### Fixed @@ -258,20 +246,20 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.30.7] - 2021-02-11 +## [0.30.7](https://github.com/rokucommunity/brighterscript/compare/v0.30.6...v0.30.7) - 2021-02-11 ### Fixed - bug in `getSignatureHelp` that wouldn't work for function calls with no leading whitespace. ([#307](https://github.com/rokucommunity/brighterscript/issues/307)) -## [0.30.6] - 2021-02-07 +## [0.30.6](https://github.com/rokucommunity/brighterscript/compare/v0.30.5...v0.30.6) - 2021-02-07 ### Fixed - bad transpile for nested class method super calls - SceneGraph node attributes being wrongly removed when modifying attributes -## [0.30.5] - 2021-02-03 +## [0.30.5](https://github.com/rokucommunity/brighterscript/compare/v0.30.4...v0.30.5) - 2021-02-03 ### Added - syntax support for dim statements - completion and code navigation for labels @@ -280,33 +268,33 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.30.4] - 2021-02-02 +## [0.30.4](https://github.com/rokucommunity/brighterscript/compare/v0.30.3...v0.30.4) - 2021-02-02 ### Fixed - fixed crash during validation caused by a missing function body when parsing malformed code -## [0.30.3] - 2021-02-01 +## [0.30.3](https://github.com/rokucommunity/brighterscript/compare/v0.30.2...v0.30.3) - 2021-02-01 ### Fixed - performance issue when transpiling larger projects, even when no brighterscript code was used -## [0.30.2] - 2021-01-31 +## [0.30.2](https://github.com/rokucommunity/brighterscript/compare/v0.30.1...v0.30.2) - 2021-01-31 ### Fixed - xml parse error crashing validation ((#294)[https://github.com/rokucommunity/brighterscript/pull/294]) - better handling for `createStringLiteral` ((#292)[https://github.com/rokucommunity/brighterscript/pull/292]) -## [0.30.1] - 2021-01-29 +## [0.30.1](https://github.com/rokucommunity/brighterscript/compare/v0.30.0...v0.30.1) - 2021-01-29 ### Fixed - bug that would crash while transpiling if a script tag didn't have a `type` attribute - XML transpile now honors the `sourceMap` option -## [0.30.0] - 2021-01-26 +## [0.30.0](https://github.com/rokucommunity/brighterscript/compare/v0.27.0...v0.28.0) - 2021-01-26 ### Added - null coalescing operator (see [the docs](https://github.com/rokucommunity/brighterscript/blob/master/docs/null-coalescing-operator.md) for more information) ### Fixed @@ -314,13 +302,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.29.0] - 2021-01-25 +## [0.29.0](https://github.com/rokucommunity/brighterscript/compare/v0.26.0...v0.27.0) - 2021-01-25 ### Added - ternary operator (see [the docs](https://github.com/rokucommunity/brighterscript/blob/master/docs/ternary-operator.md) for more information) -## [0.28.2] - 2021-01-22 +## [0.28.2](https://github.com/rokucommunity/brighterscript/compare/v0.25.0...v0.26.0) - 2021-01-22 ### Changed - config loading functions from `util.ts` are now run synchronously ### Fixed @@ -329,7 +317,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.28.1] - 2021-01-19 +## [0.28.1](https://github.com/rokucommunity/brighterscript/compare/v0.24.2...v0.25.0) - 2021-01-19 ### Changed - (For plugin authors) refactored many async methods into sync methods to simplify file creation/management. ([#278](https://github.com/rokucommunity/brighterscript/pull/278)) ### Fixed @@ -338,7 +326,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.28.0] - 2021-01-16 +## [0.28.0](https://github.com/rokucommunity/brighterscript/compare/v0.27.0...v0.28.0) - 2021-01-16 ### Added - annotation support for classes and class methods ([#270](https://github.com/rokucommunity/brighterscript/pull/270)) ### fixed @@ -346,13 +334,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.27.0] - 2021-01-15 +## [0.27.0](https://github.com/rokucommunity/brighterscript/compare/v0.26.0...v0.27.0) - 2021-01-15 ### Changed - plugin system changed to require a factory function instead of a singleton object ([#272](https://github.com/rokucommunity/brighterscript/pull/272)) -## [0.26.0] - 2021-01-14 +## [0.26.0](https://github.com/rokucommunity/brighterscript/compare/v0.25.0...v0.26.0) - 2021-01-14 ### Added - proper XML AST support - component interface validation @@ -360,25 +348,25 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.25.0] - 2021-01-12 +## [0.25.0](https://github.com/rokucommunity/brighterscript/compare/v0.24.2...v0.25.0) - 2021-01-12 ### Added - support for passing custom types as function parameters and return types ([#262](https://github.com/rokucommunity/brighterscript/issues/262)) -## [0.24.2] - 2021-01-11 +## [0.24.2](https://github.com/rokucommunity/brighterscript/compare/v0.24.1...v0.24.2) - 2021-01-11 ### Fixed - bug with transpiled child classes causing on-device stack overflows ([#267](https://github.com/rokucommunity/brighterscript/issues/267)) -## [0.24.1] - 2021-01-09 +## [0.24.1](https://github.com/rokucommunity/brighterscript/compare/v0.24.0...v0.24.1) - 2021-01-09 ### Changed - upgraded to [roku-deploy@3.2.4](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#324---2021-01-08) -## [0.24.0] - 2021-01-08 +## [0.24.0](https://github.com/rokucommunity/brighterscript/compare/v0.23.2...v0.24.0) - 2021-01-08 ### Added - `sourceMap` option to enable/disable generating sourcemaps ### Changed @@ -386,14 +374,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.23.2] - 2020-01-06 +## [0.23.2](https://github.com/rokucommunity/brighterscript/compare/v0.23.1...v0.23.2) - 2020-01-06 ### Fixed - `isLiteralInvalid` was causing infinite recursion. - lock `vscode-languageserver-protocol` package version to prevent issues with vscode not following semantic versioning. -## [0.23.1] - 2020-12-22 +## [0.23.1](https://github.com/rokucommunity/brighterscript/compare/v0.23.0...v0.23.1) - 2020-12-22 ### Changed - renamed `Scope.getFiles()` to `Scope.getAllFiles()` and added a new function called `Scope.getOwnFiles()` ### Fixed @@ -401,7 +389,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.23.0] - 2020-12-18 +## [0.23.0](https://github.com/rokucommunity/brighterscript/compare/v0.22.1...v0.23.0) - 2020-12-18 ### Changed - AST parser refactoring ([#244](https://github.com/rokucommunity/brighterscript/pull/244)) - Make `ElseIf` into an `ElseIfStatement` @@ -416,19 +404,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.22.1] - 2020-12-14 +## [0.22.1](https://github.com/rokucommunity/brighterscript/compare/v0.22.0...v0.22.1) - 2020-12-14 ### Fixed - small bug introduced by vscode-languageserver causing crashes anytime negative range values are provided. -## [0.22.0] - 2020-11-23 +## [0.22.0](https://github.com/rokucommunity/brighterscript/compare/v0.21.0...v0.22.0) - 2020-11-23 ### Added - `try/catch` and `throw` syntax support [#218](https://github.com/rokucommunity/brighterscript/issues/218) -## [0.21.0] - 2020-11-19 +## [0.21.0](https://github.com/rokucommunity/brighterscript/compare/v0.20.1...v0.21.0) - 2020-11-19 ### Added - Catch when local variables and scope functions have the same name as a class. ([#246](https://github.com/rokucommunity/brighterscript/pull/246)) - Catch when functions use keyword names ([#247](https://github.com/rokucommunity/brighterscript/pull/247)) @@ -444,20 +432,20 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - remove eliminated `BrsType` items from `reflection.ts`. -## [0.20.1] - 2020-11-16 +## [0.20.1](https://github.com/rokucommunity/brighterscript/compare/v0.20.0...v0.20.1) - 2020-11-16 ### Changed - load plugins relatively to the project ([#242](https://github.com/rokucommunity/brighterscript/pull/242)) - modified reflection utilities so they are compatible with TS strict null checks ([#243](https://github.com/rokucommunity/brighterscript/pull/243)) -## [0.20.0] - 2020-11-13 +## [0.20.0](https://github.com/rokucommunity/brighterscript/compare/v0.19.0...v0.20.0) - 2020-11-13 ### Added - more language server features: onWorkspaceSymbol, onSignatureHelp, onDocumentSymbol, onReferences, improve onDefinition ([#191](https://github.com/rokucommunity/brighterscript/pull/191)) -## [0.19.0] - 2020-11-04 +## [0.19.0](https://github.com/rokucommunity/brighterscript/compare/v0.18.2...v0.19.0) - 2020-11-04 ### Changed - `emitDefinitions` now defaults to `false` (it previously defaulted to `true`) ### Fixed @@ -465,13 +453,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.18.2] - 2020-11-2 +## [0.18.2](https://github.com/rokucommunity/brighterscript/compare/v0.18.1...v0.18.2) - 2020-11-2 ### Fixed - support on-demand parse for typedef-shadowed files ([#237](https://github.com/rokucommunity/brighterscript/pull/237)) -## [0.18.1] - 2020-10-30 +## [0.18.1](https://github.com/rokucommunity/brighterscript/compare/v0.18.0...v0.18.1) - 2020-10-30 ### Fixed - exclude bs1100 for typedef files (`Missing "super()" call in class constructor method.`) - fix some invalid class field types in typedef files @@ -479,25 +467,25 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.18.0] - 2020-10-30 +## [0.18.0](https://github.com/rokucommunity/brighterscript/compare/v0.17.0...v0.18.0) - 2020-10-30 ### Added - support for consuming and producing type definitions. ([188](https://github.com/rokucommunity/brighterscript/pull/188)) -## [0.17.0] - 2020-10-27 +## [0.17.0](https://github.com/rokucommunity/brighterscript/compare/v0.16.12...v0.17.0) - 2020-10-27 ### Added - Annotation syntax and AST support ([#234](https://github.com/rokucommunity/brighterscript/pull/234)) -## [0.16.12] - 2020-10-21 +## [0.16.12](https://github.com/rokucommunity/brighterscript/compare/v0.16.11...v0.16.12) - 2020-10-21 ### Fixed - parser bug when there was a trailing colon after `for` or `while` loop statements ([#230](https://github.com/rokucommunity/brighterscript/pull/230)) -## [0.16.11] - 2020-10-20 +## [0.16.11](https://github.com/rokucommunity/brighterscript/compare/v0.16.10...v0.16.11 - 2020-10-20 ### Fixed - bug when using single quotes in an xml script tag ### Changed @@ -505,14 +493,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.16.10] - 2020-10-20 +## [0.16.10](https://github.com/rokucommunity/brighterscript/compare/v0.16.9...v0.16.10) - 2020-10-20 ### Fixed - prevent crash when a callable has the same name as a javascript reserved name ([#226](https://github.com/rokucommunity/brighterscript/pull/226)) - prevent crash when `import` statement is malformed ([#224](https://github.com/rokucommunity/brighterscript/pull/224)) -## [0.16.9] - 2020-10-18 +## [0.16.9](https://github.com/rokucommunity/brighterscript/compare/v0.16.8...v0.16.9) - 2020-10-18 ### Fixed - reduce language server throttle for validation and parsing now that those have improved performance. - massively improve validation performance by refactoring `getFileByPkgPath` @@ -521,33 +509,33 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.16.8] - 2020-10-15 +## [0.16.8](https://github.com/rokucommunity/brighterscript/compare/v0.16.7...v0.16.8) - 2020-10-15 ### Fixed - bug when printing diagnostics that would crash if the contents were missing (like for in-memory-only files injected by plugins) ([#217](https://github.com/rokucommunity/brighterscript/pull/217)) - Drop expensive AST walking for collecting property names and instead collect them as part of parsing -## [0.16.7] - 2020-10-13 +## [0.16.7](https://github.com/rokucommunity/brighterscript/compare/v0.16.6...v0.16.7) - 2020-10-13 ### Fixed - bug when finding `bsconfig.json` that would use the wrong cwd in multi-workspace language server situations. - bug when transpiling in-memory-only files. ([#212](https://github.com/rokucommunity/brighterscript/pull/212)) -## [0.16.6] - 2020-10-13 +## [0.16.6](https://github.com/rokucommunity/brighterscript/compare/v0.16.5...v0.16.6) - 2020-10-13 ### Fixed - quirk in the GitHub actions workflow that didn't publish the correct code. -## [0.16.5] - 2020-10-13 +## [0.16.5](https://github.com/rokucommunity/brighterscript/compare/v0.16.4...v0.16.5) - 2020-10-13 ### Fixed - performance issue during the parse phase. We now defer certain collections until needed. ([#210](https://github.com/rokucommunity/brighterscript/pull/210)) -## [0.16.4] - 2020-10-12 +## [0.16.4](https://github.com/rokucommunity/brighterscript/compare/v0.16.3...v0.16.4) - 2020-10-12 ### Changed - LanguageServer now sends a _diff_ of diagnostics for files, instead of the entire project's diagnostics every time. ([#204](https://github.com/rokucommunity/brighterscript/pull/204)) ### Fixed @@ -556,7 +544,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.16.3] - 2020-10-11 +## [0.16.3](https://github.com/rokucommunity/brighterscript/compare/v0.16.2...v0.16.3) - 2020-10-11 ### Changed - Add generic type parameter for `Program` add functions. - Export `BscType` type to simplify `BrsFile | XmlFile` usage everywhere. @@ -565,13 +553,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.16.2] - 2020-10-09 +## [0.16.2](https://github.com/rokucommunity/brighterscript/compare/v0.16.1...v0.16.2) - 2020-10-09 ### Fixed - critical bug in diagnostic printing that would crash the program if a diagnostic was missing a valid range. -## [0.16.1] - 2020-10-03 +## [0.16.1](https://github.com/rokucommunity/brighterscript/compare/v0.16.0...v0.16.1) - 2020-10-03 ### Changed - rename `isEscapedCharCodeLiteral` to `isEscapedCharCodeLiteralExpression` to match other expression class names - rename `FunctionParameter` to `FunctionParameterExpression` to match other expression class names @@ -582,7 +570,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.16.0] - 2020-10-02 +## [0.16.0](https://github.com/rokucommunity/brighterscript/compare/v0.15.2...v0.16.0) - 2020-10-02 ### Added - `Expression.walk` and `Statement.walk` functions which provide shallow or deep walking of the AST - Many `ast` reflection methods to be used instead of `instanceof`. @@ -595,38 +583,38 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.15.2] - 2020-10-01 +## [0.15.2](https://github.com/rokucommunity/brighterscript/compare/v0.15.1...v0.15.2) - 2020-10-01 ### Fixed - Bug in component validation that would throw errors if component name was undefined (generally due to an XML parse error). ([#194](https://github.com/rokucommunity/brighterscript/pull/194)) -## [0.15.1] - 2020-09-30 +## [0.15.1](https://github.com/rokucommunity/brighterscript/compare/v0.15.0...v0.15.1) - 2020-09-30 ### Fixed - improved performance in the lexer and parser - potential for accidentally changing `cwd` during bsconfig resolving -## [0.15.0] - 2020-09-18 +## [0.15.0](https://github.com/rokucommunity/brighterscript/compare/v0.14.0...v0.15.0) - 2020-09-18 ### Added - plugin API to allow visibility into the various compiler phases. This is currently in alpha. ([#170](https://github.com/rokucommunity/brighterscript/pull/170)) -## [0.14.0] - 2020-09-04 +## [0.14.0](https://github.com/rokucommunity/brighterscript/compare/v0.13.2...v0.14.0) - 2020-09-04 ### Changed - Add error diagnostic BS1115 which flags duplicate component names [#186](https://github.com/rokucommunity/brighterscript/pull/186) -## [0.13.2] - 2020-08-31 +## [0.13.2](https://github.com/rokucommunity/brighterscript/compare/v0.13.1...v0.13.2) - 2020-08-31 ### Changed - Upgraded BS1104 to error (previously a warning) and refined the messaging. -## [0.13.1] - 2020-08-14 +## [0.13.1](https://github.com/rokucommunity/brighterscript/compare/v0.13.0...v0.13.1) - 2020-08-14 ### Changed - upgraded to [roku-deploy@3.2.3](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#323---2020-08-14) - throw exception when copying to staging folder and `rootDir` does not exist in the file system @@ -634,26 +622,26 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.13.0] - 2020-08-10 +## [0.13.0](https://github.com/rokucommunity/brighterscript/compare/v0.12.4...v0.13.0) - 2020-08-10 ### Added - ability to mark the `extends` and `project` options in `bsconfig.json`, API and CLI as optional. -## [0.12.4] - 2020-08-06 +## [0.12.4](https://github.com/rokucommunity/brighterscript/compare/v0.12.3...v0.12.4) - 2020-08-06 ### Fixed - bug in cli that wouldn't properly read bsconfig values. [#167](https://github.com/rokucommunity/brighterscript/issues/167) - bug in cli that doesn't use `retain-staging-folder` argument properly. [#168](https://github.com/rokucommunity/brighterscript/issues/168) -## [0.12.3] - 2020-08-03 +## [0.12.3](https://github.com/rokucommunity/brighterscript/compare/v0.12.2...v0.12.3) - 2020-08-03 ### Fixed - bug in the language server that would provide stale completions due to the file throttling introduced in v0.11.2. Now the language server will wait for the throttled parsing to complete before serving completion results. -## [0.12.2] - 2020-07-16 +## [0.12.2](https://github.com/rokucommunity/brighterscript/compare/v0.12.1...v0.12.2) - 2020-07-16 ### Added - Expose `ProgramBuilder.transpile()` method to make it easier for tools to transpile programmatically. [#154](https://github.com/rokucommunity/brighterscript/issues/154) ### Fixed @@ -661,7 +649,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.12.1] - 2020-07-15 +## [0.12.1](https://github.com/rokucommunity/brighterscript/compare/v0.12.0...v0.12.1) - 2020-07-15 ### Changed - upgraded to [roku-deploy@3.2.2](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#322---2020-07-14) ### Fixed @@ -669,7 +657,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.12.0] - 2020-07-09 +## [0.12.0](https://github.com/rokucommunity/brighterscript/compare/v0.11.2...v0.12.0) - 2020-07-09 ### Added - `diagnosticLevel` option to limit/control the noise in the console diagnostics ### Changed @@ -681,7 +669,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.11.2] - 2020-07-09 +## [0.11.2](https://github.com/rokucommunity/brighterscript/compare/v0.11.1...v0.11.2) - 2020-07-09 ### Changed - add 350ms debounce in LanguageServer `onDidChangeWatchedFiles` to increase performance by reducing the number of times a file is parsed and validated. ### Fixed @@ -691,7 +679,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.11.1] - 2020-07-07 +## [0.11.1](https://github.com/rokucommunity/brighterscript/compare/v0.11.0...v0.11.1) - 2020-07-07 ### Added - diagnostic for unknown file reference in import statements ([#139](https://github.com/rokucommunity/brighterscript/pull/139)) ### Changed @@ -704,7 +692,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.11.0] - 2020-07-06 +## [0.11.0](https://github.com/rokucommunity/brighterscript/compare/v0.10.10...v0.11.0) - 2020-07-06 ### Added - [Source literals feature](https://github.com/rokucommunity/brighterscript/blob/master/docs/source-literals.md) which adds new literals such as `SOURCE_FILE_PATH`, `SOURCE_LINE_NUM`, `FUNCTION_NAME`, and more. ([#13](https://github.com/rokucommunity/brighterscript/issues/13)) - `sourceRoot` config option to fix sourcemap paths for projects that use a temporary staging folder before calling the BrighterScript compiler. ([#134](https://github.com/rokucommunity/brighterscript/commit/e5b73ca37016d5015a389257fb259573c4721e7a)) @@ -714,18 +702,18 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.10.11] - 2020-07-05 +## [0.10.11](https://github.com/rokucommunity/brighterscript/compare/v0.10.10...v0.10.11) - 2020-07-05 - Fix bug that would fail to copy files to staging without explicitly specifying `stagingFolderpath`. [#129](https://github.com/rokucommunity/brighterscript/issues/129) -## [0.10.10] - 2020-06-12 +## [0.10.10](https://github.com/rokucommunity/brighterscript/compare/v0.10.9...v0.10.10) - 2020-06-12 ### Fixed - include the missing `bslib.brs` file in the npm package which was causing errors during transpile. -## [0.10.9] 2020-06-12 +## [0.10.9](https://github.com/rokucommunity/brighterscript/compare/v0.10.8...v0.10.9) 2020-06-12 ### Added - bslib.brs gets copied to `pkg:/source` and added as an import to every component on transpile. - several timing logs under the `"info"` log level. @@ -737,32 +725,32 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.10.8] - 2020-06-09 +## [0.10.8](https://github.com/rokucommunity/brighterscript/compare/v0.10.7...v0.10.8) - 2020-06-09 ### Fixed - Allow leading spcaes for `bs:disable-line` and `bs:disable-next-line` comments ([#108](https://github.com/rokucommunity/brighterscript/pull/108)) -## [0.10.7] - 2020-06-08 +## [0.10.7](https://github.com/rokucommunity/brighterscript/compare/v0.10.6...v0.10.7) - 2020-06-08 ### Fixed - bug in cli that was always returning a nonzero error code -## [0.10.6] - 2020-06-05 +## [0.10.6](https://github.com/rokucommunity/brighterscript/compare/v0.10.5...v0.10.6) - 2020-06-05 ### Fixed - incorrect definition for global `Left()` function. ([#100](https://github.com/rokucommunity/brighterscript/issues/100)) - missing definition for global `Tab()` and `Pos()` functions ([#101](https://github.com/rokucommunity/brighterscript/issues/101)) -## [0.10.5] - 2020-06-04 +## [0.10.5](https://github.com/rokucommunity/brighterscript/compare/v0.10.4...v0.10.5) - 2020-06-04 ### Changed - added better logging for certain critical language server crashes -## [0.10.4] - 2020-05-28 +## [0.10.4](https://github.com/rokucommunity/brighterscript/compare/v0.10.3...v0.10.4) - 2020-05-28 ### Fixed - bug where assigning a namespaced function to a variable wasn't properly transpiling the dots to underscores (fixes [#91](https://github.com/rokucommunity/brighterscript/issues/91)) - flag parameter with same name as namespace @@ -771,14 +759,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.10.3] - 2020-05-27 +## [0.10.3](https://github.com/rokucommunity/brighterscript/compare/v0.10.2...v0.10.3) - 2020-05-27 ### Changed - tokenizing a string with no closing quote will now include all of the text until the end of the line. - language server `TranspileFile` command now waits until the program is finished building before trying to transpile. -## [0.10.2] - 2020-05-23 +## [0.10.2](https://github.com/rokucommunity/brighterscript/compare/v0.10.1...v0.10.2) - 2020-05-23 ### Added - language server command `TranspileFile` which will return the transpiled contents of the requested file. ### Fixed @@ -787,7 +775,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.10.1] - 2020-05-22 +## [0.10.1](https://github.com/rokucommunity/brighterscript/compare/v0.10.0...v0.10.1) - 2020-05-22 ### Fixed - transpile bug for compound assignment statements (such as `+=`, `-=`) ([#87](https://github.com/rokucommunity/brighterscript/issues/87)) - transpile bug that was inserting function parameter types before default values ([#88](https://github.com/rokucommunity/brighterscript/issues/88)) @@ -795,13 +783,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.10.0] - 2020-05-19 +## [0.10.0](https://github.com/rokucommunity/brighterscript/compare/v0.9.8...v0.10.0) - 2020-05-19 ### Added - new callfunc operator. -## [0.9.8] - 2020-05-16 +## [0.9.8](https://github.com/rokucommunity/brighterscript/compare/v0.9.7...v0.9.8) - 2020-05-16 ### Changed - the inner event system handling file changes. This should fix several race conditions causing false negatives during CLI runs. ### Fixed @@ -809,7 +797,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.7] - 2020-05-14 +## [0.9.7](https://github.com/rokucommunity/brighterscript/compare/v0.9.6...v0.9.7) - 2020-05-14 ### Changed - TypeScript target to "ES2017" which provides a significant performance boost in lexer (~30%) and parser (~175%) ### Fixed @@ -819,7 +807,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.6] - 2020-05-11 +## [0.9.6](https://github.com/rokucommunity/brighterscript/compare/v0.9.5...v0.9.6) - 2020-05-11 ### Added - `logLevel` option from the bsconfig.json and command prompt that allows specifying how much detain the logging should contain. - additional messages during cli run @@ -831,7 +819,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.5] - 2020-05-06 +## [0.9.5](https://github.com/rokucommunity/brighterscript/compare/v0.9.4...v0.9.5) - 2020-05-06 ### Added - new config option called `showDiagnosticsInConsole` which disables printing diagnostics to the console ### Fixed @@ -841,7 +829,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.4] - 2020-05-05 +## [0.9.4](https://github.com/rokucommunity/brighterscript/compare/v0.9.3...v0.9.4) - 2020-05-05 ### Added - diagnostic for detecting unnecessary script imports when `autoImportComponentScript` is enabled ### Changed @@ -854,7 +842,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.3] - 2020-05-04 +## [0.9.3](https://github.com/rokucommunity/brighterscript/compare/v0.9.2...v0.9.3) - 2020-05-04 ### Changed - do not show BRS1013 for standalone files ([#72](https://github.com/rokucommunity/brighterscript/issues/72)) - BS1011 (same name as global function) is no longer shown for local variables that are not of type `function` ([#70](https://github.com/rokucommunity/brighterscript/issues/70)) @@ -863,7 +851,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.2] - 2020-05-02 +## [0.9.2](https://github.com/rokucommunity/brighterscript/compare/v0.9.1...v0.9.2) - 2020-05-02 ### Changed - intellisense anywhere other than next to a dot now includes keywords (#67) @@ -874,19 +862,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.9.1] - 2020-05-01 +## [0.9.1](https://github.com/rokucommunity/brighterscript/compare/v0.9.0...v0.9.1) - 2020-05-01 ### Fixed - bug with upper-case two-word conditional compile tokens (`#ELSE IF` and `#END IF`) (#63) -## [0.9.0] - 2020-05-01 +## [0.9.0](https://github.com/rokucommunity/brighterscript/compare/v0.8.2...v0.9.0) - 2020-05-01 ### Added - new compile flag `autoImportComponentScript` which will automatically import a script for a component with the same name if it exists. -## [0.8.2] - 2020-04-29 +## [0.8.2](https://github.com/rokucommunity/brighterscript/compare/v0.8.1...v0.8.2) - 2020-04-29 ### Fixed - bugs in namespace transpilation - bugs in class transpilation @@ -895,14 +883,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.8.1] - 2020-04-27 +## [0.8.1](https://github.com/rokucommunity/brighterscript/compare/v0.8.0...v0.8.1) - 2020-04-27 ### Fixed - Bug where class property initializers would cause parse error - better parse recovery for incomplete class members -## [0.8.0] - 2020-04-26 +## [0.8.0](https://github.com/rokucommunity/brighterscript/compare/v0.7.2...v0.8.0) - 2020-04-26 ### Added - new `import` syntax for BrighterScript projects. - experimental transpile support for xml files (converts `.bs` extensions to `.brs`, auto-appends the `import` statments to each xml component) @@ -910,19 +898,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - upgraded to vscode-languageserver@6.1.1 -## [0.7.2] - 2020-04-24 +## [0.7.2](https://github.com/rokucommunity/brighterscript/compare/v0.7.1...v0.7.2) - 2020-04-24 ### Fixed - runtime bug in the language server when validating incomplete class statements -## [0.7.1] - 2020-04-23 +## [0.7.1](https://github.com/rokucommunity/brighterscript/compare/v0.7.0...v0.7.1) - 2020-04-23 ### Fixed - dependency issue: `glob` is required but was not listed as a dependency -## [0.7.0] - 2020-04-23 +## [0.7.0](https://github.com/rokucommunity/brighterscript/compare/v0.6.0...v0.7.0) - 2020-04-23 ### Added - basic support for namespaces - experimental parser support for import statements (no transpile yet) @@ -931,7 +919,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.6.0] 2020-04-15 +## [0.6.0](https://github.com/rokucommunity/brighterscript/compare/v0.5.4...v0.6.0) 2020-04-15 ### Added - ability to filter out diagnostics by using the `diagnosticFilters` option in bsconfig ### Changed @@ -941,14 +929,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.5.4] 2020-04-13 +## [0.5.4](https://github.com/rokucommunity/brighterscript/compare/v0.5.3...v0.5.4) 2020-04-13 ### Fixed - Syntax bug that wasn't allowing period before indexed get expression (example: `prop.["key"]`) (#58) - Syntax bug preventing comments from being used in various locations within a class -## [0.5.3] - 2020-04-12 +## [0.5.3](https://github.com/rokucommunity/brighterscript/compare/v0.5.2...v0.5.3) - 2020-04-12 ### Added - syntax support for the xml attribute operator (`node@someAttr`) (#34) - syntax support for bitshift operators (`<<` and `>>`) (#50) @@ -958,7 +946,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.5.2] - 2020-04-11 +## [0.5.2](https://github.com/rokucommunity/brighterscript/compare/v0.5.1...v0.5.2) - 2020-04-11 ### Changed - downgrade diagnostic issue 1007 from an error to a warning, and updated the message to "Component is mising "extends" attribute and will automatically extend "Group" by default" (#53) ### Fixed @@ -967,13 +955,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.5.1] - 2020-04-10 +## [0.5.1](https://github.com/rokucommunity/brighterscript/compare/v0.5.0...v0.5.1) - 2020-04-10 ### Changed - upgraded to [roku-deploy@3.0.2](https://www.npmjs.com/package/roku-debug/v/0.3.4) which fixed a file copy bug in subdirectories of symlinked folders (fixes #41) -## [0.5.0] - 2020-04-10 +## [0.5.0](https://github.com/rokucommunity/brighterscript/compare/v0.4.4...v0.5.0) - 2020-04-10 ### Added - several new diagnostics for conditional compiles. Some of them allow the parser to recover and continue. - experimental class transpile support. There is still no intellisense for classes yet though. @@ -986,31 +974,31 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.4.4] - 2020-04-04 +## [0.4.4](https://github.com/rokucommunity/brighterscript/compare/v0.4.3...v0.4.4) - 2020-04-04 ### Fixed - bug in the ProgramBuilder that would terminate the program on first run if an error diagnostic was found, even when in watch mode. -## [0.4.3] - 2020-04-03 +## [0.4.3](https://github.com/rokucommunity/brighterscript/compare/v0.4.2...v0.4.3) - 2020-04-03 ### Changed - the `bsc` cli now emits a nonzero return code whenever parse errors are encountered, which allows tools to detect compile-time errors. (#43) -## [0.4.2] - 2020-04-01 +## [0.4.2](https://github.com/rokucommunity/brighterscript/compare/v0.4.1...v0.4.2) - 2020-04-01 ### Changed - upgraded to [roku-deploy@3.0.0](https://www.npmjs.com/package/roku-deploy/v/3.0.0) -## [0.4.1] - 2020-01-11 +## [0.4.1](https://github.com/rokucommunity/brighterscript/compare/v0.4.0...v0.4.1) - 2020-01-11 ### Changed - upgraded to [roku-deploy@3.0.0-beta.7](https://www.npmjs.com/package/roku-deploy/v/3.0.0-beta.7) which fixed a critical bug during pkg creation. -## [0.4.0] - 2020-01-07 +## [0.4.0](https://github.com/rokucommunity/brighterscript/compare/v0.3.1...v0.4.0) - 2020-01-07 ### Added - ability to specify the pkgPath of a file when adding to the project. ### Changed @@ -1023,14 +1011,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.3.1] - 2019-11-08 +## [0.3.1](https://github.com/rokucommunity/brighterscript/compare/v0.3.0...v0.3.1) - 2019-11-08 ### Fixed - language server bug that was showing error messages in certain startup race conditions. - error during hover caused by race condition during file re-parse. -## [0.3.0] - 2019-10-03 +## [0.3.0](https://github.com/rokucommunity/brighterscript/compare/v0.2.2...v0.3.0) - 2019-10-03 ### Added - support for parsing opened files not included in any project. ### Fixed @@ -1038,13 +1026,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.2.2] - 2019-09-27 +## [0.2.2](https://github.com/rokucommunity/brighterscript/compare/v0.2.1...v0.2.2) - 2019-09-27 ### Fixed - bug in language server where the server would crash when sending a diagnostic too early. Now the server waits for the program to load before sending diagnostics. -## [0.2.1] - 2019-09-24 +## [0.2.1](https://github.com/rokucommunity/brighterscript/compare/v0.2.0...v0.2.1) - 2019-09-24 ### Changed - the text for diagnostic 1010 to say "override" instead of "shadows" ### Fixed @@ -1055,7 +1043,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 -## [0.2.0] - 2019-09-20 +## [0.2.0](https://github.com/rokucommunity/brighterscript/compare/v0.1.0...v0.2.0) - 2019-09-20 ### Added - bsconfig.json validation - slightly smarter intellisense that knows when you're trying to complete an object property. @@ -1070,132 +1058,16 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Bug during file creation that wouldn't recognize the file -## [0.1.0] - 2019-08-10 +## [0.1.0](https://github.com/rokucommunity/brighterscript/compare/v0.1.0...v0.1.0) - 2019-08-10 ### Changed - Cloned from [brightscript-language](https://github.com/rokucommunity/brightscript-language) -[0.1.0]: https://github.com/rokucommunity/brighterscript/compare/v0.1.0...v0.1.0 -[0.2.0]: https://github.com/rokucommunity/brighterscript/compare/v0.1.0...v0.2.0 -[0.2.1]: https://github.com/rokucommunity/brighterscript/compare/v0.2.0...v0.2.1 -[0.2.2]: https://github.com/rokucommunity/brighterscript/compare/v0.2.1...v0.2.2 -[0.3.0]: https://github.com/rokucommunity/brighterscript/compare/v0.2.2...v0.3.0 -[0.3.1]: https://github.com/rokucommunity/brighterscript/compare/v0.3.0...v0.3.1 -[0.4.0]: https://github.com/rokucommunity/brighterscript/compare/v0.3.1...v0.4.0 -[0.4.1]: https://github.com/rokucommunity/brighterscript/compare/v0.4.0...v0.4.1 -[0.4.2]: https://github.com/rokucommunity/brighterscript/compare/v0.4.1...v0.4.2 -[0.4.3]: https://github.com/rokucommunity/brighterscript/compare/v0.4.2...v0.4.3 -[0.4.4]: https://github.com/rokucommunity/brighterscript/compare/v0.4.3...v0.4.4 -[0.5.0]: https://github.com/rokucommunity/brighterscript/compare/v0.4.4...v0.5.0 -[0.5.1]: https://github.com/rokucommunity/brighterscript/compare/v0.5.0...v0.5.1 -[0.5.2]: https://github.com/rokucommunity/brighterscript/compare/v0.5.1...v0.5.2 -[0.5.3]: https://github.com/rokucommunity/brighterscript/compare/v0.5.2...v0.5.3 -[0.5.4]: https://github.com/rokucommunity/brighterscript/compare/v0.5.3...v0.5.4 -[0.6.0]: https://github.com/rokucommunity/brighterscript/compare/v0.5.4...v0.6.0 -[0.7.0]: https://github.com/rokucommunity/brighterscript/compare/v0.6.0...v0.7.0 -[0.7.1]: https://github.com/rokucommunity/brighterscript/compare/v0.7.0...v0.7.1 -[0.7.2]: https://github.com/rokucommunity/brighterscript/compare/v0.7.1...v0.7.2 -[0.8.0]: https://github.com/rokucommunity/brighterscript/compare/v0.7.2...v0.8.0 -[0.8.1]: https://github.com/rokucommunity/brighterscript/compare/v0.8.0...v0.8.1 -[0.8.2]: https://github.com/rokucommunity/brighterscript/compare/v0.8.1...v0.8.2 -[0.9.0]: https://github.com/rokucommunity/brighterscript/compare/v0.8.2...v0.9.0 -[0.9.1]: https://github.com/rokucommunity/brighterscript/compare/v0.9.0...v0.9.1 -[0.9.2]: https://github.com/rokucommunity/brighterscript/compare/v0.9.1...v0.9.2 -[0.9.3]: https://github.com/rokucommunity/brighterscript/compare/v0.9.2...v0.9.3 -[0.9.4]: https://github.com/rokucommunity/brighterscript/compare/v0.9.3...v0.9.4 -[0.9.5]: https://github.com/rokucommunity/brighterscript/compare/v0.9.4...v0.9.5 -[0.9.6]: https://github.com/rokucommunity/brighterscript/compare/v0.9.5...v0.9.6 -[0.9.7]: https://github.com/rokucommunity/brighterscript/compare/v0.9.6...v0.9.7 -[0.9.8]: https://github.com/rokucommunity/brighterscript/compare/v0.9.7...v0.9.8 -[0.10.0]: https://github.com/rokucommunity/brighterscript/compare/v0.9.8...v0.10.0 -[0.10.1]: https://github.com/rokucommunity/brighterscript/compare/v0.10.0...v0.10.1 -[0.10.2]: https://github.com/rokucommunity/brighterscript/compare/v0.10.1...v0.10.2 -[0.10.3]: https://github.com/rokucommunity/brighterscript/compare/v0.10.2...v0.10.3 -[0.10.4]: https://github.com/rokucommunity/brighterscript/compare/v0.10.3...v0.10.4 -[0.10.5]: https://github.com/rokucommunity/brighterscript/compare/v0.10.4...v0.10.5 -[0.10.6]: https://github.com/rokucommunity/brighterscript/compare/v0.10.5...v0.10.6 -[0.10.7]: https://github.com/rokucommunity/brighterscript/compare/v0.10.6...v0.10.7 -[0.10.8]: https://github.com/rokucommunity/brighterscript/compare/v0.10.7...v0.10.8 -[0.10.9]: https://github.com/rokucommunity/brighterscript/compare/v0.10.8...v0.10.9 -[0.10.10]: https://github.com/rokucommunity/brighterscript/compare/v0.10.9...v0.10.10 -[0.11.0]: https://github.com/rokucommunity/brighterscript/compare/v0.10.10...v0.11.0 -[0.11.1]: https://github.com/rokucommunity/brighterscript/compare/v0.11.0...v0.11.1 -[0.11.2]: https://github.com/rokucommunity/brighterscript/compare/v0.11.1...v0.11.2 -[0.11.3]: https://github.com/rokucommunity/brighterscript/compare/v0.11.2...v0.11.3 -[0.12.0]: https://github.com/rokucommunity/brighterscript/compare/v0.11.3...v0.12.0 -[0.12.1]: https://github.com/rokucommunity/brighterscript/compare/v0.12.0...v0.12.1 -[0.12.2]: https://github.com/rokucommunity/brighterscript/compare/v0.12.1...v0.12.2 -[0.12.3]: https://github.com/rokucommunity/brighterscript/compare/v0.12.2...v0.12.3 -[0.12.4]: https://github.com/rokucommunity/brighterscript/compare/v0.12.3...v0.12.4 -[0.13.0]: https://github.com/rokucommunity/brighterscript/compare/v0.12.4...v0.13.0 -[0.13.1]: https://github.com/rokucommunity/brighterscript/compare/v0.13.0...v0.13.1 -[0.13.2]: https://github.com/rokucommunity/brighterscript/compare/v0.13.1...v0.13.2 -[0.14.0]: https://github.com/rokucommunity/brighterscript/compare/v0.13.2...v0.14.0 -[0.15.0]: https://github.com/rokucommunity/brighterscript/compare/v0.14.0...v0.15.0 -[0.15.1]: https://github.com/rokucommunity/brighterscript/compare/v0.15.0...v0.15.1 -[0.15.2]: https://github.com/rokucommunity/brighterscript/compare/v0.15.1...v0.15.2 -[0.16.0]: https://github.com/rokucommunity/brighterscript/compare/v0.15.2...v0.16.0 -[0.16.1]: https://github.com/rokucommunity/brighterscript/compare/v0.16.0...v0.16.1 -[0.16.2]: https://github.com/rokucommunity/brighterscript/compare/v0.16.1...v0.16.2 -[0.16.3]: https://github.com/rokucommunity/brighterscript/compare/v0.16.2...v0.16.3 -[0.16.4]: https://github.com/rokucommunity/brighterscript/compare/v0.16.3...v0.16.4 -[0.16.5]: https://github.com/rokucommunity/brighterscript/compare/v0.16.4...v0.16.5 -[0.16.6]: https://github.com/rokucommunity/brighterscript/compare/v0.16.5...v0.16.6 -[0.16.7]: https://github.com/rokucommunity/brighterscript/compare/v0.16.6...v0.16.7 -[0.16.8]: https://github.com/rokucommunity/brighterscript/compare/v0.16.7...v0.16.8 -[0.16.9]: https://github.com/rokucommunity/brighterscript/compare/v0.16.8...v0.16.9 -[0.16.10]: https://github.com/rokucommunity/brighterscript/compare/v0.16.9...v0.16.10 -[0.16.11]: https://github.com/rokucommunity/brighterscript/compare/v0.16.10...v0.16.11 -[0.16.12]: https://github.com/rokucommunity/brighterscript/compare/v0.16.11...v0.16.12 -[0.17.0]: https://github.com/rokucommunity/brighterscript/compare/v0.16.12...v0.17.0 -[0.18.0]: https://github.com/rokucommunity/brighterscript/compare/v0.17.0...v0.18.0 -[0.18.1]: https://github.com/rokucommunity/brighterscript/compare/v0.18.0...v0.18.1 -[0.18.2]: https://github.com/rokucommunity/brighterscript/compare/v0.18.1...v0.18.2 -[0.19.0]: https://github.com/rokucommunity/brighterscript/compare/v0.18.2...v0.19.0 -[0.20.0]: https://github.com/rokucommunity/brighterscript/compare/v0.19.0...v0.20.0 -[0.20.1]: https://github.com/rokucommunity/brighterscript/compare/v0.20.0...v0.20.1 -[0.21.0]: https://github.com/rokucommunity/brighterscript/compare/v0.20.1...v0.21.0 -[0.22.0]: https://github.com/rokucommunity/brighterscript/compare/v0.21.0...v0.22.0 -[0.22.1]: https://github.com/rokucommunity/brighterscript/compare/v0.22.0...v0.22.1 -[0.22.1]: https://github.com/rokucommunity/brighterscript/compare/v0.22.0...v0.22.1 -[0.23.0]: https://github.com/rokucommunity/brighterscript/compare/v0.22.1...v0.23.0 -[0.23.1]: https://github.com/rokucommunity/brighterscript/compare/v0.23.0...v0.23.1 -[0.23.2]: https://github.com/rokucommunity/brighterscript/compare/v0.23.1...v0.23.2 -[0.24.0]: https://github.com/rokucommunity/brighterscript/compare/v0.23.2...v0.24.0 -[0.24.1]: https://github.com/rokucommunity/brighterscript/compare/v0.24.0...v0.24.1 -[0.24.2]: https://github.com/rokucommunity/brighterscript/compare/v0.24.1...v0.24.2 -[0.25.0]: https://github.com/rokucommunity/brighterscript/compare/v0.24.2...v0.25.0 -[0.26.0]: https://github.com/rokucommunity/brighterscript/compare/v0.25.0...v0.26.0 -[0.27.0]: https://github.com/rokucommunity/brighterscript/compare/v0.26.0...v0.27.0 -[0.28.0]: https://github.com/rokucommunity/brighterscript/compare/v0.27.0...v0.28.0 -[0.28.1]: https://github.com/rokucommunity/brighterscript/compare/v0.28.0...v0.28.1 -[0.28.2]: https://github.com/rokucommunity/brighterscript/compare/v0.28.1...v0.28.2 -[0.29.0]: https://github.com/rokucommunity/brighterscript/compare/v0.28.2...v0.29.0 -[0.30.0]: https://github.com/rokucommunity/brighterscript/compare/v0.29.0...v0.30.0 -[0.30.1]: https://github.com/rokucommunity/brighterscript/compare/v0.30.0...v0.30.1 -[0.30.2]: https://github.com/rokucommunity/brighterscript/compare/v0.30.1...v0.30.2 -[0.30.3]: https://github.com/rokucommunity/brighterscript/compare/v0.30.2...v0.30.3 -[0.30.4]: https://github.com/rokucommunity/brighterscript/compare/v0.30.3...v0.30.4 -[0.30.5]: https://github.com/rokucommunity/brighterscript/compare/v0.30.4...v0.30.5 -[0.30.6]: https://github.com/rokucommunity/brighterscript/compare/v0.30.5...v0.30.6 -[0.30.7]: https://github.com/rokucommunity/brighterscript/compare/v0.30.6...v0.30.7 -[0.30.8]: https://github.com/rokucommunity/brighterscript/compare/v0.30.7...v0.30.8 -[0.30.9]: https://github.com/rokucommunity/brighterscript/compare/v0.30.8...v0.30.9 -[0.31.0]: https://github.com/rokucommunity/brighterscript/compare/v0.30.9...v0.31.0 -[0.31.1]: https://github.com/rokucommunity/brighterscript/compare/v0.31.0...v0.31.1 -[0.31.2]: https://github.com/rokucommunity/brighterscript/compare/v0.31.1...v0.31.2 -[0.32.2]: https://github.com/rokucommunity/brighterscript/compare/v0.31.2...v0.32.2 -[0.32.3]: https://github.com/rokucommunity/brighterscript/compare/v0.32.2...v0.32.3 -[0.33.0]: https://github.com/rokucommunity/brighterscript/compare/v0.32.3...v0.33.0 -[0.34.0]: https://github.com/rokucommunity/brighterscript/compare/v0.33.0...v0.34.0 -[0.34.1]: https://github.com/rokucommunity/brighterscript/compare/v0.34.0...v0.34.1 -[0.34.2]: https://github.com/rokucommunity/brighterscript/compare/v0.34.1...v0.34.2 -[0.34.3]: https://github.com/rokucommunity/brighterscript/compare/v0.34.2...v0.34.3 -[0.35.0]: https://github.com/rokucommunity/brighterscript/compare/v0.34.3...v0.35.0 -[0.36.0]: https://github.com/rokucommunity/brighterscript/compare/v0.35.0...v0.36.0 -[0.37.0]: https://github.com/rokucommunity/brighterscript/compare/v0.36.0...v0.37.0 -[0.37.1]: https://github.com/rokucommunity/brighterscript/compare/v0.37.0...v0.37.1 -[0.37.2]: https://github.com/rokucommunity/brighterscript/compare/v0.37.1...v0.37.2 -[0.37.3]: https://github.com/rokucommunity/brighterscript/compare/v0.37.2...v0.37.3 -[0.37.4]: https://github.com/rokucommunity/brighterscript/compare/v0.37.3...v0.37.4 -[0.38.0]: https://github.com/rokucommunity/brighterscript/compare/v0.37.4...v0.38.0 + + + + + + + + From 2612b195c21b2f26962d56cbf0c9091bc1aa4e05 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Thu, 28 Oct 2021 10:52:25 -0400 Subject: [PATCH 067/278] Update README.md --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 3675c3d42..8d7fb8f43 100644 --- a/README.md +++ b/README.md @@ -60,9 +60,9 @@ BrighterScript adds several new features to the BrightScript language such as Na ## Who uses Brighterscript? -Brighterscript is used by [applicaster](https://www.applicaster.com/), [The miracle channel](https://miraclechannel.ca/corco/), and in open source projects such as [rooibos](https://github.com/georgejecook/rooibos/blob/master/docs/index.md), and the [maestro framework](https://github.com/georgejecook/maestro/blob/master/docs/index.md). The language has been in use for almost a year, and there are at least 2,000+ BrighterScript transpiled files on published channels. +Brighterscript is used by [applicaster](https://www.applicaster.com/), [The miracle channel](https://miraclechannel.ca/corco/), and in open source projects such as [rooibos](https://github.com/georgejecook/rooibos/blob/master/docs/index.md), the [maestro framework](https://github.com/georgejecook/maestro/blob/master/docs/index.md), and more. -The BrighterScript project is used to power the popular [Brightscript Language](https://marketplace.visualstudio.com/items?itemName=celsoaf.brightscript) VSCode extension, and other tools. +The BrighterScript project is used to power the popular [Brightscript Language](https://marketplace.visualstudio.com/items?itemName=rokucommunity.brightscript) VSCode extension, and other tools. More projects are adopting BrighterScript all the time, from using the new BrighterScript language features to simply using the compiler as part of their build pipeline. Be sure to watch this space! From 199e0efb45807324d5467a950e74b9d0f9c23252 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Thu, 28 Oct 2021 15:26:08 -0400 Subject: [PATCH 068/278] Add transpile fix for instantresume components (#465) * Add transpile fix for instantresume components * Fix bug --- src/astUtils/xml.ts | 3 +++ src/files/XmlFile.spec.ts | 37 +++++++++++++++++++++++++++++++++++++ src/parser/SGParser.ts | 2 +- src/parser/SGTypes.ts | 13 ++++++++----- 4 files changed, 49 insertions(+), 6 deletions(-) diff --git a/src/astUtils/xml.ts b/src/astUtils/xml.ts index bb501eec7..11c2df7fa 100644 --- a/src/astUtils/xml.ts +++ b/src/astUtils/xml.ts @@ -21,3 +21,6 @@ export function isSGFunction(tag: SGTag): tag is SGFunction { export function isSGNode(tag: SGTag): tag is SGNode { return tag?.constructor.name === 'SGNode'; } +export function isSGCustomization(tag: SGTag): tag is SGNode { + return isSGNode(tag) && tag.tag?.text?.toLowerCase() === 'customization'; +} diff --git a/src/files/XmlFile.spec.ts b/src/files/XmlFile.spec.ts index 261dc84c7..fee1427ed 100644 --- a/src/files/XmlFile.spec.ts +++ b/src/files/XmlFile.spec.ts @@ -11,6 +11,8 @@ import { BrsFile } from './BrsFile'; import { XmlFile } from './XmlFile'; import { standardizePath as s } from '../util'; import { expectZeroDiagnostics, getTestTranspile, trim, trimMap } from '../testHelpers.spec'; +import { ProgramBuilder } from '../ProgramBuilder'; +import { LogLevel } from '../Logger'; describe('XmlFile', () => { const tempDir = s`${process.cwd()}/.tmp`; @@ -624,6 +626,41 @@ describe('XmlFile', () => { }); describe('transpile', () => { + it('supports instantresume elements', async () => { + fsExtra.outputFileSync(`${rootDir}/manifest`, ''); + fsExtra.outputFileSync(`${rootDir}/source/main.brs`, `sub main()\nend sub`); + fsExtra.outputFileSync(`${rootDir}/components/MainScene.xml`, trim` + + + + + + + + + `); + const builder = new ProgramBuilder(); + await builder.run({ + cwd: rootDir, + retainStagingFolder: true, + stagingFolderPath: stagingDir, + logLevel: LogLevel.off + }); + expect( + fsExtra.readFileSync(`${stagingDir}/components/MainScene.xml`).toString() + ).to.eql(trim` + + + \n\n\n\n\n```", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "renderLast", + "description": "| Option | Description |\n| --- | --- |\n| `\"renderFirst\"` | any drawing done by this node will be done **before** the node children are rendered |\n| `\"renderLast\"` | any drawing done by this node will be done **after** the node children are rendered |", + "name": "childRenderOrder", + "type": "option as string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ 0.0, 0.0, 0.0, 0.0 \\]", + "description": "Specifies a rectangle in the node local coordinate system that is used to limit the region where this node and its children can render. If a non-empty rectangle is specified, then all drawing by this node and its children will be limited to that rectangular area. \\* \\`ClippingRects\\` can be specified by the node or by any of its ancestors in the SceneGraph. \\* \\`ClippingRects\\` are automatically set by some nodes such as lists and grids. \\* \\`ClippingRects\\` are always clipped to the screen boundaries, so if a \\`clippingRect\\` is specified that is partially or completely offscreen, it will be clipped to the screen boundaries. With respect to render tracking, although the node could be completely within the bounds of the specified \\`clippingRect\\`, it's \\`renderTracking\\` field could be set to \\`\"none\"\\` if the portion of the \\`clippingRect\\` it occupies is completely offscreen.", + "name": "clippingRect", + "type": "array of float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "If true, renderTracking will be set to a string describing how much of the node is rendered on screen", + "name": "enableRenderTracking", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "true", + "description": "If true, the node opacity is determined by multiplying opacity attribute of the node by the opacity of the parent node, which may have been determined by multiplying the opacity of its ancestor nodes. If false, the node opacity is determined by the opacity attribute set for the node or the default opacity attribute value", + "name": "inheritParentOpacity", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "true", + "description": "If true, the node overall transformation is determined by combining the accumulated transformation matrix of all of its ancestors in the SceneGraph with the node local 2D transformation matrix described by its translation, rotation, scale and scaleRotateCenter fields. If false, the accumulated transformation of all of its ancestors in the SceneGraph is ignored and only the node local transformation matrix is used. This causes the node to be transformed relative to the root of the SceneGraph (that is, the Scene component)", + "name": "inheritParentTransform", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Set to true to suppress the default CVAA text to speech. This allows channels to provide their own custom implementation", + "name": "muteAudioGuide", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "1.0", + "description": "Sets the opacity of the node and its children. Opacity is the opposite of transparency. Opacity values range from 0.0 (fully transparent) to 1.0 (fully opaque). As the SceneGraph is traversed, the opacity values are combined by multiplying the current accumulated opacity with the node opacity, so that if the accumulated opacity of a node ancestors is 0.25 (75% transparent), the node will have opacity of 0.25 or less. This allows entire branches of the SceneGraph to fade in and out by animating the opacity of the node at the root of the branch", + "name": "opacity", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0", + "description": "Used in combination with the numRenderPasses field of nodes extended from the \\[ArrayGrid\\](https://developer.roku.com/docs/references/scenegraph/abstract-nodes/arraygrid.md\"ArrayGrid\") abstract node class, to optimize rendering of lists and grids. This should never be set to a non-zero value unless you are optimizing the performance of a list or grid rendering by specifying the sequence of rendering operations for sub-elements of the list or grid items, and have set the numRenderPasses field value for the list or grid to a value greater than 1. If the numRenderPasses field value for the list or grid is set to a value greater than 1, you must set this field to a value greater than 0 for all sub-elements of the list or grid items, and not greater than the numRenderPasses field value. If the numRenderPasses field is set to a value greater than 1, and you set this field for a list or grid item sub-element to 0 (the default), or a value greater than the numRenderPasses field value, the list or grid item sub-element will not render", + "name": "renderPass", + "type": "integer" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "disabled", + "description": "renderTracking is set to \"disabled\" when enableRenderTracking is set to false. The following options are only available when enableRenderTracking is set to true:\n\n| Option | Description |\n| --- | --- |\n| `\"none\"` | renderTracking is set to `\"none\"` if **one or more** of these conditions is true: the node's `visible` field is set to `false`the node's `opacity` field is set to `0.0`no `clippingRect` is specified and the node is completely offscreena `clippingRect` is specified and the node lies completely outside that `clippingRect's` coordinates or is completely offscreen |\n| `\"partial\"` | renderTracking is set to `\"partial\"` if **all** of the following conditions are true: the node's `visible` field is set to `true`the node's `opacity` field is greater than `0.0`no `clippingRect` is specified and the node is partially offscreena `clippingRect` is specified and the node lies partially inside the `clippingRect's` coordinates |\n| `\"full\"` | renderTracking is set to `\"full\"` if **all** of the following conditions are true: the node's `visible` field is set to `true`the node's `opacity` field is greater than `0.0`no `clippingRect` is specified and the node is completely onscreena `clippingRect` is specified and the node lies completely inside the `clippingRect's` coordinates |", + "name": "renderTracking", + "type": "option as string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0", + "description": "Defines the rotation angle about the scaleRotateCenter point (in radians) of the node local coordinate system. Positive values specify a counterclockwise rotation, negative values specify a clockwise rotation. For some Roku Player hardware, specifically Roku Players without OpenGL graphics support, only rotations of 0, 90, 180 and 270 degrees (in equivalent radians) are supported. (See \\[Roku Models and Features\\](/docs/specs/hardware.md#current-models \"Roku Models and Features\") for information on OpenGL support)", + "name": "rotation", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[1.0,1.0\\]", + "description": "Defines the scale factor to be applied to the node local coordinate", + "name": "scale", + "type": "vector2d" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[0.0,0.0\\]", + "description": "Describes the location of a point in the node local coordinate that serves as the center of the scale and rotation operations", + "name": "scaleRotateCenter", + "type": "vector2d" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[0.0,0.0\\]", + "description": "Defines the origin of the node local coordinate system relative to its parent node", + "name": "translation", + "type": "vector2d" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "true", + "description": "If true, the node and its children are rendered. If false, the node and its children do not render", + "name": "visible", + "type": "Boolean" + } + ], + "interfaces": [], + "name": "SimpleLabel", + "url": "https://developer.roku.com/docs/references/scenegraph/renderable-nodes/simplelabel.md" + }, + "soundeffect": { + "description": "Extends [**Node**](https://developer.roku.com/docs/references/scenegraph/node.md\n\nThe SoundEffect node class is used to play audio sound effects that can be triggered from events that occur in the UI. Typically, these sound effects are short audio clips, but there is no inherent limit on their length. Currently, up to four simultaneous sounds can be playing at any time, in addition to audio from streaming content and TextToSpeech audio.\n\nFiles can be installed locally as part of the channel package or dynamically downloaded from the network. All files must be WAV (i.e. PCM) format.\n\nFor local files, the convention is to include the WAV files in a directory named \"sounds\".\n\nFor downloaded files, a least-recently-used (LRU) mechanism is used to keep the most recently downloaded/played sounds in temporary storage on the device. If the limits on the maximum number/size of downloaded sounds is exceeded, the least recently used sounds are removed from temporary storage. They will be automatically reloaded the next time the control field is set to \"play\".\n\nA sample demonstrating how to use the SoundEffect node can be found here: [SimpleSoundEffect](https://github.com/rokudev/samples/blob/master/media/SimpleSoundEffect.zip)", + "events": [], + "extends": { + "name": "Node", + "url": "https://developer.roku.com/docs/references/scenegraph/node.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "none", + "description": "Set to control the audio playback. Getting the value of this field returns the most recent value set, or none if no value has been set.\n\n| Option | Effect |\n| --- | --- |\n| none | No effect |\n| play | Start playing the audio. If the audio is already playing, it will be restarted. If the `loadStatus` field is not \"ready\", the sound will not be played and the `state` field will be set to \"notready\". For networked files with the `loadStatus` field set to \"flushed\", setting `control` to \"play\" will automatically trigger a reload of the network file, but will not result in the sound being played, due to the time it takes to download the file again. In this case, the sound can be played once the `loadStatus` field changes from \"flushed\" to \"ready\" |\n| stop | If the audio is playing, stop playing the audio. If the audio is not playing, no effect. |", + "name": "control", + "type": "option string" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "none", + "description": "Indicates the status of the sound file.\n\n| Value | Meaning |\n| --- | --- |\n| none | No file has been requested. |\n| loading | (network files only) The file has been requested and is being downloaded. |\n| ready | The file is ready to be played (i.e. it is on the device and is a valid WAV file). |\n| failed | The file path or URI is incorrect or refers to a file that is not a valid WAV file. |\n| flushed | (network files only) The file was ready, but has been deleted from the LRU cache. Setting the `control` field to play will cause the file to be automatically reloaded, but not be played upon completion. |", + "name": "loadStatus", + "type": "value string" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "none", + "description": "Can be used to track the progress of current state of local and networked sound files When the field value changes to ready, the sound is ready to be played. The possible values are:\n\n| Value | Meaning |\n| --- | --- |\n| none | No current playback state |\n| playing | Audio is currently playing. |\n| stopped | The audio playback was stopped by setting control to \"stop\". The state will also be set to \"stopped\" if audio was playing and the uri is changed. |\n| finished | The audio playback reached the end of the audio |\n| toomanysounds | Control was to \"play\" while there were already the maximum number of SoundEffect sounds playing. Currently, this limit is 4. |\n| notready | The sound file is not on the device. This is set in response to the control field being set to \"play\". For local WAV files included in a channel package, it will be occur if the path to the file is not correct, or if the file is not a valid WAV file. For network-accessed WAV files, this indicates one of these three conditions is true: * The file has been requested, but is not finished downloading. In this case, the `loadStatus` field will be set to \"loading\". * The file request has completed, but the URL is incorrect or the downloaded file is not a valid WAV filed. In this case, the `loadStatus` field will be set to \"failed\" * The file has previously been downloaded, but has been flushed from the LRU cache. In this case, the `loadStatus` field will be set to \"flushed\". |", + "name": "state", + "type": "value string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "", + "description": "Specifies the URI of the WAV file. Sounds included as part of the application package can be referenced using the \\`pkg:/sounds\\` prefix. This may also specify the location of a WAV file on a remote server.", + "name": "uri", + "type": "uri" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "50", + "description": "The volume is a number between 0 and 100 (percentage of full volume). 50 should be used for normal volume.", + "name": "volume", + "type": "integer" + } + ], + "interfaces": [], + "name": "SoundEffect", + "url": "https://developer.roku.com/docs/references/scenegraph/media-playback-nodes/soundeffect.md" + }, + "standarddialog": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [Group](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\"**Group**\")\n\nThe **StandardDialog** node is the base for Roku's pre-built standard message, keyboard, pinpad, and progress dialogs. It can also be used directly with a custom dialog structure built with the **StdDialogItem** nodes.", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "Indicates the index of the button that gained focus when the user moved the focus onto one of the buttons in the button area.", + "name": "buttonFocused", + "type": "int" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "indicates the index of the selected button when the user selects one of the buttons in the button area.", + "name": "buttonSelected", + "type": "int" + }, + { + "accessPermission": "WRITE\\_ONLY", + "default": "false", + "description": "Dismisses the dialog. The dialog is dismissed whenever the close field is set, regardless of whether the field is set to true or false.", + "name": "close", + "type": "boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0f", + "description": "Sets the height of the dialog. If this field is set to greater than 0, and the layout of the dialog for the specified width results in a dialog with a height less than the value of this field, the dialog layout is increased so that the dialog height matches the value of this field. In this case, the button area is moved to the bottom of the dialog and a blank region exists between the content area and the button area.", + "name": "height", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "not set", + "description": "Sets the color palette for the dialog's background, text, buttons, and other elements. By default, no palette is specified; therefore, the dialog inherits the color palette from the nodes higher in the scene graph (typically, from the dialog's \\[Scene\\](https://developer.roku.com/docs/references/scenegraph/scene.md node, which has a \\*\\*palette\\*\\* field that can be used to consistently color the standard dialogs and keyboards in the channel). The RSGPalette color values used by the StandardDialog node are as follows:\n\n| Palette Color Name | Usages |\n| --- | --- |\n| DialogBackgroundColor | Blend color for dialog's background bitmap. |\n| DialogItemColor | Blend color for the following items: * [StdDlgProgressItem's](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-progress-item.md spinner bitmap * [StdDlgDeterminateProgressItem's](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-determinate-progress-item.md graphic |\n| DialogTextColor | Color for the text in the following items: * [StdDlgTextItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md and [StdDlgGraphicItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md if the **namedTextStyle** field is set to \"normal\" or \"bold\". * All [content area items](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md, except for [StdDlgTextItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md and [StdDlgGraphicItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md. * [Title area](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-title-area.mdfields). Unfocused button. |\n| DialogFocusColor | Blend color for the following: * The [button area](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button-area.mdfields) focus bitmap. * The focused scrollbar thumb. |\n| DialogFocusItemColor | Color for the text of the focused button. |\n| DialogSecondaryTextColor | Color for the text of in the following items: * [StdDlgTextItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md and [StdDlgGraphicItem](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md if the **namedTextStyle** field is set to \"secondary\". * Disabled button. |\n| DialogSecondaryItemColor | Color for the following items: * The divider displayed below the title area. * The unfilled portion of the [StdDlgDeterminateProgressItem's](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-determinate-progress-item.md graphic. |\n| DialogInputFieldColor | The blend color for the text edit box background bitmap for keyboards used inside dialogs. |\n| DialogKeyboardColor | The blend color for the keyboard background bitmap for keyboards used inside dialogs |\n| DialogFootprintColor | The blend color for the following items: * The button focus footprint bitmap that is displayed when the [button area](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button-area.mdfields) does not have focus. * Unfocused scrollbar thumb and scrollbar track. |", + "name": "palette", + "type": "RSGPalette node" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "N/A", + "description": "An event that indicates the dialog was dismissed. This event is triggered when one of the following occurs: \\* The \\*\\*close\\*\\* field is set. \\* The Back, Home, or Options key is pressed. \\* Another dialog is displayed.", + "name": "wasClosed", + "type": "event" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0f", + "description": "Sets the width of the dialog: \\* If set to 0, the standard system dialog width is used (1038 for FHD, 692 for HD). If the title or any button text is too wide to fit within the standard width, the dialog width will be automatically increased to show the full title or button text up to a preset maximum (1380 for FHD and 920 for HD). \\* If set to greater than 0, the specified width is used as the overall width of the dialog.", + "name": "width", + "type": "float" + } + ], + "interfaces": [], + "name": "StandardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md" + }, + "standardkeyboarddialog": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StandardDialog](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md\"**Standard Dialog**\")\n\nThe **StandardKeyboardDialog** node enables text and voice entry of strings consisting of alphanumeric characters as well as many commonly used symbols. It is similar to the legacy [KeyboardDialog](https://developer.roku.com/docs/references/scenegraph/dialog-nodes/keyboarddialog.md node, but includes voice entry functionality, which is provided through its internal **DynamicKeyboard** node.\n\n![keyboard-dialog](https://image.roku.com/ZHZscHItMTc2/keyboard-dialog.jpg)", + "events": [], + "extends": { + "name": "StandardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "List of buttons to be displayed in the button area at the bottom of the dialog. Each string in the buttons array adds a new button to the button area. > Minimize the number of buttons in the dialog to ensure that all buttons are visible without the user having to scroll up and down.", + "name": "buttons", + "type": "array of string" + }, + { + "accessPermission": "Access Permission", + "default": "Default", + "description": "Description", + "name": "Field", + "type": "Type" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"generic\"", + "description": "The type of text to be entered. This may be used by the keyboard to modify the voice entry method and to determine when a valid string has been entered. This may be one of the following values: \\* \"email\": letter-by-letter dictation for emails. \\* \"numeric\": letter-by-letter dictation for PIN codes, zip codes, and other numeric input. \\* \"alphanumeric\": letter-by-letter dication for street addresses or other sequences of numbers and letters. \\* \"generic\": Full word input for search queries or other sequences of numbers, letters and symbols. \\* \"password\": letter-by-letter dication for passwords.", + "name": "keyboardDomain", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "One or more blocks of text, which are typically used to describe information about the data to be entered. Each string in the array is displayed as a separate block of text with the standard amount of space left between the blocks. > Minimize the message length to avoid having a scrollbar automatically added to the content area. If multiple strings are specified or any string is too long, the dialog may not be able to fit within the height of the display.", + "name": "message", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The default string to be displayed in the keyboard's text edit box. When the user enters the text, this field is updated with the currently entered string.", + "name": "text", + "type": "string" + }, + { + "accessPermission": "READ", + "default": "The keyboard item's VoiceTextEditBox node", + "description": "The internal VoiceTextEditBox node used by this dialog's internal keyboard. This field should be used only to access the fields of this internal node.", + "name": "textEditBox", + "type": "VoiceTextEditBox node" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The title to be displayed at the top of the dialog.", + "name": "title", + "type": "string" + } + ], + "interfaces": [], + "name": "StandardKeyboardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-keyboard-dialog.md" + }, + "standardmessagedialog": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StandardDialog](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md\"**Standard Dialog**\")\n\nThe **StandardMessageDialog** node is used to displays a message to the user. It is similar to the legacy [Dialog](https://developer.roku.com/docs/references/scenegraph/dialog-nodes/dialog.md node. It may contain the following items (from top to bottom):\n\n* One or more blocks of text at the top.\n* One bulleted / numbered list.\n* One or more blocks of text at the bottom.\n\n![standard-message-dialog](https://image.roku.com/ZHZscHItMTc2/standard-message-dialog.jpg)", + "events": [], + "extends": { + "name": "StandardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "One or more blocks of informational text displayed at the bottom of the dialog's content area. Each string in the array is displayed as a separate block of text with the standard amount of space left between the blocks. > To separate lines of text, add each line as an element in the array. Do not use newline characters.", + "name": "bottomMessage", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "An array of strings displayed as a bulleted or numbered list. The list is displayed in the content area below the message and above the bottom message.", + "name": "bulletText", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"bullet\"", + "description": "If the \\*\\*bulletText\\*\\* field is set, specifies the type of list item delimiter, which may be one of the following: \\* \"bullet\" (this is the default) \\* \"numbered\" \\* \"lettered\" .", + "name": "bulletType", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "List of buttons to be displayed in the button area at the bottom of the dialog. Each string in the buttons array adds a new button to the button area. > Minimize the number of buttons in the dialog to ensure they are all visible when the dialog is displayed.", + "name": "buttons", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "One or more blocks of informational text displayed at the top of the dialog's content area. Each string in the array is displayed as a separate block of text with the standard amount of space left between the blocks. > To separate lines of text, add each line as an element in the array. Do not use newline characters.", + "name": "message", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The title to be displayed at the top of the dialog.", + "name": "title", + "type": "string" + } + ], + "interfaces": [], + "name": "StandardMessageDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-message-dialog.md" + }, + "standardpinpaddialog": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StandardDialog](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md\"**Standard Dialog**\")\n\nThe **StandardPinPadDialog** node enables text and voice entry of numeric characters—typically, short numeric PIN codes. It is similar to the legacy [PinDialog](https://developer.roku.com/docs/references/scenegraph/dialog-nodes/pindialog.md node, but includes additional voice entry of the numeric digits. This additional functionality is provided through the node's internal DynamicPinPad and VoiceTextEditBox nodes.\n\n![pin-pad-dialog](https://image.roku.com/ZHZscHItMTc2/pin-pad-dialog.jpg)", + "events": [], + "extends": { + "name": "StandardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "List of buttons to be displayed in the button area at the bottom of the dialog. Each string in the buttons array adds a new button to the button area. > Minimize the number of buttons in the dialog to ensure that all buttons are visible without the user having to scroll up and down.", + "name": "buttons", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "One or more blocks of text, which are typically used to describe information about the data to be entered. Each string in the array is displayed as a separate block of text with the standard amount of space left between the blocks. > Minimize the message length to avoid having a scrollbar automatically added to the content area. If multiple strings are specified or any string is too long, the dialog may not be able to fit within the height of the display.", + "name": "message", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Displays the entered PIN number in the text edit box. As the user enters each numeric digit, this field is updated with the currently entered value.", + "name": "pin", + "type": "string" + }, + { + "accessPermission": "READ", + "default": "The keyboard item's VoiceTextEditBox node", + "description": "The internal VoiceTextEditBox node used by this dialog's internal keyboard. This field should be used only to access the fields of this internal node > Use the \\*\\*textEditBox.maxTextLength\\*\\* field to limit the length of the pin to be entered.", + "name": "textEditBox", + "type": "VoiceTextEditBox node" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The title to be displayed at the top of the dialog.", + "name": "title", + "type": "string" + } + ], + "interfaces": [], + "name": "StandardPinPadDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-pinpad-dialog.md" + }, + "standardprogressdialog": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StandardDialog](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md\"**Standard Dialog**\")\n\nThe StandardProgressDialog node displays a spinning progress indicator that includes a short progress message to the user. It is similar to the legacy [ProgressDialog](https://developer.roku.com/docs/references/scenegraph/dialog-nodes/progressdialog.md) node.\n\n![progress-dialog-title](https://image.roku.com/ZHZscHItMTc2/progress-dialog-title-v2.jpg)", + "events": [], + "extends": { + "name": "StandardDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "A string to be displayed next to the spinning progress indicator. It typically tells the user why they are waiting. > Minimize the message length.", + "name": "message", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The title to be displayed at the top of the dialog.If no title is specified, the progress dialog will be displayed without a title area and will use the minimum width needed to show the spinning progress indicator and message", + "name": "title", + "type": "string" + } + ], + "interfaces": [], + "name": "StandardProgressDialog", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-progress-dialog.md" + }, + "stddlgareabase": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [Group](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\n\nThe **StdDlgAreaBase** node is the base class and provides the common functionality for the three StandardDialog area nodes: [**StdDlgTitleArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-title-area.md, [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md and [**StdDlgButtonArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button-area.md.", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [], + "interfaces": [], + "name": "StdDlgAreaBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md" + }, + "stddlgbullettextitem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgBulletTextItem** node is used to display a bulleted list of text in the dialog's content area. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![StdDlgBulletTextItem](https://image.roku.com/ZHZscHItMTc2/StdDlgBulletTextItem-v2.jpg)", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "An array of strings displayed as a bulleted or numbered list. The list is displayed in the content area below the message and above the bottom message.", + "name": "bulletText", + "type": "array of string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"bullet\"", + "description": "Specifies the type of list item delimiter, which may be one of the following: \\* \"bullet\" \\* \"numbered\" \\* \"lettered\"", + "name": "bulletType", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgBulletTextItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-bullet-text-item.md" + }, + "stddlgbutton": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [Group](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\"**Group**\")\n\n**StdDlgButton** is the class used for each button in the [button area](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog-framework-overview.mdstructure). The buttons are displayed in the order in which they are listed as children of the [**StdDlgButtonArea** node](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button-area.md. The size and layout of each button are controlled by the StandardDialog layout algorithm. **StdDlgButton** nodes should only be used as children of a **StdDlgButtonArea** node.\n\n![std-dlg-button](https://image.roku.com/ZHZscHItMTc2/std-dlg-button-3.jpg)", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Specifies whether the button can receive focus. If this field is set to true, the button has an inactive appearance and is unable to receive focus.", + "name": "disabled", + "type": "boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The text to be displayed on the button", + "name": "text", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgButton", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button.md" + }, + "stddlgbuttonarea": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgAreaBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md\"**StdDlgAreaBase**\")\n\nThe **StdDlgButtonArea** node is always positioned at the bottom of the [StandardDialog](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md. It contains zero or more child nodes of type [**StdDlgButton**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button.md or a type that extends **StdDlgButton**. Each of the **StdDlgButton** nodes provides an option to perform some task related to the purpose of the dialog. For example, dialogs often have \"Continue\" and \"Cancel\" buttons in the bottom area. The buttons are positioned and sized so that they are arranged vertically in the order in which their **StdDlgButton** child nodes are listed.\n\nA dialog may only have a single button area, and the button area is optional.\n\n![std-dlg-button-area](https://image.roku.com/ZHZscHItMTc2/std-dlg-button-area.jpg)", + "events": [], + "extends": { + "name": "StdDlgAreaBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md" + }, + "fields": [], + "interfaces": [], + "name": "StdDlgButtonArea", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-button-area.md" + }, + "stddlgcontentarea": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgAreaBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md\"**StdDlgAreaBase**\")\n\nThe **StdDlgContentArea** node contains the main body of the dialog. It is positioned between the title area and the button area.\n\nIt contains zero or more child nodes that extend [**StdDlgItemBase**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md (for example, [**StdDlgTextItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md, [**StdDlgProgressItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-progress-item.md, [**StdDlgGraphicItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md, and other dialog building blocks). The layout and position of the [**StdDlgItemBase** nodes](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md are based on the dialog's width; the nodes are arranged vertically from top to bottom in the content area based on the order in which they are listed. The content area should contain only [**StdDlgItemBase** nodes](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md; otherwise, its layout and rendering are undefined.\n\nA dialog may only have a single content area, and the content area is optional.\n\n![content-area](https://image.roku.com/ZHZscHItMTc2/content-area.jpg)", + "events": [], + "extends": { + "name": "StdDlgAreaBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md" + }, + "fields": [], + "interfaces": [], + "name": "StdDlgContentArea", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md" + }, + "stddlgdeterminateprogressitem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgDeterminateProgressItem** node is used to display a progress indicator in the dialog's content area. It provides the percentage of progress that has been completed for a task that takes a limited amount of time. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![std-dlg-determinate-progress-item](https://image.roku.com/ZHZscHItMTc2/std-dlg-determinate-progress-item-2.jpg)", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"0\"", + "description": "Specifies the current completion percentage text and graphic to be displayed (for example \"35%\" with more than a third of the indicator filled). If this is set to a number less than 0 or greater than 100, the progress indicator will display \"0%\" or \"100%\" completion, respectively.", + "name": "percent", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the text to be displayed next to the progress graphic. If the text width does not fit within the width of the content area, the text will wrap onto multiple lines.", + "name": "text", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgDeterminateProgressItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-determinate-progress-item.md" + }, + "stddlggraphicitem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgGraphicItem** node is used to display an image in the dialog's content area with an optional text label displayed to the left, right, above, or below the image. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![std-dlg-graphic-item](https://image.roku.com/ZHZscHItMTc2/std-dlg-graphic-item.jpg)", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"left\"", + "description": "Specifies where to position and align the graphic and its text label, relative to the content area. This may be one of the following values:\n\n| Value | Text Position |\n| --- | --- |\n| left | The graphic is left-aligned in the content area. The text label is positioned horizontally to the right of the graphic, and centered vertically. |\n| right | The graphic is right-aligned in the content area. The text label is positioned horizontally to the left of the graphic, and centered vertically. |\n| center\\_below | The graphic and text label are centered horizontally in the content area. The graphic is positioned below the text label. |\n| center\\_above | The graphic and text label are centered horizontally in the content area. The graphic is positioned above the text label. |", + "name": "graphicAlign", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0", + "description": "The image height to be used instead of the image's actual height.", + "name": "graphicHeight", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The URI of the image to be displayed.", + "name": "graphicUri", + "type": "uri" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0", + "description": "The image width to be used instead of the image's actual width.", + "name": "graphicWidth", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the text to be displayed next to the graphic. If the text width does not fit within the width of the content area, the text will wrap onto multiple lines.", + "name": "text", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgGraphicItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md" + }, + "stddlgitembase": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [Group](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md)\n\n**StdDlgItemBase** is the base class for all the content area items. It provides the common functionality for all StdDlg\\[_x_\\]Item nodes (for example, [**StdDlgBulletTextItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-bullet-text-item.md, [**StdDlgTextItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md, [**StdDlgKeyboardItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-keyboard-item.md, [**StdDlgProgressItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-progress-item.md, [**StdDlgGraphicItem**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-graphic-item.md, and the other dialog building block nodes).", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.comhttps://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Indicates whether the item can be scrolled vertically by the user. The StandardDialog layout algorithm reduces the height of a scrollable item as needed if the overall height of the dialog is too large to fit on the display.", + "name": "scrollable", + "type": "boolean" + } + ], + "interfaces": [], + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "stddlgkeyboarditem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgKeyboardItem** node is used to display a keyboard or PINpad in the dialog's content area. It provides text and voice entry of strings containing alphanumeric characters and symbols or numeric digits. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![std-dlg-keyboard-item](https://image.roku.com/ZHZscHItMTc2/std-dlg-keyboard-item.jpg)", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"unspecified\"", + "description": "Specifies the type of keyboard to be displayed: \\* \"unspecified\": no keyboard is displayed. \\* \"keyboard\": A \\[\\*\\*DynamicKeyboard\\*\\*\\](https://developer.roku.com/docs/references/scenegraph/dynamic-voice-keyboard-nodes/dynamic-keyboard.md node is displayed. \\* \"pinpad\": A \\[\\*\\*DynamicPinPad\\*\\*\\](https://developer.roku.com/docs/references/scenegraph/dynamic-voice-keyboard-nodes/dynamic-pinpad.md node is displayed.", + "name": "keyLayout", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The default string to be displayed in the keyboard's text edit box. When the user enters the text, this field is updated with the currently entered string.", + "name": "text", + "type": "string" + }, + { + "accessPermission": "READ", + "default": "The [**VoiceTextEditBox**](/docs/references/scenegraph/dynamic-voice-keyboard-nodes/voice-text-edit-box.md) associated with the keyboard", + "description": "The internal \\[\\*\\*VoiceTextEditBox\\*\\* node\\](https://developer.roku.com/docs/references/scenegraph/dynamic-voice-keyboard-nodes/voice-text-edit-box.md used by this dialog's internal keyboard. This field should be used only to access the fields of this internal node.", + "name": "textEditBox", + "type": "VoiceTextEditBox node" + } + ], + "interfaces": [], + "name": "StdDlgKeyboardItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-keyboard-item.md" + }, + "stddlgprogressitem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgProgressItem** node is used to display a spinning progress indicator in the dialog's content area. It provides the status of a task that takes an indeterminate amount of time. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![std-dlg-progress-item](https://image.roku.com/ZHZscHItMTc2/std-dlg-progress-item.jpg)", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the text to be displayed next to the progress graphic. If the text width does not fit within the width of the content area, the text will wrap onto multiple lines.", + "name": "text", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgProgressItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-progress-item.md" + }, + "stddlgtextitem": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgItemBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md\"**StdDlgItemBase**\")\n\nThe **StdDlgTextItem** node is used to display a block of text. It should only be used as a child of a [**StdDlgContentArea**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-content-area.md node.\n\n![StdDlgTextItem](https://image.roku.com/ZHZscHItMTc2/std-dlg-text-item.jpg)\n\n> To separate lines of text, use multiple **StdDlgTextItem** nodes. Do not use newline characters.", + "events": [], + "extends": { + "name": "StdDlgItemBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-item-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the string to be spoken when the audio guide reads the text item. By default, the audio guide reads the string specified in the \\*\\*text\\*\\* field.", + "name": "audioGuideText", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"normal\"", + "description": "Specifies a named style to be used for the displayed text's color and font. The supported styles include:\n\n| Style Name | Palette Color | Font |\n| --- | --- | --- |\n| \"normal\" | DialogTextColor | SmallSystemFont |\n| \"secondary\" | DialogSecondaryTextColor | SmallestSystemFont |\n| \"bold\" | DialogTextColor | SmallBoldSystemFont |", + "name": "namedTextStyle", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the text to be displayed. If the text width does not fit within the width of the content area, the text will wrap onto multiple lines.", + "name": "text", + "type": "string" + } + ], + "interfaces": [], + "name": "StdDlgTextItem", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-text-item.md" + }, + "stddlgtitlearea": { + "availableSince": "10.0", + "description": "_Available since Roku OS 10.0_\n\nExtends [StdDlgAreaBase](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md\"**StdDlgAreaBase**\")\n\nThe **StdDlgTitleArea** node contains the dialog's title information, which is always displayed at the top of the dialog. The title area may also include optional icons that appear left or right justified. The **StdDlgTitleArea** should only be used as a child node of a [**StandardDialog**](https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/standard-dialog.md.\n\nA dialog may a single title area, and the title area is optional (but is typically used in nearly all cases)\n\n![title-area-icon](https://image.roku.com/ZHZscHItMTc2/title-area-icon.jpg)", + "events": [], + "extends": { + "name": "StdDlgAreaBase", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-area-base.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies a bitmap to be displayed at the left edge of the dialog's title area (to the left of dialog's primary title).", + "name": "primaryIcon", + "type": "URL" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0f", + "description": "Adjusts the vertical position of the primary icon relative to the baseline of the dialog's primary title. By default, the bottom of the primary icon is aligned with the primary title's baseline.", + "name": "primaryIconVertOffset", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the title to be displayed in the dialog's title area.", + "name": "primaryTitle", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies a bitmap to be displayed at the right edge of the dialog's title area.", + "name": "secondaryIcon", + "type": "URL" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0f", + "description": "Adjusts the vertical position of the secondary icon relative to the baseline of the dialog's primary title. By default, the bottom of the secondary icon is aligned with the primary title's baseline.", + "name": "secondaryIconVertOffset", + "type": "float" + } + ], + "interfaces": [], + "name": "StdDlgTitleArea", + "url": "https://developer.roku.com/docs/references/scenegraph/standard-dialog-framework-nodes/std-dlg-title-area.md" + }, + "targetgroup": { + "description": "Extends [**Group**](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\"**Group**\")\n\nThe TargetGroup node class associates a set of rectangular regions that children of the group will occupy. Like MarkupList, the TargetGroup has a content field containing the data for each item and an itemComponentName field that specifies an RSG component that will be used to render a content item. It also has a targetSet field that contains a [TargetSet](https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/targetset.md\"TargetSet\") that define a set of rectangular targets where children of the TargetGroup will be rendered.\n\nThe TargetGroup node is typically used to create a scrolling list (or row) of items where the focused item occupies more space than the other items.\n\nFor example, a TargetGroup could be used to create a full screen vertical scrolling list of item where the focused item is larger than the other items in the list. As the list items scroll, the appearance of the item moving into the focus region would be dynamically adjusted to fill the larger focus region. Simultaneously, the appearance of the item leaving the focus region would be dynamically adjust to return to the unfocused size. To set up this use case, you might set the targetSet field to a TargetSet node that specifies nine rectangles. The first rectangle would be specified to have the width and height of an unfocused item and be positioned so that it's bottom is above the top of the screen. The last rectangle would be specified to have the width and height of an unfocused item and be positioned so that's top is below the bottom of the screen. The remaining seven rectangles would define the rectangular regions of the onscreen items. Suppose the design calls for the focus item to be centered vertically at the center of the screen. To do that, you would specify the 5th rectangle to be larger than the other eight and position it so that it is centered vertically, you would specify the remaining rectangles to form a column of rectangular regions where the top three and bottom three visible items would be located.\n\nThe second step of setting up this use case would be to implement an RSG component that will be used to render each item. The TargetGroup node manages the creation of the items for the visible components, associates each with a ContentNode, and updates fields of the item component with information such as the current width and height of the item and the focus status of the item.\n\nThe TargetGroup's jumpToItem field is set to identify which content item is to be located at the TargetSet's targetRects field target rectangle identified by the TargetSet's focusindex field.\n\nThe final step of setting up this use case would be to create a VerticalList component that extends TargetGroup, sets up the TargetGroup's TargetSet node, and as the user presses up and down buttons on the remote, sets the TargetGroup's animateToTargetItem field to the prior or next index. Setting the animateToTargetItem field causes the displayed items to smoothly animate from their current target region to another target region, such that the specified index ends up at the TargetSet's target rectangle that is identified by the TargetSet's focusIndex field.\n\nThe above use case specifies the most common use case for the TargetGroup node, but only hints at the possible uses. For example, you could create your own RSG components with various custom behaviors. There might include:\n\n* A list where all the items are small when the list does not have the focus, but when the list receives the focus, all of the items smoothly adjust their size and position so that the focus item is largest, the items on either side of the focus item are slightly larger than the unfocused size and the remaining items remain the same size as the unfocused items. To do this, you would create two TargetSet's in your RSG component, one that defines the regions when the list is unfocused and one that defines the regions when the list is focused. Initially, the TargetGroup's targetSet field would be set to the unfocused TargetSet node. Then, when the list is focused, the targetSet's animateToTargetSet field would be set to the focused TargetSet node, causing all of the target regions to smoothly animate to their new size and position, taking along the associated item component's with them.\n* A horizontal scrolling list of items where the focused region floats across the screen as the user presses left/right until the focus region reaches the edge of the display, at which point the focus region remains stationary and the items scroll left or right. This would require the use of several TargetSet nodes (one for each possible position of the focus region). Initally, the TargetGroup's targetSet field would be set to one of these TargetSet's. Then while the focus region is not at one of the edges, key presses would set the animateToTargetSet field to animate the focus region to its next location. Once the focus region reaches an edge, another key press in the same direction would set the animateToTargetIndex field to cause the items to scroll so that the next content item occupies the focus region.\n* A list where when an item is selected, all of the items fly off the screen while the selected item zooms up and moves to the center of the screen. To set up this use case, you would specify a TargetSet for when the list items are onscreen and another TargetSet for the onscreen location of the focused item and the offscreen locations where each items will disappear.\n* A circular arrangement of a fixed number of items with the item at the 6 o'clock position being larger and having the focus. Note that in this case, no offscreen targets would be specified.", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "WRITE\\_ONLY", + "default": "0", + "description": "When set to a valid item index, causes the group to quickly scroll so that the specified index moves into the to the target region specified by the TargetSet's focusIndex", + "name": "animateToItem", + "type": "integer" + }, + { + "accessPermission": "WRITE\\_ONLY", + "default": "invalid", + "description": "When set to a valid TargetSet, causes the group to quickly animate so that the target regions of the initial TargetSet node are smoothly interpolated to the corresponding target regions of the new TargetSet node. If the two TargetSet's focusIndex fields are different, the focusIndex is also animated from the old to the new value", + "name": "animateToTargetSet", + "type": "TargetSet" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "none", + "description": "Specifies the content for the group. See \\[Data Bindings\\](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.mddata-bindings \"Data Bindings\") below for more details", + "name": "content", + "type": "ContentNode" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "\\-1.0", + "description": "As the TargetGroup animation occurs, this field is constantly updated to represent the index of the ContentNode currently occupying the focus target region. When currFocusItemIndex is an integer value, the specified ContentNode occupies the focus target. When currFocusItemIndex has a fractional part, the value indicates that an animation is in process. For example, a value of 5.7 would indicate that items 5 and 6 are currently overlapping the focus region, with item 6 occupying 70% and item 5 the other 30%", + "name": "currFocusItemIndex", + "type": "float" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "invalid", + "description": "As the TargetGroup animation occurs that is initiated by setting the animateToTargetSet field, currTargetSet contains the current values of the target regions as the animation proceeds from the initial TargetSet's targets to the new TargetSet's targets", + "name": "currTargetSet", + "type": "TargetSet" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0", + "description": "For TargetSet's that do not specify a focusIndex, this value will be used as the index of the TargetSet where the focused item is located. If a TargetSet specifies any value for the focusIndex, that value will be used instead of defaultTargetSetFocusIndex", + "name": "defaultTargetSetFocusIndex", + "type": "int" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.3 seconds", + "description": "Specifies the time, in seconds, to perform the animation when the animateToItem or animateToTargetSet fields are set", + "name": "duration", + "type": "Time" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "", + "description": "Specifies the name of a XML component for the group items. An instance of this component is created on demand for each visible item of the group. The XML component must define a specific interface as detailed in \\[TargetGroup XML Component\\](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.mdtargetgroup-xml-component \"TargetGroup XML Component\") below", + "name": "itemComponentName", + "type": "string" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "When a group item gains the key focus, set to the index of the focused item", + "name": "itemFocused", + "type": "integer" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "When a group item is selected, set to the index of the selected item", + "name": "itemSelected", + "type": "integer" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "When a group item loses the key focus, set to the index of the unfocused item", + "name": "itemUnfocused", + "type": "integer" + }, + { + "accessPermission": "WRITE\\_ONLY", + "default": "0", + "description": "When set to a valid item index, causes the group to immediately update so that the specified index moves to the target region specified by the TargetSet's focusIndex", + "name": "jumpToItem", + "type": "integer" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Specifies whether the current target rectangles (as defined in the read-only currTargetSet field's TargetSet) are drawn or not. Typically this would only be set to true while debugging a channel, although in some use cases its possible that you might want to display the current target rectangles. The rectangles are drawn using the color in the targetSet's TargetSet node's color field", + "name": "showTargetRects", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "invalid", + "description": "Specifies the TargetSet to use to define the target regions of the items in the group. When set or modified, the target regions are immediately adjusted to use the new values", + "name": "targetSet", + "type": "TargetSet" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Specifies whether the content items wraparound at the end of the TargetGroup to fill all of the targets rectangles", + "name": "wrap", + "type": "Boolean" + } + ], + "interfaces": [], + "name": "TargetGroup", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.md" + }, + "targetlist": { + "description": "Extends [**TargetGroup**](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.md\n\nThe TargetList node class adds useful functionality to the TargetGroup node by making is easy to set up lists and rows of items with limited amounts of scripting required. In particular, TargetList provides a built-in focused/unfocused transition, as well as a simple way to implement various focus management policies (i.e. fixed focus, floating focus, etc.). It also provides default key handling for navigating the list or row.", + "events": [], + "extends": { + "name": "TargetGroup", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"down\"", + "description": "Specifies which remote button will move the focus forward. For vertical lists, this will typically be set to \"down\". For horizontal rows, this will typically be set to \"right\".", + "name": "advanceKey", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "Specifies one or more TargetSet's to be used when the TargetList has the focus. If a single TargetSet is specified, focus will stay fixed on the targetRect of that TargetSet that corresponds to the TargetGroup's focus index. The focus index will come from the TargetSet if explicitly specified or from the TargetGroup's defaultTargetSetFocusIndex field if not. If focusedTargetSet includes more than one TargetSet node, that defines a sequence of TargetSet's that will be advanced through as the user presses the advance or reverse key. When advancing, the focus floats from one TargetSet's to the next TargetSet in the array until the last element of the focusedTargetSet is reached, at which point the focus is fixed to the last element and the items begin to scroll. When reversing, the focus floats from one TargetSet to the previous TargetSet in the array until the first element of the focusedTargetSet is reached, at which point the focus is fixed to the first element and the items begin to scroll. See above for more discussion of setting up fixed and floating focus use cases.", + "name": "focusedTargetSet", + "type": "array of TargetSet nodes" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"up\"", + "description": "Specifies which remote button will move the focus backward. For vertical lists, this will typically be set to \"up\". For horizontal rows, this will typically be set to \"left\".", + "name": "reverseKey", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "invalid", + "description": "Specifies the TargetSet to be used when the TargetList does not have the focus.", + "name": "unfocusedTargetSet", + "type": "TargetSet" + } + ], + "interfaces": [], + "name": "TargetList", + "url": "https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/targetlist.md" + }, + "targetset": { + "description": "Extends [**Node**](https://developer.roku.com/docs/references/scenegraph/node.md\n\nThe TargetSet node class is used to specify a set of target regions where items in a [TargetGroup](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/targetgroup.md node are rendered. This information includes an array of rectangles that is used to define the location and size of a region that will be occupied by an item in the TargetGroup as well as an optional index that identifies one rectangle in the array to be treated as the region where the item with focus is located.", + "events": [], + "extends": { + "name": "Node", + "url": "https://developer.roku.com/docs/references/scenegraph/node.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "0xFFFFFF80", + "description": "If the TargetGroup using this TargetSet has its showTargetRects field set to true, the target rectangles of the current TargetSet will be drawn using the specified color. Drawing the TargetSet's target rectangles is generally only done when debugging an application.", + "name": "color", + "type": "Color" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\-1", + "description": "Identifies the index of an element of the targetRects array that will be treated as the region occupied by the focus item. The default of of -1 indicates that the TargetGroup's current focus index will not be changed when the TargetGroup is set to use the TargetSet to define its target regions.", + "name": "focusIndex", + "type": "integer" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "Specifies an array of rectangles that define the target regions used by a TargetGroup node. To specify a rectangle, you can either specify a associative array with x, y, width and height elements or an array of 4 numeric values. For example, you could specify an array of two rectangles like this: \\\\\\[ \\\\\\[ x:10, y:5, width: 200, height:150 \\\\\\], \\\\\\[ x:10, y:160, width: 200, height:150 \\\\\\] \\\\\\] Alternately, you could specify the same array like this: \\\\\\[ \\\\\\[ 10, 5, 200, 150 \\\\\\], \\\\\\[ 10, 160, 200, 150 \\\\\\] \\\\\\]", + "name": "targetRects", + "type": "array of rectangles" + } + ], + "interfaces": [], + "name": "TargetSet", + "url": "https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/targetset.md" + }, + "task": { + "description": "Extends [**Node**](https://developer.roku.com/docs/references/scenegraph/node.md\n\nThe Task node class allows you to specify a function to be spawned in a different thread, and run asynchronously with respect to both the scene rendering thread and the main application thread. A Task node also allows you to run functions that cannot be run in SceneGraph node or component, typically BrightScript code functions involving operations such as reading data from servers and file system manipulation. (You also cannot, and should not, run functions in a SceneGraph application for operations that are functionally the same as SceneGraph nodes and components, such as playing videos.) A list of all the BrightScript functions and components that cannot be used in SceneGraph applications or can only be used in a Task node can be found in [BrightScript Support](/docs/developer-program/core-concepts/scenegraph-brightscript/brightscript-support.md \"BrightScript Support\").\n\nA Task node is typically used to read data from a server to create a ContentNode node to configure a SceneGraph node or component (see [ContentNode](https://developer.roku.com/docs/references/scenegraph/control-nodes/contentnode.md\"ContentNode\")). A Task node used for this purpose can be thought of as a content reader. Since ContentNode nodes are required to configure many components rendered in a scene, such as lists, panels, and grids, and you will generally want to read the data for those types of nodes from your server, you should create a Task node as a content reader for each of those components that you use in your scene.\n\nThe Task node class was designed with three general development use cases:\n\n* A new Task node object is created for each asynchronous operation. The input data needed for the operation is set in the Task node object [](https://developer.roku.com/docs/references/scenegraph/xml-elements/interface.md\"\") fields in the render thread, along with an observer of the output field data, and the Task node control field is set to RUN. After the output data is returned to the render thread, the Task node object is not used again.\n* A Task node object is used multiple times for several identical asynchronous operations. In this case, the input data for each operation is set in the existing Task node object, with another observer for the output field data, and the Task node control field is again set to RUN. This may be more efficient than creating a new Task node object for each of the identical operations.\n* A Task node observes its input fields using the port form of the ifSGNodeField [observeField()](https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.mdobservefieldfieldname-as-string-functionname-as-string-as-boolean \"observeField()\") method, and returns output data with each field change. In this case, the Task node acts like a continuous server.\n\nSince Task nodes launch asynchronous threads, and have no provisions for locks and mutexes, you must be careful to avoid race, deadlock, and other asynchronous thread errors. Here are a few tips for using Task nodes:\n\n* Avoid accessing files which must be persistent before thread completion, to avoid a subsequent Task node or other thread access of the same file before the thread completes. It is easier and safer to use a dynamically-created string or other data object to hold temporary thread data to avoid having a subsequent or existing thread overwrite and corrupt the data.\n* Be very careful if you access any object in a Task node that may exist in another thread. It is better to completely separate all objects in any other possible thread from the Task node thread by setting the fields of the Task node with copies of the minimum amount of data needed to run the thread.\n* In the Task node init() function, perform the minimum required amount of initialization of the Task node and any included thread functions. If you intend to trigger an asynchronous task based on a Task node input field change, in many cases, you should only set up the observer for the field in init().\n* Use the port form of the ifSGNodeField [observeField()](https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.mdobservefieldfieldname-as-string-functionname-as-string-as-boolean \"observeField()\") rather than the onChange attribute. This will avoid triggering the thread in response to a render thread event before the Task node observers are set up.\n* It is more efficient to use a persistent Task node that is triggered by an field change than to create a new Task node every time a particular asynchronous thread is required. If needed, you can communicate that the particular asynchronous thread is no longer required through an field as well, either through the triggering field, or a special field used for control of the Task node.\n* You can use a single Task node object to run any number of different asynchronous threads by setting the functionName field to the Task node function you want before setting the control field to RUN. If you do not use the input data fields to trigger running the thread, this is equivalent to calling an asynchronous function, and passing the input data fields as arguments to the function. The output data fields can likewise be considered as the return value of a asynchronous function call, but to avoid blocking you must observe the fields, or the state field, as a callback event to handle the results in the calling thread.\n\nAlso review \"[SceneGraph threads](/docs/developer-program/core-concepts/threads.md \"SceneGraph\")\" for in-depth information on using Task nodes most efficiently.", + "events": [], + "extends": { + "name": "Node", + "url": "https://developer.roku.com/docs/references/scenegraph/node.md" + }, + "fields": [ + { + "accessPermission": "WRITE\\_ONLY", + "default": "init", + "description": "Requests a change in the run state of the spawned task. The valid options are the same as for the state field, but case-insensitive (i.e. can set \"RUN\" or \"run\")", + "name": "control", + "type": "option string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "The name of the function in the Task node component to be executed when the state field changes to RUN. The function must be declared within the scope of the Task node component", + "name": "functionName", + "type": "string" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "init", + "description": "Inquires about the run state of the spawned task. Note that the values are in lowercase: \"init\", \"stop\", \"run\", \"done\"", + "name": "state", + "type": "value string" + } + ], + "interfaces": [], + "name": "Task", + "url": "https://developer.roku.com/docs/references/scenegraph/control-nodes/task.md" + }, + "texteditbox": { + "description": "Extends [**Group**](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\"**Group**\")\n\nThe **TextEditBox** node class is intended to display a string of characters as they are typed. When focused, it displays a flashing cursor to indicate the text insertion position.\n\n**TextEditBox** nodes are automatically included in the [**Keyboard**](https://developer.roku.com/docs/references/scenegraph/widget-nodes/keyboard.md\"**Keyboard**\") and [**MiniKeyboard**](https://developer.roku.com/docs/references/scenegraph/widget-nodes/minikeyboard.md\"**MiniKeyboard**\") node classes.\n\nThe default appearance of the **TextEditBox** is very transparent, allowing it to pick up most of its color from what is rendered underneath it. The appearance can be customized by changing the backgroundUri and other fields.", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "Read-Write", + "default": "false", + "description": "When active is set to true, the cursor is displayed. When set to false, the cursor is hidden. When used internal to the \\*\\*Keyboard\\*\\* and \\*\\*MiniKeyboard\\*\\* nodes, those nodes set this field to true when the keyboard has focus, and false when it does not.", + "name": "active", + "type": "boolean" + }, + { + "accessPermission": "Read-Write", + "default": "\"\"", + "description": "Specifies the URI of the image rendered as the background of the \\*\\*TextEditBox\\*\\* node.", + "name": "backgroundUri", + "type": "string" + }, + { + "accessPermission": "Read-Write", + "default": "true", + "description": "When clearOnDownKey is set to true, the textEditBox erases all the characters when down key is pressed (focus does not move down). When set to false, the characters are not erased and focus moves down.", + "name": "clearOnDownKey", + "type": "boolean" + }, + { + "accessPermission": "Read-Write", + "default": "0", + "description": "By default, this is set to the length of the text field, indicating that the next character to be entered should be appended at the end of the string. When used internal to the \\*\\*Keyboard\\*\\* and \\*\\*MiniKeyboard\\*\\* nodes, those nodes use this field to move the text insertion point.", + "name": "cursorPosition", + "type": "integer" + }, + { + "accessPermission": "Read-Write", + "default": "\"\"", + "description": "Specifies a string to be displayed if the length of the text field string is zero. The typical usage of this field is to prompt the user about what to enter (such as, \"Enter your WiFi password\").", + "name": "hintText", + "type": "string" + }, + { + "accessPermission": "Read-Write", + "default": "0xffffffff", + "description": "Specifies the color of the hint text string.", + "name": "hintTextColor", + "type": "color" + }, + { + "accessPermission": "Read-Write", + "default": "15", + "description": "Specifies the maximum length of the string that can be displayed. When used internal to the \\*\\*Keyboard\\*\\* node, maxTextLength is initialized to 75. When used in the \\*\\*MiniKeyboard\\*\\* node, maxTextLength is initialized to 25.", + "name": "maxTextLength", + "type": "integer" + }, + { + "accessPermission": "Read-Write", + "default": "false", + "description": "When set to true, the characters entered are briefly displayed, then replaced with an asterisk. When set to false, the characters entered are always displayed. When used internal to the \\*\\*Keyboard\\*\\* and \\*\\*MiniKeyboard\\*\\* nodes, you can access the keyboard \\*\\*textEditBox\\*\\* field to set its secureMode field. For example: \\`myKeyboard.textEditBox.secureMode = true\\`", + "name": "secureMode", + "type": "boolean" + }, + { + "accessPermission": "Read-Write", + "default": "\"\"", + "description": "Contains the string of characters being displayed.", + "name": "text", + "type": "string" + }, + { + "accessPermission": "Read-Write", + "default": "0xffffffff", + "description": "Specifies the color of the text string displayed.", + "name": "textColor", + "type": "color" + }, + { + "accessPermission": "Read-Write", + "default": "\\-1.0", + "description": "Specifies the width of the \\*\\*TextEditBox\\*\\* node. When used internal to the \\*\\*Keyboard\\*\\* and \\*\\*MiniKeyboard\\*\\* nodes, those nodes set this field to match the width of the keyboard.", + "name": "width", + "type": "float" + } + ], + "interfaces": [], + "name": "TextEditBox", + "url": "https://developer.roku.com/docs/references/scenegraph/widget-nodes/texteditbox.md" + }, + "timegrid": { + "description": "OTT providers can use the TimeGrid node to implement an Electronic Program Guide (EPG) in their channels. In an EPG, channels are represented as horizontal rows, one for each channel. Each row has a channel name on the left, and a set of programs airing on that channel to the right. The size of each program depends on its duration. One of these programs has a remote control focus highlight indicator on it, and this highlight can be moved around using the remote control (as long as the TimeGrid node has remote control focus).\n\nThe TimeGrid node also features an alternative Now/Next view that lists only the programs currently airing and airing next, with their respective start times. See [Now/Next mode](https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/timegrid.mdnownext-mode) for more information.\n\n![time grid](https://image.roku.com/ZHZscHItMTc2/epg-standard.jpg \"time grid\")", + "events": [], + "fields": [], + "interfaces": [], + "name": "TimeGrid", + "url": "https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/timegrid.md" + }, + "timer": { + "description": "Extends [**Node**](https://developer.roku.com/docs/references/scenegraph/node.md\n\nThe Timer node class generates an observable event after a specified amount of time has elapsed.", + "events": [], + "extends": { + "name": "Node", + "url": "https://developer.roku.com/docs/references/scenegraph/node.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "none", + "description": "Used to control the operation of the Timer node. Recognized values include:\n\n| Value | Effect |\n| --- | --- |\n| none | No effect |\n| start | Starts the **Timer** node operation |\n| stop | Stops a running **Timer** node |", + "name": "control", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "1", + "description": "Specifies the time in seconds before the Timer node fires after the control field value is set to start. To specify time values down to millisecond granularity, use a float type (0.001 equals one millisecond)", + "name": "duration", + "type": "time" + }, + { + "accessPermission": "OBSERVE\\_ONLY", + "default": "N/A", + "description": "Triggers observer callback functions when the Timer node fires. Please note that the timer observer callback executes on the render thread", + "name": "fire", + "type": "Event" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "If set to true, the Timer node fires repeatedly, each time the specified duration field value elapses. If set to false, the Timer node only fires once until restarted", + "name": "repeat", + "type": "Boolean" + } + ], + "interfaces": [], + "name": "Timer", + "url": "https://developer.roku.com/docs/references/scenegraph/control-nodes/timer.md" + }, + "vector2dfieldinterpolator": { + "description": "Extends [**Node**](https://developer.roku.com/docs/references/scenegraph/node.md\n\nVector2DFieldInterpolator specifies a keyframe animation sequence to be applied to a pair Vector2D field of a node. Most typically, this is used to animate the (x,y) coordinates of a node's translation field.\n\nAll field interpolators include a set of key/keyValue pairs that define a keyframe of the animation. Field interpolators are generally used as children of an Animation node. As the animation progresses, it sets the fraction field of its field interpolators to a value between 0 and 1, indicating the percentage of the Animation's progress. The keyframes of the interpolator include a \"key\", the percentage where the keyframe should occur, and a \"keyValue\", the value that the field should have at that percentage.\n\nFor example, if a Vector2DFieldInterpolator had three keyframes, (0.0, \\[0.0, 0.0\\]), (0.4, \\[500.0, 0.0\\]) and (1.0, \\[500, 200.0\\]), then when the interpolator's fraction field was 0.0 (i.e. 0%), the field would be set to \\[0.0, 0.0\\]. When fraction was 0.4 (i.e. 40%), the field would be set to \\[500.0, 0.0\\]. When fraction was 1.0 (i.e. 100%), the field would be set to \\[500.0, 200.0\\].\n\nFor values of fraction between 0.0 and 0.4 (e.g. 0.2 or 20%), the field value is determined by linearly interpolating the keyValues for the first two keyframes. In this case, since the key of 0.2 is halfway between the key at 0.0 and the key at 0.4, the field would be set to \\[250.0, 0.0\\] (halfway between the point \\[0.0, 0.0\\] and \\[200.0, 0.0\\]. Similarly, when fraction is between the second and third keys (i.e. between 0.4 and 1.0), the field value is determined by linearly interpolating the keyValues of the second and third keyframes.\n\nFor this example, if the field being interpolated were the translation field of a Poster node parented to the Scene node, the Poster would originally be positioned with its top/left corner at the upper, left corner of the screen. As the animation proceeded from 0% to 40% complete, the Poster would slide horizontally to the right until it's top/left corner was at x=500.0, y=0.0. As the animation continued from 40% to 100% complete, the Poster would slide vertically down until its top/left corner was at x=500.0, y=200.0.\n\nIf the first keyframe has a key percentage greater than zero, then the field value will be equal to the keyValue of the first keyframe until fraction reaches the first keyframe's key percentage. Similarly, if the last keyframe has a key percentage less than one, the field value will be set to the keyValue of the last keyframe from when fraction equals the the last keyframe's key percentage and will not change as fraction increases from that value to 1.0.\n\n> While linearly interpolation is used to compute the keyValue's for fraction values between successive keys, non-linear easing functions may be applied to the fraction values computed by the Animation node, so the overall animation may vary in speed.", + "events": [], + "extends": { + "name": "Node", + "url": "https://developer.roku.com/docs/references/scenegraph/node.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "\"\"", + "description": "Specifies the field to interpolate. The string should contain the ID of a node in the scene and the name of a field of that node, separated by a dot \".\". For example, \"title.width\" would indicate that the interpolator should be applied to the width field of a node whose id field was \"title\". The specified field must be of type float", + "name": "fieldToInterp", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "0.0", + "description": "Specifies the percentage to be used to compute a value for the field", + "name": "fraction", + "type": "float" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "Specifies the key percentages for the interpolator's keyframes. Each key percentage should be a unique value from 0 to 1 indicating the percentage of the animation where the keyValue should occur. Behavior is undefined if the number of values in the key field does not match the number of values in the keyValue field", + "name": "key", + "type": "array of float's" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\[ \\]", + "description": "Specifies the key values or the interpolator's keyframes. Each value in the keyValue array corresponds to a value in the key field's array. The interpolator's behavior is undefined if the number of values in the key field does not match the number of values in the keyValue field", + "name": "keyValue", + "type": "array of float's" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Enables animation to be played in reverse.", + "name": "reverse", + "type": "boolean" + } + ], + "interfaces": [], + "name": "Vector2DFieldInterpolator", + "url": "https://developer.roku.com/docs/references/scenegraph/animation-nodes/vector2dfieldinterpolator.md" + }, + "video": { + "description": "Extends [**Group**](https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md\n\nThe Video node class provides a controlled play of live or VOD video.\n\nThe Video node includes a wide variety of internal nodes to support trick play, playback buffering indicators, and so forth. Playback buffering indicators, to indicate buffering before initial playback as well as re-buffering, use an internal instance of a ProgressBar node. For trick play, an internal instance of a TrickPlayBar node is provided. For display of BIF images for DVD-like chapter selection, an internal instance of a BIFDisplay node is provided.\n\nStarting from Roku OS 8, the behavior of the Roku system overlay is such that the system overlay now slides in whenever the \\* button is pressed, the Video node is in focus, and the app does not have its OnKeyEvent() handler fired. When the Video node is not in focus, the system overlay does not slide in and the OnKeyEvent() handler is fired.", + "events": [], + "extends": { + "name": "Group", + "url": "https://developer.roku.com/docs/references/scenegraph/layout-group-nodes/group.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "NULL", + "description": "The ContentNode node with the \\[Content Meta-Data\\](/docs/developer-program/getting-started/architecture/content-metadata.md) for the video, or a video playlist (a sequence of videos) to be played. If a video playlist is to be played, the children of this ContentNode node comprise the playlist, and each ContentNode child must have all attributes required to play that video. For example, if the videos \"A\" and \"B\" are to be played, three ContentNode nodes must be created: the parent ContentNode (which is largely ignored), one ContentNode child for \"A,\" and one ContentNode child for \"B.\" The parent node is set into this content field, and when video playback is started, all of its children will be played in sequence. Any changes made to the playlist after playback has started are ignored. See the \\`contentIsPlaylist\\` and \\`contentIndex\\` fields, for more information on playlists.", + "name": "content", + "type": "ContentNode" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "\\-1", + "description": "The index of the video in the video playlist that is currently playing. Generally, you would only want to check this field if video playlists are enabled (by setting the \\`contentIsPlaylist\\` field to true), but it is set to 0 when a single video is playing, and video playlists are not enabled.", + "name": "contentIndex", + "type": "integer" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "If set to true, enables video playlists (a sequence of videos to be played). See the \\`content\\` and \\`contentIndex\\` field for more information on playlists.", + "name": "contentIsPlaylist", + "type": "Boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "none", + "description": "Sets the desired play state for the video, such as starting or stopping the video play. Getting the value of this field returns the most recent value set, or none if no value has been set. To dynamically monitor the actual state of the video, see the \\`state\\` field. The play and stop commands to commence and discontinue playback should not be used to implement trick modes like rewind, or replay. For that use the \\`seek\\` field.\n\n| Option | Effect |\n| --- | --- |\n| none | No play state set |\n| play | Start video play |\n| stop | Stop video play |\n| pause | Pause video play |\n| resume | Resume video play after a pause |\n| replay | Replay video |\n| prebuffer | Starts buffering the video stream before the Video node actually begins playback. Only one video stream can be buffering in the application at any time. Setting the `control` field to `prebuffer` for another video stream after setting `prebuffer` for a previous video stream stops the buffering of the previous video stream. |\n| skipcontent | Skip the currently-playing content and begin playing the next content in the playlist. If the content is not a playlist, or if the current content is the end of the playlist, this will end playback. |", + "name": "control", + "type": "option string" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "0", + "description": "The error code associated with the video play error set in the \\`state\\` field. Use the \\*\\*errorStr\\*\\* and and \\*\\*errorInfo\\*\\* fields for more descriptive diagnostic information to help identify and resolve the cause of the error.", + "name": "errorCode", + "type": "integer" + }, + { + "accessPermission": "READ-ONLY", + "default": "", + "description": "A diagnostic message to help resolve the video play error set in the \\`state\\` field. The roAssociativeArray contains the following fields:\n\n| Field | Type | Description |\n| --- | --- | --- |\n| clip\\_id | integer | The unique ID for the clip |\n| ignored | integer | Indicates whether the error generated an exception (0) or was ignored resulting in the next item in the content list being played (1). |\n| source | string | The module that generated the error. |\n| category | String | The type of error, which includes: \"http\", \"drm\", \"mediaerror\", or \"mediaplayer\". |\n| error\\_code | integer | The internal Roku code associated with the error. Use the **dbgmsg** field for debugging. |\n| dbgmsg | string | A verbose debug message that can help identify the root cause of the error. |\n| error\\_attributes | string | The error attribute, which includes the clip\\_id (the unique ID of the clip that failed to play). |", + "name": "errorInfo", + "type": "roAssociativeArray" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "", + "description": "An error message describing the video play error set in the \\`state\\` field. Use the \\*\\*errorStr\\*\\* and and \\*\\*errorInfo\\*\\* fields for more descriptive diagnostic information to help identify and resolve the cause of the error.", + "name": "errorMsg", + "type": "string" + }, + { + "accessPermission": "READ-ONLY", + "default": "", + "description": "A diagnostic message to help resolve the video play error set in the \\`state\\` field. The format of the errorStr is as follows: category:{category\\\\\\_name}:error:{error\\\\\\_code}:ignored:{0|1}:{source}:{source\\\\\\_name}:{additional catcher comment}:{error\\\\\\_string}:extra:{error\\\\\\_attributes}\n\n| errorStr Field | Type | Description |\n| --- | --- | --- |\n| category\\_name | string | The type of error, which includes: \"http\", \"drm\", \"mediaerror\", or \"mediaplayer\". |\n| error\\_code | integer | The unique code associated with the error. |\n| ignored | integer | Indicates whether the error generated an exception (0) or was ignored resulting in the next item in the content list being played (1). |\n| source | string | The module that generated the error. |\n| source\\_name | string | The module that generated the error. |\n| additional catcher comment | string | Typically, the comment added when the exception is caught. |\n| error\\_string | string | A text message describing the video play error. |\n| error\\_attributes | string | The error attribute, which includes the clip\\_id (the unique ID of the clip that failed to play). |", + "name": "errorStr", + "type": "string" + }, + { + "accessPermission": "READ\\_ONL", + "default": "", + "description": "Indicates whether the DRM license was acquired. If a failure occurs, this field provides additional details about the error. The roAssociativeArray contains the following fields:\n\n| Key | Type | Value |\n| --- | --- | --- |\n| response | string | The server response. If a license is not retrieved, the response is empty and the HTTP response code is returned instead. |\n| status | string | The HTTP response code. |\n| keysystem | string | The DRM technology used. |\n| duration | string | The total time elapsed in sending a request to the license server and receiving a response (in milliseconds). |", + "name": "licenseStatus", + "type": "roAssociativeArray" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\\-1", + "description": "If the \\`contentIsPlaylist\\` field is set to true to enable video playlists, sets the index of the next video in the playlist to be played. Setting this field does not immediately change the video being played, but takes effect when the current video is completed or skipped. By default, this value is -1, which performs the default index increment operation. After the video specified by the index in this field begins playing, the field is set to the default -1 again, so the next video played will be set by the default index increment operation unless the field is set again to a different index.", + "name": "nextContentIndex", + "type": "integer" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "", + "description": "Provides timing measurements related to the start of video playback. All measurements are in seconds. The roAssociativeArray contains the following fields:\n\n| Field | Type | Access Permission | Description |\n| --- | --- | --- | --- |\n| total\\_dur | float | READ\\_ONLY | Total video start duration. |\n| manifest\\_dur | float | READ\\_ONLY | Manifest download and parsing. |\n| drm\\_load\\_dur | float | READ\\_ONLY | DRM system initialization. |\n| drm\\_lic\\_acq\\_dur | float | READ\\_ONLY | License acquisition. This typically includes interactions with the license server. |\n| prebuf\\_dur | float | READ\\_ONLY | Prebuffer content. |\n| manifest\\_start (_Available since Roku OS 10.0_) | Float | READ\\_ONLY | Point at which manifest download and parsing begins. |\n| drm\\_load\\_start (_Available since Roku OS 10.0_) | Float | READ\\_ONLY | Point at which DRM system initialization begins. |\n| drm\\_lic\\_acq\\_start (_Available since Roku OS 10.0_) | Float | READ\\_ONLY | Point at which license acquisition begins. |\n| prebuf\\_start (_Available since Roku OS 10.0_) | Float | READ\\_ONLY | Point at which content pre-buffering begins. |\n\n\\> The \\\\\\_start fields correspond to the similarly named \\\\\\_dur (duration) fields in this structure. In each case, the \\\\\\_start point is the number of milliseconds elapsed from the initialization of the media player (t=0.000). If required, ending points for each interval can be derived from its associated starting-point and duration.", + "name": "playStartInfo _Available since Roku OS 9.3_", + "type": "roAssociativeArray" + }, + { + "accessPermission": "READ\\_ONLY", + "default": "none", + "description": "Describes the current video play state, such as if the video play has been paused.\n\n| Value | Meaning |\n| --- | --- |\n| none | No current play state |\n| buffering | Video stream is currently buffering |\n| playing | Video is currently playing |\n| paused | Video is currently paused |\n| stopped | Video is currently stopped |\n| finished | Video has successfully completed playback |\n| error | An error has occurred in the video play. The error code, message, and diagnostics can be found in the `errorCode`, `errorMsg`, and `errorStr` fields respectively. |", + "name": "state", + "type": "value string" + } + ], + "interfaces": [], + "name": "Video", + "url": "https://developer.roku.com/docs/references/scenegraph/media-playback-nodes/video.md" + }, + "voicetexteditbox": { + "availableSince": "9.4", + "description": "_Available since Roku OS 9.4_\n\nExtends [TextEditBox](https://developer.roku.com/docs/references/scenegraph/widget-nodes/texteditbox.md\n\nThe **VoiceTextEditBox** node is similar to the [legacy **TextEditBox** node](https://developer.roku.com/docs/references/scenegraph/widget-nodes/texteditbox.md, but with additional voice entry functionality. Only one voice-enabled **VoiceTextEditBox** node may be on the screen at a time. If another VoiceTextEditBox is rendered on the screen, its voice functionality is disabled implicitly.", + "events": [], + "extends": { + "name": "TextEditBox", + "url": "https://developer.roku.com/docs/references/scenegraph/widget-nodes/texteditbox.md" + }, + "fields": [ + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "The direction in which the voice hint tooltip points: \\* true. The voice hint tooltip points right. \\* false. The voice hint tooltip points left.", + "name": "flipVoiceToolTip", + "type": "boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "false", + "description": "Enables the text box to be voice-enabled. In this case, it will display a mic icon and have a voice UI with voice hints.", + "name": "voiceEnabled", + "type": "boolean" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "\"generic\"", + "description": "The type of voice entry mode to be used: \\* \"email\": letter-by-letter dictation for emails. \\* \"numeric\": letter-by-letter dictation for PIN codes, zip codes, and other numeric input. \\* \"alphanumeric\": letter-by-letter dication for street addresses or other sequences of numbers and letters. \\* \"generic\": Full word input for search queries or other sequences of numbers, letters and symbols. \\* \"password\": letter-by-letter dication for passwords.", + "name": "voiceEntryType", + "type": "string" + }, + { + "accessPermission": "READ\\_WRITE", + "default": "FHD: 321HD: 214", + "description": "The maximum width of the voice hint tootip. The height scales based on the specified width.", + "name": "voiceToolTipWidth", + "type": "float" + } + ], + "interfaces": [], + "name": "VoiceTextEditBox", + "url": "https://developer.roku.com/docs/references/scenegraph/dynamic-voice-keyboard-nodes/voice-text-edit-box.md" + }, + "zoomrowlist": { + "description": "Extends [**ArrayGrid**](https://developer.roku.com/docs/references/scenegraph/abstract-nodes/arraygrid.md\n\nThe ZoomRowList node allows a row of the Row-Row Grid to smoothly zoom up to a larger size when that row has focus. Rows in this node are capable of gaining the focus while scrolling, and smoothly zooming up by the specified amount. The amount to zoom can be specified on a per row basis so that some rows can zoom up by a larger amount than others.", + "events": [], + "extends": { + "name": "ArrayGrid", + "url": "https://developer.roku.com/docs/references/scenegraph/abstract-nodes/arraygrid.md" + }, + "fields": [], + "interfaces": [], + "name": "ZoomRowList", + "url": "https://developer.roku.com/docs/references/scenegraph/list-and-grid-nodes/zoomrowlist.md" + } + }, + "components": { + "roappinfo": { + "constructors": [ + { + "params": [], + "returnType": "roAppInfo" + } + ], + "description": "roAppInfo retrieves the developer ID, which can be useful during development. It also retrieves manifest values, such as the title and version number, avoiding the need to parse the manifest file from BrightScript. This object is created with no parameters.", + "events": [], + "interfaces": [ + { + "name": "ifAppInfo", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappinfo.md" + } + ], + "name": "roAppInfo", + "url": "https://developer.roku.com/docs/references/brightscript/components/roappinfo.md" + }, + "roappmanager": { + "constructors": [ + { + "params": [], + "returnType": "roAppManager" + } + ], + "description": "The Application Manager APIs set application level attributes, which mostly affect the look-and-feel of the application. The use of screen styles gives each application a consistent look-and-feel, but it's often desirable to customize attributes such as colors, fonts, and logos for each application. Setting artwork and colors allows the developer to specify a theme for their application. If these values are not set, the application will use default values.\n\nThe table below describes each attribute and its values, the screen types to which it applies, and the Roku OS version in which the attribute was first supported. Unless otherwise indicated, an attribute is supported in all Roku OS versions after the one in which it was first supported.\n\nTo save space, the screen types in the table are specified by a two letter code:\n\n| Code | Screen Type |\n| --- | --- |\n| Co | roCodeRegistrationScreen |\n| Di | roMessageDialog, roOneLineDialog, roPinEntryDialog |\n| Gr | roGridScreen |\n| Ke | roKeyboardScreen |\n| Li | roListScreen |\n| Pa | roParagraphScreen |\n| Po | roPosterScreen |\n| Se | roSearchScreen |\n| Sp | roSpringboardScreen |\n| Te | roTextScreen |\n\nAll attribute values are strings. Numeric values are specified as decimal strings.\n\n| Attribute | Screen Types | Values | Example | Version |\n| --- | --- | --- | --- | --- |\n| BackgroundColor | Gr Li Pa Po Se Sp Te | HTML HEX Color Value | #E0DFDF | 1.0 |\n| BreadcrumbDelimiter | Gr Li Pa Po Se Sp Te | HTML HEX Color Value | #FF00FF | 1.0 |\n| BreadcrumbTextLeft | Gr Li Pa Po Se Sp Te | HTML HEX Color Value | #FF00FF | 1.0 |\n| BreadcrumbTextRight | Gr Li Pa Po Se Sp Te | HTML HEX Color Value | #FF00FF | 1.0 |\n| ButtonHighlightColor | Di Se Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| ButtonMenuHighlightText | Di Se Sp | HTML HEX Color Value | #0033FF | 1.0 |\n| ButtonMenuNormalOverlayText | Di Se Sp | HTML HEX Color Value | #B0B0B0 | 1.0 |\n| ButtonMenuNormalText | Di Se Sp | HTML HEX Color Value | #686868 | 1.0 |\n| ButtonNormalColor | Di Se Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| CounterSeparator | Gr Po | HTML HEX Color Value | #00FF00 | 2.7 |\n| CounterTextLeft | Gr Po | HTML HEX Color Value | #FF0000 | 2.7 |\n| CounterTextRight | Gr Po | HTML HEX Color Value | #0000FF | 2.7 |\n| DialogBodyText | Di | HTML HEX Color Value. Must be a grayscale value | #808080 | 3.1 |\n| DialogTitleText | Di | HTML HEX Color Value. Must be a grayscale value | #363636 | 3.1 |\n| EpisodeSynopsisText | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| FilterBannerActiveColor | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| FilterBannerActiveHD | Po | URL to set HD Filter Banner Active/Focus Highlighter | pkg:/images/Filter\\_ActiveHint\\_HD.png | 1.0 |\n| FilterBannerActiveSD | Po | URL to set SD Filter Banner Active/Focus Highlighter | pkg:/images/Filter\\_ActiveHint\\_SD43.png | 1.0 |\n| FilterBannerInactiveColor | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| FilterBannerInactiveHD | Po | URL to set HD Filter Banner Inactive Highlighter | pkg:/images/Filter\\_InactiveHint\\_HD.png | 1.0 |\n| FilterBannerInactiveSD | Po | URL to set SD Filter Banner Inactive Highlighter | pkg:/images/Filter\\_ActiveHint\\_SD43.png | 1.0 |\n| FilterBannerSideColor | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| FilterBannerSliceHD | Po | URL to set HD Filter Banner Background Image | pkg:/images/Filter\\_ActiveHint\\_HD.png | 1.0 |\n| FilterBannerSliceSD | Po | URL to set SD Filter Banner Background Image | pkg:/images/Filter\\_ActiveHint\\_SD43.png | 1.0 |\n| GridScreenBackgroundColor | Gr | HTML HEX Color Value Must be a grayscale value | #363636 | 2.7 |\n| GridScreenBorderOffsetHD | Gr | String representing point \"(x, y)\" that is the offset from the upper left corner of the focused HD image. Set to the negative width & height of border | (-25,-25) | 2.8 |\n| GridScreenBorderOffsetSD | Gr | String representing point \"(x, y)\" that is the offset from the upper left corner of the focused SD image. Set to the negative width & height of border | (-20,-20) | 2.8 |\n| GridScreenDescriptionDateColor | Gr | HTML HEX Color Value | #FF005B | 2.7 |\n| GridScreenDescriptionImageHD | Gr | URL to set HD Description callout background image on Grid | pkg:/images/Description\\_Background\\_HD.ng | 2.8 |\n| GridScreenDescriptionImageSD | Gr | URL to set SD Description callout background image on Grid | pkg:/images/Description\\_Background\\_SD43.ng | 2.8 |\n| GridScreenDescriptionOffsetHD | Gr | String representing point \"(x, y)\" that is the offset from the upper left corner of the focused HD image. Negative values have the description above and to the left of the focused image | (190,255) | 2.8 |\n| GridScreenDescriptionOffsetSD | Gr | String representing point \"(x, y)\" that is the offset from the upper left corner of the focused SD image. Negative values have the description above and to the left of the focused image | (125,170) | 2.8 |\n| GridScreenDescriptionRuntimeColor | Gr | HTML HEX Color Value | #5B005B | 2.7 |\n| GridScreenDescriptionSynopsisColor | Gr | HTML HEX Color Value | #606000 | 2.7 |\n| GridScreenDescriptionTitleColor | Gr | HTML HEX Color Value | #00FFFF | 2.7 |\n| GridScreenFocusBorderHD | Gr | URL to set HD Focus image on Active Grid Poster | pkg:/images/Border\\_16x9\\_HD.png | 2.8 |\n| GridScreenFocusBorderSD | Gr | URL to set SD Focus image on Active Grid Poster | pkg:/images/Border\\_16x9\\_SD43.png | 2.8 |\n| GridScreenListNameColor | Gr | HTML HEX Color Value. Must be a grayscale value | #FFFFFF | 2.7 |\n| GridScreenLogoHD | Gr | Logo formatted for display in the overhang | pkg:/images/gridlogoHD.png | 2.7 |\n| GridScreenLogoOffsetHD\\_X | Gr | Offset in pixels from the top-left origin of the display. Range 0 to 1280 | 592 | 2.7 |\n| GridScreenLogoOffsetHD\\_Y | Gr | Offset in pixels from the top-left origin of the display. Range 0 to 720 | 31 | 2.7 |\n| GridScreenLogoOffsetSD\\_X | Gr | Offset in pixels from the top-left origin of the display. Range 0 to 720 | 324 | 2.7 |\n| GridScreenLogoOffsetSD\\_Y | Gr | Offset in pixels from the top-left origin of the display. Range 0 to 480 | 21 | 2.7 |\n| GridScreenLogoSD | Gr | Logo formatted for display in the overhang | pkg:/images/gridlogoSD.png | 2.7 |\n| GridScreenMessageColor | Gr | HTML HEX Color Value. Must be a grayscale value | #808080 | 2.7 |\n| GridScreenOverhangHeightHD | Gr | The HD overhang height. Default: \"69\" | 75 | 2.8 |\n| GridScreenOverhangHeightSD | Gr | The SD overhang height. Default: \"49\" | 55 | 2.8 |\n| GridScreenOverhangSliceHD | Gr | URI for the overhang slice (thin piece of top of screen border) | pkg:/images/gridoverhangHD.png | 2.7 |\n| GridScreenOverhangSliceSD | Gr | URI for the overhang slice (thin piece of top of screen border) | pkg:/images/gridoverhangSD.png | 2.7 |\n| GridScreenRetrievingColor | Gr | HTML HEX Color Value. Must be a grayscale value | #CCCCCC | 2.7 |\n| ListItemHighlightHD | Gr Li Po | URL to set HD highlight image | pkg:/images/listitem\\_highlight\\_hd.png | 3.1 |\n| ListItemHighlightSD | Gr Li Po | URL to set SD highlight image | pkg:/images/listitem\\_highlight\\_sd.png | 3.1 |\n| ListItemHighlightText | Gr Li Po | HTML HEX Color Value | #CCCC00 | 3.1 |\n| ListItemText | Gr Li Po | HTML HEX Color Value | #CCCC00 | 3.1 |\n| ListScreenDescriptionText | Li | HTML HEX Color Value | #CCCC00 | 3.1 |\n| ListScreenTitleColor | Li | HTML HEX Color Value | #CC0000 | 3.1 |\n| OverhangPrimaryLogoHD | Co Ke Li Pa Po Se Sp Te | Small application logo formatted for display in overhang top left | pkg:/images/co\\_logo\\_sd.png | 1.0 |\n| OverhangPrimaryLogoOffsetHD\\_X | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films.Range 0 to 1280 | 25 | 1.0 |\n| OverhangPrimaryLogoOffsetHD\\_Y | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films.Range 0 to 720 | 50 | 1.0 |\n| OverhangPrimaryLogoOffsetSD\\_X | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films.Range 0 to 720 | 25 | 1.0 |\n| OverhangPrimaryLogoOffsetSD\\_Y | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films.Range 0 to 480 | 50 | 1.0 |\n| OverhangPrimaryLogoSD | Co Ke Li Pa Po Se Sp Te | Small application logo formatted for display in overhang top left | pkg:/images/co\\_logo\\_sd.png | 1.0 |\n| OverhangSecondaryLogoHD | Co Ke Li Pa Po Se Sp Te | Small application logo formatted for display in overhang top left | pkg:/images/co\\_logo\\_hd.png | 1.0 |\n| OverhangSecondaryLogoOffsetHD\\_X | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films. Range 0 to 1280 | 25 | 1.0 |\n| OverhangSecondaryLogoOffsetHD\\_Y | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films. Range 0 to 720 | 50 | 1.0 |\n| OverhangSecondaryLogoOffsetSD\\_X | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films. Range 0 to 720 | 25 | 1.0 |\n| OverhangSecondaryLogoOffsetSD\\_Y | Co Ke Li Pa Po Se Sp Te | Offset in pixels from the top-left origin of the display films. Range 0 to 480 | 50 | 1.0 |\n| OverhangSecondaryLogoSD | Co Ke Li Pa Po Se Sp Te | Small application logo formatted for display in overhang top left | pkg:/images/co\\_logo\\_sd.png | 1.0 |\n| OverhangSliceHD | Co Ke Li Pa Po Se Sp Te | URI for the overhang slice (thin piece of border at the top of the screen in HD size) | pkg:/images/overhang\\_hd.png | 1.0 |\n| OverhangSliceSD | Co Ke Li Pa Po Se Sp Te | URI for the overhang slice (thin piece of top of screen border) | pkg:/images/overhang\\_sd.png | 1.0 |\n| ParagraphBodyText | Co Pa Te | HTML HEX Color Value | #FF00FF | 1.0 |\n| ParagraphHeaderText | Co Pa Te | HTML HEX Color Value | #FF00FF | 1.0 |\n| PosterScreenLine1Text | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| PosterScreenLine2Text | Po | HTML HEX Color Value | #FF00FF | 1.0 |\n| RegistrationCodeColor | Co | HTML HEX Color Value | #FF00FF | 1.0 |\n| RegistrationFocalColor | Co | HTML HEX Color Value | #FF00FF | 1.0 |\n| RegistrationFocalRectColor | Co | HTML HEX Color Value | #10FF80 | |\n| RegistrationFocalRectHD | Co | Position and size of the HD focal rectangle. Four integer: (x,y,width,height) | (228,360,120,82) | |\n| RegistrationFocalRectSD | Co | Position and size of the SD focal rectangle. Four integer: (x,y,width,height) | (172,220,90,76) | |\n| SpringboardActorColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardAlbumColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardAlbumLabel | Sp | Album Label | on | 1.0 |\n| SpringboardAlbumLabelColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardAllow6Buttons | Sp | boolean string | true | |\n| SpringboardArtistColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardArtistLabel | Sp | Artist Label | by | 1.0 |\n| SpringboardArtistLabelColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardDirectorColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardDirectorText | Sp | Director Label | Written by | 1.0 |\n| SpringboardDirectorLabelColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardDirectorPrefixText | Sp | HTML HEX Color Value | #FF00FF | |\n| SpringboardGenreColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardRuntimeColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardSynopsisColor | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| SpringboardTitleText | Sp | HTML HEX Color Value | #FF00FF | 1.0 |\n| TextScreenBodyBackgroundColor | Te | HTML HEX Color Value. Must be a grayscale value | #808080 | 4.3 |\n| TextScreenBodyText | Te | HTML HEX Color Value | #363636 | 4.3 |\n| TextScreenScrollBarColor | Te | HTML HEX Color Value | #CC0000 | 4.3 |\n| TextScreenScrollThumbColor | Te | HTML HEX Color Value | #00CC00 | 4.3 |\n| ThemeType | | Theme type. Generic-dark is the only valid value. Otherwise the default theme applies | generic-dark | 2.7 |\n\n**Example**\n\n```\nSub SetApplicationTheme()\n app = CreateObject(\"roAppManager\")\n theme = CreateObject(\"roAssociativeArray\")\n theme.OverhangSliceHD = \"pkg:/images/Overhang_Slice_HD.png\"\n theme.OverhangPrimaryLogoSD = \"pkg:/images/Logo_Overhang_SD43.png\"\n theme.OverhangPrimaryLogoOffsetSD_X = \"72\"\n theme.OverhangPrimaryLogoOffsetSD_Y = \"25\"\n theme.OverhangPrimaryLogoHD = \"pkg:/images/Logo_Overhang_HD.png\"\n theme.OverhangPrimaryLogoOffsetHD_X = \"123\"\n theme.OverhangPrimaryLogoOffsetHD_Y = \"48\"\n app.SetTheme(theme)\nEnd Sub\n```", + "events": [], + "interfaces": [ + { + "name": "ifAppManager", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappmanager.md" + } + ], + "name": "roAppManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/roappmanager.md" + }, + "roarray": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "size", + "type": "Integer" + }, + { + "isRequired": true, + "name": "resizeAs", + "type": "Boolean" + } + ], + "returnType": "roArray" + } + ], + "description": "An array stores an indexed collection of BrightScript objects. Each entry of an array can be a different type, or they may all of the same type.\n\nAn roArray is created with two parameters:\n\n**CreateObject(\"roArray\", size As Integer, resizeAs Boolean)**\n\nSize is the initial number of elements allocated for the array. If resize is true, the array will be resized if needed to accommodate more elements. If the array is large, this might be slow. The \"dim\" statement may be used instead of CreateObject to allocate a new array. Dim has the advantage in that it automatically creates arrays of arrays for multi-dimensional arrays.", + "events": [], + "interfaces": [ + { + "name": "ifArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarray.md" + }, + { + "name": "ifArrayGet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayget.md" + }, + { + "name": "ifArrayJoin", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayjoin.md" + }, + { + "name": "ifArraySet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayset.md" + }, + { + "name": "ifArraySort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarraysort.md" + }, + { + "name": "ifEnum", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifenum.md" + } + ], + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + }, + "roassociativearray": { + "constructors": [ + { + "params": [], + "returnType": "roAssociativeArray" + } + ], + "description": "An associative array (also known as a map, dictionary or hash table) allows objects to be associated with string keys. Associative arrays are built into the language. They can be accessed implicitly by using the dot or bracket operators, or by calling functions from the [ifAssociativeArray](https://developer.roku.com/docs/references/brightscript/interfaces/ifassociativearray.md\"ifAssociativeArray\") interface. For example, the last three lines in this example are equivalent:\n\n```\naa = { one : 1, two : 2, three : 3 }\nx = aa[\"two\"]\nx = aa.two\nx = aa.Lookup(\"two\")\n```\n\nThis object is created with no parameters:\n\n```\nCreateObject(\"roAssociativeArray\")\n```\n\nIt can also be created implicitly by using an Associative Array literal.\n\nStarting from Roku OS 8, the quoted keys in Associative Array literals are now case-preserving. This change improves the readability of your code and is compatible with JSON usage.\n\n**Example**\n\n```\n' Creation of associative arrays\n\naa1 = CreateObject(\"roAssociativeArray\") ' Explicitly \naa2 = {} ' Implicitly\naa3 = { ' With some initial values\n foo : 12,\n bar : 13\n}\n\n' Assigning values\n\naa1.AddReplace(\"Bright\", \"Script\") ' With explicit function calls\naa1.AddReplace(\"TMOL\", 42)\naa1.boo = 112 ' With dot operator\naa1[\"baz\"] = \"abcdefg\" ' With bracket operator\n\n' Accessing values\n\nprint aa1.Bright ' With dot operator (will print 'Script')\nprint aa1.Lookup(\"TMOL\") ' With function call (will print 42)\nprint aa1[\"boo\"] ' With bracket operator (will print 112)\n\n' Using ifEnum interface to walk through keys in an associative array\nfor each key in aa1\n\n print \" \" key \"=\" aa1[key]\n\nend for\n```", + "events": [], + "interfaces": [ + { + "name": "ifAssociativeArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifassociativearray.md" + }, + { + "name": "ifEnum", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifenum.md" + } + ], + "name": "roAssociativeArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md" + }, + "roaudioguide": { + "constructors": [], + "description": "> This component is only available on the following devices: Roku Streaming Stick (3600X), Roku Express (3700X) and Express+ (3710X), Roku Premiere (4620X) and Premiere+ (4630X), Roku Ultra (4640X), and any Roku TV running Roku OS version 7.5 and later.\n\nThe roAudioGuide component provides Audio Guide support for applications that require custom speech beyond what is provided by automatic Audio Guide in SDK and Scene Graph components.\n\nThough some of the roAudioGuide API is similar to [roTextToSpeech](https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md\"roTextToSpeech\"), roAudioGuide provides support specific to Audio Guide, including:\n\n* Speaks when Audio Guide is enabled, and doesn't speak if it isn't.\n* Automatically splits up text to reduce lag.\n* Uses the correct voice, language, volume, and speech rate for Audio Guide.\n* Tries to be \"smart\" by pre-processing the text for correct pronunciation of things like currency, email addresses, acronyms, media-related names and titles, etc.\n\nUsually, roAudioGuide would be used on its own, but it can be used in conjunction with [roTextToSpeech](https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md\"roTextToSpeech\").", + "events": [], + "interfaces": [ + { + "name": "ifAudioGuide", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioguide.md" + } + ], + "name": "roAudioGuide", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioguide.md" + }, + "roaudiometadata": { + "constructors": [ + { + "params": [], + "returnType": "roAudioMetadata" + } + ], + "description": "The roAudioMetadata component provides developers access to audio file metadata included in many audio files. This should enable some audiofiles to deliver the information needed to fill out an roSpringboard screen without passing the info in a separate xml feed. roAudioMetadata currently only works with local file URLs.\n\nThe component requires the use of a dynamically loaded library that is not part of the initially booted image. Therefore, an entry must be added to the manifest of any applications that use the roAudioMetadata component so it can be loaded when the channel is launched. Here's the manifest entry:\n\n_requires\\_audiometadata=1_\n\nThis object is created without any arguments:\n\n`CreateObject(\"roAudioMetadata\")`\n\n**Example**\n\n```\nREM printAA() is from generalUtils.brs in our sample apps\nREM and used to print an associative Array\n\nSub SaveCoverArtFile(filename As String)\n meta = CreateObject(\"roAudioMetadata\")\n meta.SetUrl(filename)\n print \"------------- GetTags() -------------------------\"\n tags = meta.GetTags()\n printAA(tags)\n print \"------------- GetAudioProperties() --------------\"\n properties = meta.GetAudioProperties()\n printAA(properties)\n print \"------------- GetCoverArt() ---------------------\"\n thumbnail = meta.GetCoverArt()\n if (thumbnail <> invalid) then\n if (thumbnail.bytes = invalid) then\n return\n end if\n imgtype = thumbnail.type\n image_ext=\"\"\n if (imgtype = \"image/jpeg\" or imgtype = \"jpg\") then\n image_ext = \"jpg\"\n else if (imgtype = \"image/png\" or imgtype = \"png\") then\n image_ext = \"png\"\n else\n image_ext = \"jpg\"\n end if\n tmp_img = \"tmp:/CoverArtImage\" + \".\" + image_ext\n if (tmp_img <> invalid) then\n DeleteFile(tmp_img)\n end if\n thumbnail.bytes.Writefile(tmp_img)\n end if\nEnd Sub\n```", + "events": [], + "interfaces": [ + { + "name": "ifAudioMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudiometadata.md" + } + ], + "name": "roAudioMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudiometadata.md" + }, + "roaudioplayer": { + "constructors": [ + { + "params": [], + "returnType": "roAudioPlayer" + } + ], + "description": "The Audio Player object provides the ability to setup the playing of a series of audio streams. The object accepts an array of content meta-data objects, describing the audio and providing URLs for accessing each stream. The component understands the following streamformat values: \"mp3\", \"wma\", \"mp4\", \"hls\", \"es.aac-adts\", \"flac.\"\n\nThis object does not provide an interface to a screen. In order to get events both from the screen you are using and the Audio Player, you should use the same Message Port for both objects.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roAudioPlayer\")`\n\n**Example**\n\n```\nSub Main()\n audioPlayer = CreateObject(\"roAudioPlayer\")\n port = CreateObject(\"roMessagePort\")\n audioPlayer.SetMessagePort(port)\n song = CreateObject(\"roAssociativeArray\")\n song.url = \"http://www.theflute.co.uk/media/BachCPE_SonataAmin_1.wma\"\n audioplayer.addcontent(song)\n audioplayer.setloop(false)\n audioPlayer.play()\n while true\n msg = wait(0, port)\n if type(msg) = \"roAudioPlayerEvent\"\n if msg.isStatusMessage() then\n print \"roAudioPlayerEvent: \"; msg.getmessage()\n if msg.getmessage() = \"end of playlist\" return\n endif\n endif\n end while\nEnd Sub\n```", + "events": [ + { + "name": "roAudioPlayerEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roaudioplayerevent.md" + } + ], + "interfaces": [ + { + "name": "ifAudioPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioplayer.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roAudioPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioplayer.md" + }, + "roaudioresource": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "dynamic" + } + ], + "returnType": "roAudioResource" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roAudioResource" + } + ], + "description": "The roAudioResouce allows .wav files to be cached to memory and quickly played at any time. roAudioResource is intended to support short audio clips which need to be played with very little latency. The system caches the entire wav file in memory so that playback can begin very quickly.\n\nOn Roku \"Classic\" models, roAudioResource does not support mixing of sounds. So when you play a sound effect, any background music is paused while the sound effect plays and then resumes after the sound effect ends. On later models, sound effects are mixed with background music. See the [Hardware specifications document](/docs/specs/hardware.md#current-models) for a list of Current and Classic models.\n\nThis object is created with a filename parameter that is a path to the sound resource file:\n\n`CreateObject(\"roAudioResource\", filename)`\n\nThe filename must be the name of a local file and cannot be a URL. To use a URL, you may download the file to the application's \"tmp:\" file system using [roUrlTransfer](https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md and pass a filename of the form \"tmp:/file.wav\" to CreateObject.\n\n```\nsound = CreateObject(\"roAudioResource\", \"pkg:/sounds/beep1.wav\")\nsound.Trigger(75)\n```\n\nAn object can also be created using the name of a system sound effect:\n\n* \"select\" - the sound effect to be played when a selection is made, e.g. when OK is pressed.\n* \"navsingle\" - the sound effect to be played when navigating a list or grid, e.g. when left or right is pressed.\n* \"navmulti\" - the sound effect to be played when paging through a list or grid, e.g. when rewind or fast-forward is pressed.\n* \"deadend\" - the sound effect to be played when a button press could not be processed.\n\nNote that system sound effects are played at the volume selected in the user's settings, or not played at all if the user has turned sound effects off, regardless of the volume value passed to Trigger.\n\n```\nsound = CreateObject(\"roAudioResource\", \"select\")\nsound.Trigger(50)\n```\n\nMult iple sounds can be mixed and played at the same time:\n\n```\nsound1 = CreateObject(\"roAudioResource\", \"pkg:/sounds/beep1.wav\")\nsound2 = CreateObject(\"roAudioResource\", \"select\")\nsound1.Trigger(50, 0)\nif sound2.maxSimulStreams() > 1\n sound2.Trigger(50, 1)\nend if\n```", + "events": [], + "interfaces": [ + { + "name": "ifAudioResource", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioresource.md" + } + ], + "name": "roAudioResource", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioresource.md" + }, + "robitmap": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "bitmapProps", + "type": "Object" + } + ], + "returnType": "roBitmap" + }, + { + "params": [ + { + "isRequired": true, + "name": "filename", + "type": "String" + } + ], + "returnType": "roBitmap" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roBitmap" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "dynamic" + } + ], + "returnType": "roBitmap" + } + ], + "description": "The roBitmap component contains image data and provides an interface (ifDraw2D) for drawing. Bitmaps can be used for a variety of purposes, such as for sprites, compositing, or as double buffers.\n\nIt stores four color channels: red, green, blue, and alpha, with 32-bits per pixel. They can be any arbitrary size up to 2048x2048. However, the maximum size bitmap uses 16MB of memory, so there are practical memory limitations which would compel smaller bitmap sizes. Coordinates (x,y) for 2D bitmaps have an origin (0,0) at the top left. roBitmap is always offscreen. The top roScreen is the only ifDraw2D surface which is displayed. roBitmap represents something that can be drawn onto, as well as something that can be drawn.\n\nDrawing operations into a roBitmap (or other surface with ifDraw2D interface, such as an [roScreen](https://developer.roku.com/docs/references/brightscript/components/roscreen.md\"roScreen\")) are clipped so the only the part that is within its bounds is rendered. X,Y coordinates that specify a location in a bitmap to render to (for example, as used by DrawObject() or DrawText() ) may be positive or negative. Negative implies that the left and top of the rendered object will be clipped.The same bitmap cannot be used as a source and a destination in a single DrawObject() call.\n\nThere are limitations when using the onscreen bitmap as a source. For example, Alpha blending may not work.\n\nAn empty roBitmap object can be created with CreateObject():\n\n`CreateObject(\"roBitmap\", bitmapProps As Object)`\n\nbitmapProps is an roAssociativeArray with Integers width (Integer), height (Integer), and AlphaEnable (Boolean), and name (String) parameters. The contents of an empty RoBitmap are initialized to zero (transparent black).\n\nExample: `CreateObject(\"roBitmap\", {width:10, height:10, AlphaEnable:false, name:\"MyBitmapName\"})`\n\nAn roBitmap can also load its image data from a file:\n\n`CreateObject(\"roBitmap\", String filename)`\n\n**Example**\n\n```\n' Draw three bitmaps as fast as we can\n'\nScreen=CreateObject(\"roScreen\")\nbm1=CreateObject(\"roBitmap\", \"pkg:/images/myphoto1.jpg\")\nbm2=CreateObject(\"roBitmap\", \"pkg:/images/myphoto2.jpg\")\nbm3=CreateObject(\"roBitmap\", \"pkg:/images/myphoto3.jpg\")\nbmarray=[bm1, bm2, bm3]\nWhile true\n For each bitmap in bmarray\n Screen.DrawObject(0,0, bitmap)\n Screen.Finish()\n End for\nEnd While\n```\n\n**Example: Double buffering with roBitmap**\n\n```\nscreen1=CreateObject(\"roScreen\")\noff=CreateObject(\"roBitmap\", {width:1280, height:720, AlphaEnable:false})\noff.Clear(white)\ndfDrawImage(off, \"pkg:/images/myimage.png\", 50, 50)\noff.DrawRect(150, 150, 200, 200, &hFF) ' black, alpha: all source\nscreen1.DrawObject(0, 0, off)\nScreen1.Finish()\n```", + "events": [], + "interfaces": [ + { + "name": "ifDraw2D", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdraw2d.md" + } + ], + "name": "roBitmap", + "url": "https://developer.roku.com/docs/references/brightscript/components/robitmap.md" + }, + "roboolean": { + "constructors": [], + "description": "roBoolean is the object equivalent for intrinsic type Boolean.\n\nThis is useful in the following situations:\n\n* When an object is needed, instead of an intrinsic value. For example, \"roList\" maintains a list of objects. If an Boolean is added to roList, it will be automatically wrapped in an roBoolean by the language interpreter. When a function that expects a BrightScript Component as a parameter is passed a boolean, BrightScript automatically creates the equivalent BrightScript Component.\n* If any object exposes the ifBoolean interface, that object can be used in any expression that expects an intrinsic value.", + "events": [], + "interfaces": [ + { + "name": "ifBoolean", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifboolean.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roBoolean", + "url": "https://developer.roku.com/docs/references/brightscript/components/roboolean.md" + }, + "robytearray": { + "constructors": [ + { + "params": [], + "returnType": "roByteArray" + } + ], + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes.\n\nThis object contains functions to convert strings to or from a byte array, as well as to or from ascii hex or ascii base 64. Note that if you are converting a byte array to a string, and the byte array contains a zero, the string conversion will end at that point. roByteArray will autosize to become larger as needed. If you wish to turn off this behavior, then use the SetResize() function. If you read an uninitialized index, \"invalid\" is returned. roByteArray supports the [ifArray](https://developer.roku.com/docs/references/brightscript/interfaces/ifarray.md\"ifArray\") interface, and so can be accessed with the array \\[\\] operator. The byte array is always accessed as unsigned bytes when using this interface. roByteArray also supports the ifEnum interface, and so can be used with a \"for each\" statement.\n\n**Example**\n\n```\nba=CreateObject(\"roByteArray\")\nba.FromAsciiString(\"leasure.\")\nif ba.ToBase64String()<>\"bGVhc3VyZS4=\" then stop\n\nba=CreateObject(\"roByteArray\")\nba.fromhexstring(\"00FF1001\")\nif ba[0]<>0 or ba[1]<>255 or ba[2]<>16 or ba[3]<>1 then stop\n\nba=CreateObject(\"roByteArray\")\nfor x=0 to 4000\n ba.push(x)\nend for\n\nba.WriteFile(\"tmp:/ByteArrayTestFile\")\nba2=CreateObject(\"roByteArray\")\nba2.ReadFile(\"tmp:/ByteArrayTestFile\")\nif ba.Count()<>ba2.Count() then stop\nfor x=0 to 4000\n if ba[x]<>ba2[x] then stop\nend for\n\nba2.ReadFile(\"tmp:/ByteArrayTestFile\", 10, 100)\nif ba2.count()<>100 then stop\nfor x=10 to 100\n if ba2[x-10]<>x then stop\nend for\n```", + "events": [], + "interfaces": [ + { + "name": "ifArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarray.md" + }, + { + "name": "ifArrayGet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayget.md" + }, + { + "name": "ifArraySet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayset.md" + }, + { + "name": "ifByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifbytearray.md" + }, + { + "name": "ifEnum", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifenum.md" + } + ], + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + }, + "rocaptionrenderer": { + "constructors": [], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roCaptionRenderer component provides a mechanism for BrightScript channels to render closed captions in video played back with the roVideoPlayer. Prior to the v 5.2 Roku firmware, captions could only be rendered in roVideoScreen.\n\nPrior to the 5.2 Roku OS version, closed captions could only be rendered in roVideoScreen. Now channels that use roVideoPlayer embedded in an roScreen or roImageCanvas can also take advantage of Roku's closed captioning support. roCaptionRenderer supports two different modes, which is set using the [SetMode()](https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.mdsetmodemode-as-integer-as-void \"SetMode()\") method. Depending on the mode set, and the type of screen being used, the BrightScript channel needs to do different levels of work to render captions. These different workflows are highlighted in the tables below:\n\n**Model 1**\n\n| roScreen | roImageCanvas |\n| --- | --- |\n| Call [SetScreen()](https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.mdsetscreenscreen-as-object-as-void \"SetScreen()\") | Call [SetScreen()](https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.mdsetscreenscreen-as-object-as-void \"SetScreen()\") |\n| Call [UpdateCaption()](https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.mdupdatecaption-as-void \"UpdateCaption()\") | |\n\n**Model 2**\n\n| roScreen | roImageCanvas |\n| --- | --- |\n| All caption rendering is done by the channel's BrightScript code | All caption rendering is done by the channel's BrightScript code |\n\nBrightScript channels do not create roCaptionRenderer instances directly using CreateObject(). Instead, when an roVideoPlayer is created, it contains an roCaptionRenderer. BrightScript channels call [ifVideoPlayer.GetCaptionRenderer()](https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoplayer.mdgetcaptionrenderer-as-object \"ifVideoPlayer.GetCaptionRenderer()\") to get the caption renderer associated with their video player.\n\n**Example**\n\n```\nFunction Main() as void\n mode = 1\n fonts = CreateObject(\"roFontRegistry\")\n fonts.Register(\"pkg:/fonts/vSHandprinted.otf\")\n font = fonts.GetFont(\"vSHandprinted\", 28, 500, false)\n screen = CreateObject(\"roScreen\", true)\n port = CreateObject(\"roMessagePort\")\n screen.Clear(&h00)\n screen.SwapBuffers()\n screen.SetMessagePort(port)\n timer = CreateObject(\"roTimespan\")\n screenSize = {}\n screenSize.width = screen.GetWidth()\n screenSize.height = screen.GetHeight()\n\n player = CreateObject(\"roVideoPlayer\")\n player.SetContentList([\n {\n Stream : { url :\"http://ecn.channel9.msdn.com/o9/content/smf/smoothcontent/elephantsdream/Elephants_Dream_1024-h264-st-aac.ism/manifest\" }\n StreamFormat : \"ism\"\n TrackIDAudio: \"audio_eng\"\n TrackIDSubtitle: \"ism/textstream_eng\"\n }\n ])\n\n captions = player.GetCaptionRenderer()\n if (mode = 1)\n captions.SetScreen(screen)\n endif\n captions.SetMode(mode)\n captions.SetMessagePort(port)\n captions.ShowSubtitle(true)\n\n player.play()\n\n while true\n msg = wait(250, port)\n if type(msg) = \"roCaptionRendererEvent\"\n if msg.isCaptionText()\n print \"isCaptionText\"\n if msg.GetMessage() <> invalid and msg.GetMessage() <> \"\"\n DrawCaptionString(screen, screenSize, msg.GetMessage(), font)\n timer.Mark()\n else if timer.TotalSeconds() > 2\n ClearCaptionString(screen)\n endif\n else if msg.isCaptionUpdateRequest()\n print \"isCaptionUpdateRequest()\"\n UpdateCaptions(screen, captions)\n end if\n endif\n end while\nEnd Function\n\nFunction UpdateCaptions(screen as object, captions as object) as Void\n screen.Clear(&h00)\n captions.UpdateCaption()\n screen.SwapBuffers()\nEnd Function\n\nFunction DrawCaptionString(screen as object, screenSize as object, caption as String, font as object) as Void\n screen.Clear(&h00)\n textHeight = font.GetOneLineHeight()\n textWidth = font.GetOneLineWidth(caption, screenSize.width)\n x = (screenSize.width - textWidth) / 2\n y = screenSize.height - textHeight\n screen.DrawText(caption, x, y, &hd5d522ff, font)\n screen.SwapBuffers()\nEnd Function\n\nFunction ClearCaptionString(screen as object) as void\n screen.Clear(&h00)\n screen.SwapBuffers()\nEnd Function\n```", + "events": [ + { + "name": "roCaptionRendererEvent", + "url": "https://developer.roku.com/docs/references/brightscript/event/rocaptionrendererevent.md" + } + ], + "interfaces": [ + { + "name": "ifCaptionRenderer", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.md" + } + ], + "isDeprecated": true, + "name": "roCaptionRenderer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocaptionrenderer.md" + }, + "rochannelstore": { + "constructors": [ + { + "params": [], + "returnType": "roChannelStore" + } + ], + "description": "The roChannelStore component allows the application to perform a purchase of an In-Channel Product or upgrade a channel. Most of the purchase flow, screens and messaging associated with the financial transaction are handled by the Roku OS outside of control or monitoring by BrightScript code. The BrightScript code merely initiates the purchase and receives a final result. This will engender trust with users and give them confidence that they are dealing with the Roku Channel Store.\n\nThe roChannelStore component allows purchasing only those In-Channel Products which are associated with the running channel. Please see [Adding in-channel products](/docs/developer-program/roku-pay/quickstart/in-channel-products.md \"Adding in-channel products\") for details on how to create an In-Channel Product and associate it with a channel. After one or Products are created, GetCatalog() can be used to retrieve a list of Products and their attributes. DoOrder() can be called to initiate a purchase of one or more of the Products.\n\nThe roChannelStore object has a FakeServer() method that will enable you to test the purchase flow scenarios without actually making a real transaction in the Roku channel store. This will be useful in the development of your channel, but should never be used in the actual channel you publish.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roChannelStore\")`\n\n> Because [ifChannelStore.DoOrder()](https://developer.roku.com/docs/references/brightscript/interfaces/ifchannelstore.mddoorder-as-boolean \"ifChannelStore.DoOrder()\") needs to read the product type returned by GetCatalog(), only one instance of roChannelStore should ever be used for a complete purchase flow - you should never create a separate roChannelStore object to complete a purchase that was initiated by a different instance of roChannelStore.", + "events": [ + { + "name": "roChannelStoreEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rochannelstoreevent.md" + } + ], + "interfaces": [ + { + "name": "ifChannelStore", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifchannelstore.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roChannelStore", + "url": "https://developer.roku.com/docs/references/brightscript/components/rochannelstore.md" + }, + "rocoderegistrationscreen": { + "constructors": [ + { + "params": [], + "returnType": "roCodeRegistrationScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Code Registration Screen is designed to present the user a registration code, and the information required to instruct the user on how to register with a service provider. This screen is designed for a rendezvous registration process, where the user is presented a code and the URL for a registration site. The user goes to the site and enters their code, which causes the device and the account to be linked. In the background, the script is polling for completion and the screen is closed to display an activation successful screen when done.\n\n**Diagram: roCodeRegistrationScreen**\n\n![Diagram: roCodeRegistrationScreen](https://image.roku.com/ZHZscHItMTc2/roCodeRegistrationScreenImage1.png \"roCodeRegistrationScreenImage1\")\n\n**Example**\n\n```\nFunction ShowMessageDialog() As Void\n port = CreateObject(\"roMessagePort\")\n screen = CreateObject(\"roCodeRegistrationScreen\")\n screen.SetMessagePort(port)\n screen.SetTitle(\"[Registration screen title]\")\n screen.AddParagraph(\"[Registration screen paragraphs are justified to right and left edges]\")\n screen.AddFocalText(\" \", \"spacing-dense\")\n screen.AddFocalText(\"From your computer,\", \"spacing-dense\")\n screen.AddFocalText(\"go to mysite.com/roku\", \"spacing-dense\")\n screen.AddFocalText(\"and enter this code:\", \"spacing-dense\")\n screen.AddFocalText(\" \", \"spacing-dense\")\n screen.SetRegistrationCode(\"retrieving code...\")\n screen.AddParagraph(\"[Registration screen paragraphs are justified to right and left edges and may continue on multiple lines]\")\n screen.AddButton(0, \"get a new code\")\n screen.AddButton(1, \"back\")\n screen.Show()\n sleep (10000) 'simulate fetching registration code from webapi\n screen.SetRegistrationCode(\"ABC7TG\")\n screen.Show()\n while true\n dlgMsg = wait(0, dialog.GetMessagePort())\n exit while\n end while\n End Function\n```\n\n**Image: roCodeRegistrationScreen example results**\n\n![cdregistscrn1](https://image.roku.com/ZHZscHItMTc2/cdregistscrn1.jpg \"cdregistscrn1\")\n\n![cdregistscrn2](https://image.roku.com/ZHZscHItMTc2/cdregistscrn2.jpg \"cdregistscrn2\")", + "events": [ + { + "name": "roCodeRegistrationScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rocoderegistrationscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifCodeRegistrationScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iifCodeRegistrationScreen.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roCodeRegistrationScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocoderegistrationscreen.md" + }, + "rocompositor": { + "constructors": [ + { + "params": [], + "returnType": "roCompositor" + } + ], + "description": "The roCompositor allows the composition and animation of multiple roBitmaps and roRegions.\n\nThis object can create and manage roSprites in a z-ordered list. The sprites can be of arbitrary size and can be thought of as planes. The compositor can manage collision detection between the sprites, support scrolling the sprite bitmap source, and support animated sprites (multi-frame sprites with frame-flipping animation). You may have multiple roCompositor components, and they can composite onto the same or separate bitmaps. That said, the most common scenario is to have a single roCompositor.\n\n**Example: Scrolling a bitmap**\n\n```\nLibrary \"v30/bslCore.brs\"\nFunction main()\n black=&hFF'RGBA\n screen=CreateObject(\"roScreen\")\n compositor=CreateObject(\"roCompositor\")\n compositor.SetDrawTo(screen, black)\n http = CreateObject(\"roUrlTransfer\")\n http.SetMessagePort(CreateObject(\"roMessagePort\"))\n http.SetUrl(\"http://rokudev.roku.com/rokudev/examples/scroll/VeryBigPng.png\")\n http.AsyncGetToFile(\"tmp:/VeryBigPng.png\")\n wait(0, http.GetPort())\n bigbm=CreateObject(\"roBitmap\",\"tmp:/VeryBigPng.png\")\n region=CreateObject(\"roRegion\", bigbm, 0, 0, 1280, 720)\n region.SetWrap(True)\n\n view_sprite=compositor.NewSprite(0, 0, region)\n compositor.draw()\n screen.SwapBuffers()\n msgport = CreateObject(\"roMessagePort\")\n screen.SetMessagePort(msgport)\n codes = bslUniversalControlEventCodes()\n While True\n msg=wait(0, msgport) ' wait for a button\n print \"Msg: \"; type(msg); \" event: \"; msg.GetInt()\n If type(msg)=\"roUniversalControlEvent\" Then\n If msg.GetInt()=codes.BUTTON_UP_PRESSED Then\n Zip(screen, view_sprite, compositor, 0,-4) 'up\n Else If msg.GetInt()=codes.BUTTON_DOWN_PRESSED Then\n Zip(screen, view_sprite, compositor, 0,+4) ' down\n Else If msg.GetInt()=codes.BUTTON_RIGHT_PRESSED Then\n Zip(screen, view_sprite, compositor, +4,0) ' right\n Else If msg.GetInt()=codes.BUTTON_LEFT_PRESSED Then\n Zip(screen, view_sprite, compositor, -4, 0) ' left\n Else If msg.GetInt() = codes.BUTTON_BACK_PRESSED ' back button\n Exit While\n End If\n End If\n End While\nEnd Function\nFunction Zip(screen, view_sprite, compositor, xd, yd)\n For x=1 To 60\n view_sprite.OffsetRegion(xd, yd, 0, 0)\n compositor.draw()\n screen.SwapBuffers()\n End For\nEnd Function\n```", + "events": [], + "interfaces": [ + { + "name": "ifCompositor", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifcompositor.md" + } + ], + "name": "roCompositor", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocompositor.md" + }, + "rodatagramsocket": { + "constructors": [ + { + "params": [], + "returnType": "roDataGramSocket" + } + ], + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets. The interface is modeled on and works much like standard Berkeley sockets.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roDataGramSocket\")`\n\n**Example**\n\n```\n' UDP 2-way peer-to-peer asynchronous comm on port 54321\n' periodically sends out a message to a specific address and port\n' prints any message it receives\nFunction UDPPeer()\n msgPort = createobject(\"roMessagePort\")\n udp = createobject(\"roDatagramSocket\")\n udp.setMessagePort(msgPort) 'notifications for udp come to msgPort\n addr = createobject(\"roSocketAddress\")\n addr.setPort(54321)\n udp.setAddress(addr) ' bind to all host addresses on port 54321\n addr.SetHostName(\"10.1.1.1\")\n udp.setSendToAddress(addr) ' peer IP and port\n udp.notifyReadable(true)\n timeout = 1 * 10 * 1000 ' ten seconds in milliseconds\n deviceName = Createobject(\"roDeviceInfo\").GetFriendlyName()\n message = \"Datagram from \" + deviceName\n udp.sendStr(message)\n continue = udp.eOK()\n While continue\n event = wait(timeout, msgPort)\n If type(event)=\"roSocketEvent\"\n If event.getSocketID()=udp.getID()\n If udp.isReadable()\n message = udp.receiveStr(512) ' max 512 characters\n print \"Received message: '\"; message; \"'\"\n End If\n End If\n Else If event=invalid\n print \"Timeout\"\n udp.sendStr(message) ' periodic send\n End If\n End While\n udp.close() ' would happen automatically as udp goes out of scope\n\nEnd Function\n```\n\n> GetDeviceUniqueId() was deprecated in Spring OS 2019.", + "events": [ + { + "name": "roSocketEvent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappinfo.md" + } + ], + "interfaces": [ + { + "name": "ifSocket", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocket.md" + }, + { + "name": "ifSocketAsync", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketasync.md" + }, + { + "name": "ifSocketCastOption", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketcastoption.md" + }, + { + "name": "ifSocketOption", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketoption.md" + }, + { + "name": "ifSocketStatus", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketstatus.md" + } + ], + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + "rodatetime": { + "constructors": [ + { + "params": [], + "returnType": "roDateTime" + } + ], + "description": "The roDateTime provides an interface to obtain the current date/time for the player and manipulate date/times.\n\nThis component provides several options for obtaining attributes about the date/time. All times are GMT unless they are converted to the system timezone with a call to the method: toLocalTime().\n\nThis object is created with no parameters:\n\n`CreateObject(\"roDateTime\")`\n\nThe date/time of the object is set to the current system time when the object is created. The date/time represented by the object can be changed by calling Mark(), FromSeconds(), or FromISO8601String().\n\n**Example**\n\n```\ndate = CreateObject(\"roDateTime\")\nprint \"The date is now \"; date.AsDateString(\"long-date\")\n```", + "events": [], + "interfaces": [ + { + "name": "ifDateTime", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdatetime.md" + } + ], + "name": "roDateTime", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatetime.md" + }, + "rodevicecrypto": { + "constructors": [ + { + "params": [], + "returnType": "roDeviceCrypto" + } + ], + "description": "The roDeviceCrypto component enables you to encrypt and decrypt data on a device using a key that is unique per channel, device, or model. Using a channel key for example, you can encrypt data for a channel so that it may only be decrypted by that same channel. In this case, you could provision credentials or an API key from the cloud to devices securely. With a device key for example, you could implement a secure-storage like algorithm.", + "events": [], + "interfaces": [ + { + "name": "ifDeviceCrypto", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdevicecrypto.md" + } + ], + "name": "roDeviceCrypto", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodevicecrypto.md" + }, + "rodeviceinfo": { + "constructors": [ + { + "params": [], + "returnType": "roDeviceInfo" + } + ], + "description": "The Device Info provides an interface to obtain attributes about the device.\n\nThese attributes are not changeable by the script, but may be queried to obtain values for the player. It is common for scripts to need access to the serial number and model info for negotiating with backend services.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roDeviceInfo\")`\n\n**Example**\n\n```\ndi = CreateObject(\"roDeviceInfo\")\nprint di.GetModel()\nprint di.GetVersion()\nprint di.GetChannelClientId()\n```\n\n**Output**\n\nThe output from the above code would like the following:\n\n```\n N1000\n 999.99E99999X\n 20E825000036\n```", + "events": [ + { + "name": "roDeviceInfoEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md" + } + ], + "interfaces": [ + { + "name": "ifDeviceInfo", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdeviceinfo.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roDeviceInfo", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodeviceinfo.md" + }, + "rodouble": { + "constructors": [], + "description": "roDouble is the object equivalent for intrinsic type 'Double'.\n\nIt is a legacy object name, corresponding to the intrinsic Double object. Applications should use Double literal values and/or Double-typed variables directly.", + "events": [], + "interfaces": [ + { + "name": "ifDouble", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdouble.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roDouble", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodouble.md" + }, + "roevpcipher": { + "constructors": [], + "description": "The EVP Cipher component provides an interface to the OpenSSL EVP library of symmetric cipher commands. The EVP library provides a high-level interface to cryptographic functions to implement digital \"envelopes\".\n\nThese commands allow data to be encrypted or decrypted using various block and stream ciphers using keys based on passwords or explicitly provided.\n\nSome of the ciphers do not have large keys and others have security implications if not used correctly. A beginner is advised to just use a strong block cipher in CBC mode such as bf or des3. All the block ciphers normally use PKCS#5 padding also known as standard block padding. If padding is disabled then the input data must be a multiple of the cipher block length.\n\n> For additional information on the OpenSSL library of symmetric ciphers see: [https://www.openssl.org/docs/manmaster/man1/enc.html](https://www.openssl.org/docs/manmaster/man1/enc.html).\n\n**List of supported ciphers**\n\n| Name | Cipher | Key size (bits) | Block size (bits) |\n| --- | --- | --- | --- |\n| bf-cbc | Blowfish in CBC mode | 128 | 64 |\n| bf | Alias for bf-cbc | 128 | 64 |\n| bf-cfb | Blowfish in CFB mode | 128 | 64 |\n| bf-ecb | Blowfish in ECB mode | 128 | 64 |\n| bf-ofb | Blowfish in OFB mode | 128 | 64 |\n| des-cbc | DES in CBC mode | 56 | 64 |\n| des | Alias for des-cbc | 56 | 64 |\n| des-cfb | DES in CBC mode | 56 | 64 |\n| des-ecb | DES in ECB mode | 56 | 64 |\n| des-ofb | DES in OFB mode | 56 | 64 |\n| des-ede-cbc | Two key triple DES EDE in CBC mode | 80 | 64 |\n| des-ede | Two key triple DES EDE in ECB mode | 80 | 64 |\n| des-ede-cfb | Two key triple DES EDE in CFB mode | 80 | 64 |\n| des-ede-ofb | Two key triple DES EDE in OFB mode | 80 | 64 |\n| des-ede3-cbc | Three key triple DES EDE in CBC mode | 112 | 64 |\n| des-ede3 | Three key triple DES EDE in ECB mode | 112 | 64 |\n| des3 | Alias for des-ede3-cbc | 112 | 64 |\n| des-ede3-cfb | Three key triple DES EDE in CFB mode | 112 | 64 |\n| des-ede3-ofb | Three key triple DES EDE in OFB mode | 112 | 64 |\n| desx | DESX algorithm | approx. 119 | 64 |\n| desx-cbc | DESX in CBC mode | approx. 119 | 64 |\n| aes-\\[128/192/256\\]-cbc | 128/192/256 bit AES in CBC mode | 128,192,256 | 128 |\n| aes-\\[128/192/256\\] | Alias for aes-\\[128/192/256\\]-cbc | 128,192,256 | 128 |\n| aes-\\[128/192/256\\]-cfb | 128/192/256 bit AES in 128 bit CFB mode | 128,192,256 | 128 |\n| aes-\\[128/192/256\\]-cfb1 | 128/192/256 bit AES in 1 bit CFB mode | 128,192,256 | 128 |\n| aes-\\[128/192/256\\]-cfb8 | 128/192/256 bit AES in 8 bit CFB mode | 128,192,256 | 128 |\n| aes-\\[128/192/256\\]-ecb | 128/192/256 bit AES in ECB mode | 128,192,256 | 128 |\n| aes-\\[128/192/256\\]-ofb | 128/192/256 bit AES in OFB mode | 128,192,256 | 128 |", + "events": [], + "interfaces": [ + { + "name": "ifEVPCipher", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifevpcipher.md" + } + ], + "name": "roEVPCipher", + "url": "https://developer.roku.com/docs/references/brightscript/components/roevpcipher.md" + }, + "roevpdigest": { + "constructors": [ + { + "params": [], + "returnType": "roEVPDigest" + } + ], + "description": "The EVP Digest component provides an interface to the OpenSSL EVP library of message digest algorithms. The EVP library provides a high-level interface to cryptographic hash functions.\n\nroEVPDigest processes an arbitrary amount of data and generates a hash of the data, using a selected algorithm.\n\n> For additional information on the OpenSSL library of message digest algorithms see: [http://www.openssl.org/docs/apps/dgst.html](http://www.openssl.org/docs/apps/dgst.html)\n\n**List of Supported Digest Algorithms**\n\n* md5 - MD5 message digest algorithm (default)\n* sha1 - SHA-1 message digest algorithm\n* sha224 - SHA-2, 224 bit variant\n* sha256 - SHA-2, 256 bit variant\n* sha384 - SHA-2, 384 bit variant\n* sha512 - SHA-2, 512 bit variant\n\n**Example: SHA1 Message Digest with roEVPDigest**\n\n```\nba = CreateObject(\"roByteArray\")\n' ...populate bytearray...\ndigest = CreateObject(\"roEVPDigest\")\ndigest.Setup(\"sha1\")\nresult = digest.Process(ba)\nprint result\n```\n\n**Example: MD5 Message Digest with roEVPDigest**\n\n```\nba1 = CreateOjbect(\"roByteArray\")\n' ...populate ba1...\nba2 = CreateObject(\"roByteArray\")\nba2.FromAsciiString(somestring)\ndigest = CreateObject(\"roEVPDigest\")\ndigest.Setup(\"md5\")\ndigest.Update(ba1)\ndigest.Update(ba2)\nresult = digest.Final()\nprint result\n```", + "events": [], + "interfaces": [ + { + "name": "ifEVPDigest", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifevpdigest.md" + } + ], + "name": "roEVPDigest", + "url": "https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md" + }, + "rofilesystem": { + "constructors": [ + { + "params": [], + "returnType": "roFileSystem" + } + ], + "description": "The roFilesystem component implements common filesystem inspection and modificationroutines.\n\nAll paths are matched case-insensitively, regardless of the case-sensitivity of the underlying filesystem. The supported character set is limited to only those characters supported by vfat filesystems (valid Windows characters). The usbplayer sample application is a good example of roFileSystem usage. USB devices with VFAT, NTFS, HFS, and HFS Plus filesystems are supported. The USB filesystems are currently mounted read only.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roFileSystem\")`", + "events": [ + { + "name": "roFileSystemEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rofilesystemevent.md" + } + ], + "interfaces": [ + { + "name": "ifFileSystem", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffilesystem.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roFileSystem", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofilesystem.md" + }, + "rofloat": { + "constructors": [], + "description": "roFloat is the object equivalent for intrinsic type 'Float'.\n\nThis is useful in the following situations:\n\n* If any object exposes the ifFloat interface, that object can be used in any expression that expects an intrinsic value.", + "events": [], + "interfaces": [ + { + "name": "ifFloat", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffloat.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roFloat", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofloat.md" + }, + "rofont": { + "constructors": [], + "description": "roFont represents a particular font, from a font-family (eg. Arial), with a particular pixel size (e.g 20), and a particular boldness or italicness.\n\nIt is used in conjunction with [roFontRegistry](https://developer.roku.com/docs/references/brightscript/components/rofontregistry.md\"roFontRegistry\") to create and manage fonts. Font files are registered with roFontRegistry and then various methods in roFontRegistry can be used to create roFont objects. Applications should not create roFonts with CreateObject() but should always use roFontRegistry to create them. roFont objects in turn can be used with [ifDraw2D.DrawText](https://developer.roku.com/docs/references/brightscript/interfaces/ifdraw2d.mddrawtextrgba-as-integer-x-as-integer-y-as-integer-text-as-string-font-as-object-as-boolean \"ifDraw2D.DrawText\") to draw text on the screen or into bitmaps.\n\n**Example**\n\n```\nscreen = CreateObject(\"roScreen\")\nwhite = &hFFFFFFFF\nblue = &h0000FFFF\nfont_registry = CreateObject(\"roFontRegistry\")\nfont = font_registry.GetDefaultFont()\n\n' Draw white text in a blue rectangle\ntext = \"Hello world\"\nw = font.GetOneLineWidth(text, screen.GetWidth())\nh = font.GetOneLineHeight()\nx = 200\ny = 100\nborder = 8\nscreen.DrawRect(x, y, w + 2*border, h + 2*border, blue)\nscreen.DrawText(text, x+border, y+border, white, font)\n```", + "events": [], + "interfaces": [ + { + "name": "ifFont", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffont.md" + } + ], + "name": "roFont", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofont.md" + }, + "rofontmetrics": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "font", + "type": "String" + } + ], + "returnType": "roFontMetrics" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "dynamic" + } + ], + "returnType": "roFontMetrics" + } + ], + "deprecatedDescription": "This class is deprecated. Developers should use [roFont](/docs/references/brightscript/components/roFont.md \"roFont\") methods (GetOneLineHeight and GetOneLineWidth).\n", + "description": "> This class is deprecated. Developers should use [roFont](https://developer.roku.com/docs/references/brightscript/components/roFont.md\"roFont\") methods (GetOneLineHeight and GetOneLineWidth).\n\nThe roFontMetrics object allows you to get display size information for a specific font returned by the roFontRegistry.Get() method.\n\nIn order to use this object, you must first initialize the roFontMetrics object with a font name that had been previously registered with the roFontRegistry, then the total rendered size of strings in that font can be returned by roFontMetrics.Size().\n\nThis object is created with a string that represents the font to use in its size calculations:\n\n`CreateObject(\"roFontMetrics\", String font)`\n\n**Example: Simple use of roFontRegistry and roFontMetrics to render a string on the roImageCanvas**\n\n```\nhelloString = \"Hello ImageCanvas\"\n\nfontReg = CreateObject(\"roFontRegistry\")\nfontReg.Register(\"pkg:/fonts/LCDMono.ttf\")\nfont = fontReg.Get(\"LCDMono\",36,50,false) ' 36pt, 50 is normal\n ' weight, no italics\n\nfontMetrics = CreateObject(\"roFontMetrics\", font)\nstringSize = fontMetrics.size(helloString)\n\ncanvasItem = { \n Text:helloString\n TextAttrs:{Color:\"#FFCCCCCC\", Font:font, \n HAlign:\"HCenter\",\n VAlign:\"VCenter\", Direction:\"LeftToRight\"}\n TargetRect:{x:390,y:357, w:stringSize.w,h:stringSize.h}\n}\n\ncanvas = CreateObject(\"roImageCanvas\")\nport = CreateObject(\"roMessagePort\")\ncanvas.SetMessagePort(m.port)\n'Set opaque background\ncanvas.SetLayer(0, {Color:\"#FF000000\", CompositionMode:\"Source\"})\ncanvas.SetRequireAllImagesToDraw(true)\ncanvas.SetLayer(1, canvasItem)\ncanvas.Show()\n```", + "events": [], + "interfaces": [ + { + "name": "ifFontMetrics", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffontmetrics.md" + } + ], + "isDeprecated": true, + "name": "roFontMetrics", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofontmetrics.md" + }, + "rofontregistry": { + "constructors": [ + { + "params": [], + "returnType": "roFontRegistry" + } + ], + "description": "The roFontRegistry object allows you to create roFont objects, either using the default font or using fonts in TrueType or OpenType files packaged with your application.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roFontRegistry\")`\n\n**Example**\n\n```\nreg = CreateObject(\"roFontRegistry\")\nfont = reg.GetDefaultFont(30, false, false)\nscreen = CreateObject(\"roScreen\")\nscreen.DrawText(\"hello world\", 100, 100, &hFFFFFFFF, font)\n```\n\n**Example using a font file**\n\n```\nreg.Register(\"pkg:/fonts/myfont.ttf\")\nfont = reg.GetFont(\"MyFont\", 30, false, false)\nscreen = CreateObject(\"roScreen\")\nscreen.DrawText(\"hello world\", 100, 100, &hFFFFFFFF, font)\n```\n\nFont files can quickly get very large, so be conscious of the size of the font files you include with your application. You should be able to find very good font files that are 50k or less. Anything larger is probably too big. The customvideoplayer sample application is a good example of usage.", + "events": [], + "interfaces": [ + { + "name": "ifFontRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffontregistry.md" + } + ], + "name": "roFontRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofontregistry.md" + }, + "rofunction": { + "constructors": [], + "description": "roFunction is the object equivalent for intrinsic type Function.", + "events": [], + "interfaces": [ + { + "name": "ifFunction", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffunction.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roFunction", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofunction.md" + }, + "rogridscreen": { + "constructors": [ + { + "params": [], + "returnType": "roGridScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Grid Screen provides a graphical display of poster art from multiple content categories from within a single screen.\n\nUsers can browse within a category list by scrolling horizonally and between category lists by scrolling vertically. There is an optional callout box in the lower right corner of the screen that can display details about the focused item without leaving the screen. Each item in the grid screen is represented by an image (aka poster), so any type of item that can be visually represented by an image can be displayed in the poster screen. It is used to show lists of data to users and common patterns include content categories, movies, podcasts, pictures, and search results. The initial release of roGridScreen only enabled the default list style, \"portrait\", using the following art sizes:\n\n* Artwork sizes: SD=110x150; HD=210x270\n\nIt also required grid posters to be .jpg files.\n\nLater Roku OS versions added mixed aspect ratio grids, and the [ifGridScreen](https://developer.roku.com/docs/references/brightscript/interfaces/ifgridscreen.md\"ifGridScreen\") interface [SetListPosterStyles()](https://developer.roku.com/docs/references/brightscript/interfaces/ifgridscreen.mdsetlistposterstylesstyles-as-object-as-void \"SetListPosterStyles()\") to set the aspect ratio of each row in the grid. If you want a mixed aspect ratio grid, you must call SetListPosterStyles() before you call [SetContentList()](https://developer.roku.com/docs/references/brightscript/interfaces/ifgridscreen.mdsetcontentlistrowindex-as-integer-contentlist-as-object-as-void \"SetContentList()\"), to avoid possible distortion of the graphic images in the grid.\n\n**Since Roku OS version 2.8**\n\nFile types of .png and .gif files are now supported, though they are converted internally to .jpg by the roGridScreen so they have a performance penalty.\n\nIn v2.8, there are now multiple grid styles that are specified in the SetGridStyle()method below. It's also worth going back and reviewing the appManager theme parameters in [roAppManager](https://developer.roku.com/docs/references/brightscript/components/roappmanager.md\"roAppManager\"), as v2.8 adds some new grid parameters. The border around the focused poster screen can be customized with the GridScreenFocusBorder images in png format. PNG files can have a transparent color value that you will need to allow the focused poster image to show through the border image. The corresponding offsets should be negative offsets that would be up and to the left of the top left corner of the poster image. The width of the borders should be the absolute values of the offsets and the rest of the image should be transparent inside. The GridScreenDescriptionImage is also positioned relative to the top left corner of the focused image. It can be positioned up and to the left with negative x and y offsets, below and to the right with positive offsets, or in the other corners with mixed signed x and y offsets. It's recommended that you include a \"callout\" tip pointing to the focused image in the GridScreenDescriptionImage.\n\n**Diagram: roGridScreen**\n\n![Diagram: roGridScreen](https://image.roku.com/ZHZscHItMTc2/roGridScreen.png \"roGridScreen\")\n\nThis object is created with no parameters:\n\n`CreateObject(\"roGridScreen\")`\n\n**Example**\n\n```\nFunction Main()\n port = CreateObject(\"roMessagePort\")\n grid = CreateObject(\"roGridScreen\")\n grid.SetMessagePort(port)\n rowTitles = CreateObject(\"roArray\", 10, true)\n for j = 0 to 10\n rowTitles.Push(\"[Row Title \" + j.toStr() + \" ] \")\n end for\n grid.SetupLists(rowTitles.Count())\n grid.SetListNames(rowTitles)\n for j = 0 to 10\n list = CreateObject(\"roArray\", 10, true)\n for i = 0 to 10\n o = CreateObject(\"roAssociativeArray\")\n o.ContentType = \"episode\"\n o.Title = \"[Title\" + i.toStr() + \"]\"\n o.ShortDescriptionLine1 = \"[ShortDescriptionLine1]\"\n o.ShortDescriptionLine2 = \"[ShortDescriptionLine2]\"\n o.Description = \"\"\n o.Description = \"[Description] \"\n o.Rating = \"NR\"\n o.StarRating = \"75\"\n o.ReleaseDate = \"[ For additional information on the OpenSSL HMAC functions, please see: [http://www.openssl.org/docs/crypto/hmac.html](http://www.openssl.org/docs/crypto/hmac.html)\n\n**Supported Digest Algorithms**\n\nThe supported digest algorithms are the same as those supported by [roEVPDigest](https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md\"roEVPDigest\").\n\n**Example**\n\n```\nhmac = CreateObject(\"roHMAC\")\nsignature_key = CreateObject(\"roByteArray\")\nsignature_key.fromAsciiString(getKey())\nIf hmac.setup(\"sha1\", signature_key) = 0\n message = CreateObject(\"roByteArray\")\n message.fromAsciiString(getMessage())\n result = hmac.process(message)\n print result.toBase64String()\nEnd If\n\n\nhmac = CreateObject(\"roHMAC\")\nsignature_key = CreateObject(\"roByteArray\")\nsignature_key.fromAsciiString(getKey())\nIf hmac.setup(\"sha1\", signature_key) = 0\n message1 = CreateObject(\"roByteArray\")\n message1.fromAsciiString(getMessage1())\n hmac.update(message1)\n message2 = CreateObject(\"roByteArray\")\n message2.fromAsciiString(getMessage2())\n hmac.update(message2)\n result = hmac.final()\n print result.toBase64String()\nEnd If\n```", + "events": [], + "interfaces": [ + { + "name": "ifHMAC", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhmac.md" + } + ], + "name": "roHMAC", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohmac.md" + }, + "rohttpagent": { + "constructors": [ + { + "params": [], + "returnType": "roHttpAgent" + } + ], + "description": "All SceneGraph nodes can use the roHttpAgent component to support cookies, custom HTTP headers, and support secure HTTP file transfer protocols, such as passing certificates to the server as part of a URL transfer. An roHttpAgent component object is created by default for all SceneGraph nodes for this purpose. The roHttpAgent object supports the [ifHttpAgent](https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md\"ifHttpAgent\") interface used by many BrightScript components to allow secure HTTP file transfer protocols. Child nodes of a SceneGraph node automatically inherit the parent roHttpAgent object, unless a new roHttpAgent object is created, or an existing roHttpAgent is set for a child node. There are two roSGNode [ifSGNodeHttpAgentAccess](https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodehttpagentaccess.md\"ifSGNodeHttpAgentAccess\") interface methods that allow a specific roHttpAgent object to be selected and set for a specific SceneGraph node.\n\nAn roHttpAgent object is created automatically for all SceneGraph nodes, or can be created with no parameters:\n\n`CreateObject(\"roHttpAgent\")`\n\n> SceneGraph Audio and Video nodes always create a new roHttpAgent object and do not share it, and can use a different mechanism for HTTPS and cookie support, that involves setting certificates and cookies as Content Meta-Data attributes for the node ContentNode node.", + "events": [], + "interfaces": [ + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + } + ], + "name": "roHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohttpagent.md" + }, + "roimagecanvas": { + "constructors": [ + { + "params": [], + "returnType": "roImageCanvas" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roImageCanvas component provides an interface to render graphic elements at specific spots on the screen.\n\nAlthough it is not intended to be a full-fledged graphics component for high-performance gaming, it does provide a simple interface for building custom animations out of arrays of images displayed on the screen.\n\nAn item (graphical element) may be one of three types: image, text, or colored rectangle. The item type is determined by the [Content Meta-data](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-data\") fields set on the item.", + "events": [ + { + "name": "roImageCanvasEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roimagecanvasevent.md" + } + ], + "interfaces": [ + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifImageCanvas", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifimagecanvas.md" + } + ], + "isDeprecated": true, + "name": "roImageCanvas", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md" + }, + "roimagemetadata": { + "constructors": [], + "description": "The roImageMetadata component provides developers access to image file metadata included in many .jpg EXIF headers.\n\nroImageMetadata currently only works with local file Urls.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roImageMetadata\")`", + "events": [], + "interfaces": [ + { + "name": "ifImageMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifimagemetadata.md" + } + ], + "name": "roImageMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagemetadata.md" + }, + "roinput": { + "constructors": [ + { + "params": [], + "returnType": "roInput" + } + ], + "description": "An roInput object can be used to receive events sent from a network client using the External Control Protocol (ECP), as described in [External Control API](/docs/developer-program/debugging/external-control-api.md \"External Control API\").\n\n> The [supports\\_input\\_launch manifest flag](/docs/developer-program/getting-started/architecture/channel-manifest.md#launch-requirement-attributes) must be set for channels to accept deep linking parameters when already running. This flag enables deep linking into content without relaunching the channel. See the [Deep Linking sample channel](https://github.com/rokudev/deep-Linking-samples) for how to use roInput to handle deep links into content while the channel is already running.\n\nRefer to [External Control Service Commands](/docs/developer-program/debugging/external-control-api.md#external-control-service-commands \"External Control Service Commands\") for information about the ECP input command.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roInput\")`\n\n**Example**\n\nThe following prints information received from an external device in JSON format. If the external device sends the following input command:\n\n```\ncurl -d '' ':8060/input?my_event=My%20Test&x=100&y=200&action=start'\n```\n\nThe following will be printed:\n\n```\n{\"action\":\"start\",\"my_event\":\"My Test\",\"x\":\"100\",\"y\":\"200\"}\n```\n\n**roInput Example**\n\n```\nmsgPort = CreateObject(\"roMessagePort\")\n\ninput = CreateObject(\"roInput\")\ninput.SetMessagePort(msgPort)\n\nprint \"Waiting for messages...\"\nwhile true\n msg = wait(0, msgPort)\n if type(msg) = \"roInputEvent\"\n if msg.IsInput()\n info = msg.GetInfo()\n print \"Received input: \"; FormatJSON(info)\n end if\n end if\nend while\n```", + "events": [ + { + "name": "roInputEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roinputevent.md" + } + ], + "interfaces": [ + { + "name": "ifInput", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifinput.md" + } + ], + "name": "roInput", + "url": "https://developer.roku.com/docs/references/brightscript/components/roinput.md" + }, + "roint": { + "constructors": [ + { + "params": [], + "returnType": "roInt" + } + ], + "description": "roInt is the object equivalent for intrinsic type Integer.\n\nThis is useful in the following situations:\n\n* When an object is needed, instead of an intrinsic value. For example, \"roList\" maintains a list of objects. If an Integer is added to roList, it will be automatically wrapped in an roInt by the language interpreter. When a function that expects a BrightScript Component as a parameter is passed an int, BrightScript automatically creates the equivalent BrightScript Component.\n* If any object exposes the ifInt interface, that object can be used in any expression that expects an intrinsic value. For example, in this way an roTouchEvent can be used as an integer whose value is the userid of the roTouchEvent.\n\n> If o is an roInt, then the following statements have the following effects\n> \n> print o ' prints o.GetInt()\n> \n> i%=o ' assigns the integer i% the value of o.GetInt()\n> \n> k=o 'presumably k is dynamic typed, so it becomes another reference to the roInt o\n> \n> o=5 'this is NOT the same as o.SetInt(5). Instead it releases o, and 'changes the type of o to Integer (o is dynamically typed). And assigns it to 5.\n\n**Example**\n\n```\nBrightScript> o=CreateObject(\"roInt\")\nBrightScript> o.SetInt(555)\nBrightScript> print o\n555\nBrightScript> print o.GetInt()\n555\nBrightScript> print o-55\n500\n```", + "events": [], + "interfaces": [ + { + "name": "ifInt", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifint.md" + }, + { + "name": "ifIntOps", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifintops.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roInt", + "url": "https://developer.roku.com/docs/references/brightscript/components/roint.md" + }, + "roinvalid": { + "constructors": [], + "description": "roInvalid is the object equivalent for intrinsic type 'Invalid'.", + "events": [], + "interfaces": [ + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roInvalid", + "url": "https://developer.roku.com/docs/references/brightscript/components/roinvalid.md" + }, + "rokeyboardscreen": { + "constructors": [ + { + "params": [], + "returnType": "roKeyboardScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Keyboard Screen is designed to allow the user to enter an alpha-numeric string for searching, username/password registration or other purposes.\n\nThis component is generally used as part of a sequence of screens and the results are displayed on the subsequent screen in the sequence. In the case of a search screen, results are displayed on the roPosterScreen and categories may be used to segregate TV and Movie results.\n\n**Diagram: roKeyboardScreen**\n\n![**Diagram: roKeyboardScreen**](https://image.roku.com/ZHZscHItMTc2/roKeyboardScreenImage1.png \"roKeyboardScreenImage1\")\n\n**Example**\n\n```\nSub Main()\n screen = CreateObject(\"roKeyboardScreen\")\n port = CreateObject(\"roMessagePort\")\n screen.SetMessagePort(port)\n screen.SetTitle(\"Search Screen\")\n screen.SetText(\"default\")\n screen.SetDisplayText(\"enter text to search\")\n screen.SetMaxLength(8)\n screen.AddButton(1, \"finished\")\n screen.AddButton(2, \"back\")\n screen.Show()\n\n while true\n msg = wait(0, screen.GetMessagePort())\n print \"message received\"\n if type(msg) = \"roKeyboardScreenEvent\"\n if msg.isScreenClosed()\n return\n else if msg.isButtonPressed() then\n print \"Evt:\"; msg.GetMessage ();\" idx:\"; msg.GetIndex()\n if msg.GetIndex() = 1\n searchText = screen.GetText()\n print \"search text: \"; searchText\n return\n endif\n endif\n endif\n end while\nEnd Sub\n```\n\n**Image: roKeyboardScreen example results**\n\n![Image: roKeyboardScreen example results](https://image.roku.com/ZHZscHItMTc2/roKeyboardScreenImage2.png \"roKeyboardScreenImage2\")", + "events": [ + { + "name": "roKeyboardScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rokeyboardscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifKeyboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifkeyboardscreen.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roKeyboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rokeyboardscreen.md" + }, + "rolist": { + "constructors": [ + { + "params": [], + "returnType": "roList" + } + ], + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members. The array operator \\[ \\] can be used to access any element in the ordered list.\n\n**Example**\n\nImplementation:\n\n```\nlist = CreateObject(\"roList\")\nlist.AddTail(\"a\")\nlist.AddTail(\"b\")\nlist.AddTail(\"c\")\nlist.AddTail(\"d\")\nlist.ResetIndex()\nx= list.GetIndex()\nwhile x <> invalid\n print x\n x = list.GetIndex()\nend while\n\n\nprint list[2]\n```\n\nOutput:\n\n```\na\nb\nc\nd\nc \n```", + "events": [], + "interfaces": [ + { + "name": "ifArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarray.md" + }, + { + "name": "ifArrayGet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayget.md" + }, + { + "name": "ifArraySet", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayset.md" + }, + { + "name": "ifEnum", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifenum.md" + }, + { + "name": "ifList", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflist.md" + }, + { + "name": "ifListToArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflisttoarray.md" + } + ], + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + "rolistscreen": { + "constructors": [], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe List Screen provides a graphical display of content in a vertical list within a single screen.\n\nUsers can browse the content by scrolling the text list vertically. The vertical list is displayed on the left side of the screen and the poster is displayed on the right side of the screen. As the user scrolls through the content, the poster is updated with the poster art of the focused list item. There is an optional short description text below the poster that can display the description of the focused item and gets updated as the user scrolls the list.\n\nThe poster art uses the following art sizes:\n\n```\n Artwork sizes: SD=136x124; HD=250x250\n```\n\nroListScreen has a default dark highlight for the focused list item. The highlight can be customized by including a .png file with the following dimensions:\n\n```\n Highlight sizes: SD=304x38; HD=511x54\n```\n\n![List Screen Draft](https://image.roku.com/ZHZscHItMTc2/roListScreen.png \"roListScreen\")", + "events": [ + { + "name": "roListScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rolistscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifListScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflistscreen.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roListScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolistscreen.md" + }, + "rolocalization": { + "constructors": [ + { + "params": [], + "returnType": "roLocalization" + } + ], + "description": "The roLocalization object provides functions to assist in localization. This object provides functions to assist in localization.\n\nIt is created with no parameters:\n\n`CreateObject(\"roLocalization\")`\n\n**Example**\n\n```\nloc = CreateObject(\"roLocalization\")\nimage = loc.GetLocalizedAsset(\"images\", \"splash.png\")\n```", + "events": [], + "interfaces": [ + { + "name": "ifLocalization", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflocalization.md" + } + ], + "name": "roLocalization", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolocalization.md" + }, + "rolonginteger": { + "constructors": [], + "description": "roLongInteger is the object name corresponding to the intrinsic LongInteger object.", + "events": [], + "interfaces": [ + { + "name": "ifLongInt", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflongint.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roLongInteger", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolonginteger.md" + }, + "romessagedialog": { + "constructors": [ + { + "params": [], + "returnType": "roMessageDialog" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Message Dialog displays a formatted, multi-line text message to the user. The dialog may optionally be displayed with a busy animation to indicate progress on a long running operation. The dialog will automatically handle formatting of text and resize to fit. It may also display buttons to get user acknowledgment or a selection choice.\n\nThe following example shows an roMessageDialog with a single done button. When the title, text and button are added, the dialog automatically formats and resizes the dialog as needed for display when Show() is called.\n\n**Diagram: roMessageDialog**\n\n![Diagram: roMessageDialog](https://image.roku.com/ZHZscHItMTc2/roMessageDialog.png \"roMessageDialog\")\n\n**Example**\n\nThe following code example creates a message dialog and displays it to the user. Note that dialogs are not full screen and that the previous screen is dimmed and displays in the background. When the user presses the message dialog button, the dialog is dismissed and the previous screen comes to the foreground. Since dialog.EnableBackButton(true) is also called, the message dialog is dismissed when the remote control's back button is pressed as well. You can of course add additional buttons to your message dialogs that do things other than dismiss the dialog. You would simply need to implement button specific event handling code for these cases in the dlgMsg.isButtonPressed() code block.\n\n```\nFunction ShowMessageDialog() As Void\n port = CreateObject(\"roMessagePort\")\n dialog = CreateObject(\"roMessageDialog\")\n dialog.SetMessagePort(port)\n dialog.SetTitle(\"[Message dialog title]\")\n dialog.SetText(\"[Message dialog text............]\")\n\n dialog.AddButton(1, \"[button text]\")\n dialog.EnableBackButton(true)\n dialog.Show()\n While True\n dlgMsg = wait(0, dialog.GetMessagePort())\n If type(dlgMsg) = \"roMessageDialogEvent\"\n if dlgMsg.isButtonPressed()\n if dlgMsg.GetIndex() = 1\n exit while\n end if\n else if dlgMsg.isScreenClosed()\n exit while\n end if\n end if\n end while\nEnd Function\n```\n\n**Image: roMessageDialog example results**\n\n![roMessageDialog example results](https://image.roku.com/ZHZscHItMTc2/roMessageDialogimage2.png \"roMessageDialogimage2\")", + "events": [ + { + "name": "roMessageDialogEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/romessagedialogevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifMessageDialog", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmessagedialog.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roMessageDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessagedialog.md" + }, + "romessageport": { + "constructors": [ + { + "params": [], + "returnType": "roMessagePort" + } + ], + "description": "A Message Port is the place messages ([events](/docs/developer-program/core-concepts/event-loops.md)) are sent.\n\nWhen using BrightScript, you would not call these functions directly. Instead, use the \"Wait\" BrightScript statement.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roMessagePort\")`", + "events": [], + "interfaces": [ + { + "name": "ifMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmessageport.md" + } + ], + "name": "roMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessageport.md" + }, + "romicrophone": { + "constructors": [], + "description": "The roMicrophone API allows channel applications to receive audio data from the user’s microphone-supported remote control device or mobile phone. When a user initiates recording on their remote control device or mobile phone (via the Roku Mobile App) for the first time within the application, the application will request the user’s permission for the application to access the microphone by displaying a UI dialog box.\n\nThe application will only receive microphone access if the permission is granted by the user.\n\n![microphone-access](https://image.roku.com/ZHZscHItMTc2/microphone-access.jpg \"microphone-access\")\n\nAfter the permission is granted, whenever a user activates the microphone, the application will display a notice informing the user that the microphone is currently being used by the application.\n\nFrom the settings menu (Settings > Privacy > Microphone), the user can revoke microphone permissions from individual applications, at which time the particular application will not be able to access the microphone unless the user re-enables microphone permissions.\n\nFrom the settings menu, the user may also:\n\n* (a) enable universal microphone access permissions for all applications (thereby eliminating the need to request microphone permission on an application by application basis), and\n* (b) prohibit all applications from accessing the microphone.\n\n![microphone-setting](https://image.roku.com/ZHZscHItMTc2/microphone-setting.jpg \"microphone-setting\")\n\nWhen integrating the roMicrophone API, you acknowledge and agree to the following:\n\n* (i) that you will notify your users of your collection, use, and disclosure of any voice recordings or other derived data that you receive through the roMicrophone API;\n* (ii) you will not modify, circumvent, obscure, or otherwise diminish the notices provided by the roMicrophone API to users when they activate or enable microphone recording from their remote control device or mobile phone;\n* (iii) you will not collect any information from, or otherwise activate, the microphone on any remote control device or mobile phone using the roMicrophone API feature without receiving the requisite permissions from the user;\n* (iv) you have and will maintain a legally adequate privacy policy;\n* (v) you have and will maintain all necessary rights and consents from users to use the roMicrophone API features; and\n* (vi) your use of the roMicrophone API features will comply with all applicable laws, rules, and regulations.\n\nYOU FURTHER AGREE YOU WILL NOT USE THE roMicrophone API AND FEATURES IN CONNECTION WITH CONTENT OR CHANNELS DIRECTED TOWARD CHILDREN OR IN CONNECTION WITH USERS KNOWN TO BE CHILDREN. If Roku discovers or determines that you are using the roMicrophone API and features in connection with content or channels directed toward children or with users known to be children, Roku reserves the right to disable or otherwise limit your access to the roMicrophone API feature and related functionality.\n\nYOU MAY NOT ENABLE THE roMicrophone API FEATURES IF YOU DO NOT AGREE TO ABOVE. PLEASE CONTACT ROKU FOR FURTHER INFORMATION. Implementation\n\nThe application should display a focusable button or indicator in the UI that the user selects by pressing and holding the OK button. In response to the OK press event, the application can call:\n\n* [StartRecording()](https://developer.roku.com/docs/references/brightscript/interfaces/ifmicrophone.md\"StartRecording\") - to receive streamed audio data from the microphone asynchronously or\n* [RecordToFile()](https://developer.roku.com/docs/references/brightscript/interfaces/ifmicrophone.md\"RecordToFile()\") - to have the audio data directly captured to a WAV format output file.\n\n> Roku OS will display a HUD to let the user initially consent to be recorded and to subsequently be informed when the microphone is being used. Recording is performed as long as the user holds down the OK button, or until a limit is reached or if an error should occur.", + "events": [ + { + "name": "roMicrophoneEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/romicrophoneevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifMicrophone", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmicrophone.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roMicrophone", + "url": "https://developer.roku.com/docs/references/brightscript/components/romicrophone.md" + }, + "roonelinedialog": { + "constructors": [], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe One Line Dialog is a special type of dialog optimized for single line text.\n\nUnlike the message dialog ([roMessageDialog](https://developer.roku.com/docs/references/brightscript/components/roMessageDialog.md\"roMessageDialog\")) which displays formatted multi-line messages, roOneLineDialog displays a single line of text centered for the user.\n\nThis dialog is optimized for rendering of single-line text strings. It is generally used for displaying text to indicate that an operation is in progress. When the operation completes, the dialog is destroyed and the message dialog disappears.", + "events": [ + { + "name": "roOneLineDialogEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roonelinedialogevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifGetMessagePort.md" + }, + { + "name": "ifOneLineDialog", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifonelinedialog.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roOneLineDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/roonelinedialog.md" + }, + "roparagraphscreen": { + "constructors": [ + { + "params": [], + "returnType": "roParagraphScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Paragraph Screen provides a way to display text and selection choices to the user.\n\nThis type of screen is frequently used for implementing wizard functionality to guide the user through a specific task. The caller may specify header text which is displayed at the top of the screen and one or more paragraphs of text on the screen. In addition, one or more buttons may be added to the screen to get user input or allow navigation. The screen is designed to automatically format the text, headings and buttons and create the photo-fit for them on screen. Some care must be taken to not provide too much text or clipping may occur.\n\n![roParagraphScreen image](https://image.roku.com/ZHZscHItMTc2/roParagraphScreenImage1.png \"roParagraphScreenImage1\")\n\nThis object is created with no parameters:\n\n`CreateObject(\"roParagraphScreen\")`\n\n**Example**\n\n```\nFunction ShowParagraphScreen() As Void\n port = CreateObject(\"roMessagePort\")\n screen = CreateObject(\"roParagraphScreen\")\n screen.SetMessagePort(port)\n screen.SetTitle(\"[Screen Title]\")\n screen.AddHeaderText(\"[Header Text]\")\n screen.AddParagraph(\"[Paragraph text 1 - Text in the paragraph screen is justified to the right and left edges]\")\n screen.AddParagraph(\"[Paragraph text 2 - Multiple paragraphs may be added to the screen by simply making additional calls]\")\n screen.AddButton(1, \"[button text 1]\")\n screen.AddButton(2, \"[button text 2]\")\n screen.Show()\n while true\n msg = wait(0, screen.GetMessagePort())\n if type(msg) = \" roParagraphScreenEvent\"\n exit while\n endif\n end while\nEnd Function\n```\n\n**Image: roParagraphScreen example results**\n\n![Image: roParagraphScreen example results](https://image.roku.com/ZHZscHItMTc2/roParagraphScreenImage2.png \"roParagraphScreenImage2\")", + "events": [ + { + "name": "roParagraphScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roparagraphscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifParagraphScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifparagraphscreen.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roParagraphScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roparagraphscreen.md" + }, + "ropath": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roPath" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "dynamic" + } + ], + "returnType": "roPath" + } + ], + "description": "The roPath component provides developers an easy way to create valid file system paths.\n\nThe roPath component is a convenience class that implements [ifString](https://developer.roku.com/docs/references/brightscript/interfaces/ifstring.md\"ifString\") while providing additional validation and path inspection functionality. See [File System](docs/developer-program/getting-started/architecture/file-system.md \"File System\") for more information about valid path names.\n\nThis object is created with a string that represents the initial path:\n\n`CreateObject(\"roPath\", \"ext1:/vid\")`\n\n**Example**\n\n```\npath = CreateObject(\"roPath\", filename)\nparts = path.Split()\nif parts.phy = \"tmp:\" then print \"this is a temp file\"\nif parts.extension = \".bmp\" then print \"this is a bitmap file\"\n```", + "events": [], + "interfaces": [ + { + "name": "ifPath", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifpath.md" + }, + { + "name": "ifString", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifstring.md" + } + ], + "name": "roPath", + "url": "https://developer.roku.com/docs/references/brightscript/components/ropath.md" + }, + "ropinentrydialog": { + "constructors": [], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Pin Entry Dialog is designed to allow the user to enter a numeric PIN for purchasing content.\n\nUsers establish a PIN on the partner's website for purchasing transactions. The roPinEntryDialog allows the script to present the user with a pop-up, modal dialog for PIN entry and then the script can subsequently call the API's to conclude the purchase transaction. When the last digit is entered, focus jumps to the first button.\n\n**Image: roPinEntryDialog sample**\n\n![roPinEntryDialog](https://image.roku.com/ZHZscHItMTc2/roPinEntryDialog.png \"roPinEntryDialog\")", + "events": [ + { + "name": "roPinEntryDialogEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifPinEntryDialog", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifpinentrydialog.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roPinEntryDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/ropinentrydialog.md" + }, + "roposterscreen": { + "constructors": [ + { + "params": [], + "returnType": "roPosterScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roPosterScreen\")`\n\nThe Poster Screen provides a graphical display of poster art for content selection or can be used as a submenu to provide hierarchical structure to the application.\n\nIn some cases, applications may wish to present a flat single-level list of titles in a queue, but the Poster Screen can also be used at multiple levels in the application to provide hierarchical browsing. It also provides an optional \"filter banner\" for displaying categories representing filtered subsets of the data or categorized groups.\n\nEach item in the poster screen is represented by an image (aka poster), so any type of item that can be visually represented by an image can be displayed in the poster screen. It is used to show lists of data to users and common patterns include content categories, movies, podcasts, and search results.\n\nJust below the overhang is the filter banner. It allows a method of easily selecting or filtering content based on categories. The categories are set by the developer during screen initialization, and the script is notified when a new category is highlighted or selected. Based on the event notification, the script can set the desired content in the view. The filter banner is optional.\n\n**Diagram: roPosterScreen (flat-category)**\n\n![Diagram: roPosterScreen (flat-category)](https://image.roku.com/ZHZscHItMTc2/roPosterScreenImage1.png \"roPosterScreenImage1\")\n\nShortDescriptionLine1 from the content metadata. Generally the category title.\n\nShortDescriptionLine2 from the content metadata. Generally a description for the category.\n\n**Diagram: roPosterScreen (arced-landscape)**\n\n![Diagram: roPosterScreen (arced-landscape](https://image.roku.com/ZHZscHItMTc2/roPosterScreenImage2.png \"roPosterScreenImage2\")\n\n**Diagram: roPosterScreen (arced-portrait)**\n\n![Diagram: roPosterScreen (arced-portrait)](https://image.roku.com/ZHZscHItMTc2/roPosterScreenImage3.png \"roPosterScreenImage3\")\n\n**Diagram: roPosterScreen (flat-episodic)**\n\n![Diagram: roPosterScreen (flat-episodic)](https://image.roku.com/ZHZscHItMTc2/roPosterScreenImage4.png \"roPosterScreenImage4\")\n\nTV content is often displayed as a series of episodes within a season. The flat-episodic screen type provides a standard way to display episodic content, such as a TV series.\n\nThere is also a flat-episodic-16x9-episodic screen type to display episodic content with 16x9 images.\n\nThe paragraph text allows the user to view the synopsis for the currently selected episode. As the user scrolls right/left to select a new episode, the paragraph text and the short description lines are updated to reflect the description of the highlighted episode\n\nIn order to see poster art in the side posters instead of episode numbers, please ensure that the SDPosterUrl and HDPosterUrl are defined for the content and that episodeNumber is not defined for that content. EpisodeNumber overrides the poster URL.\n\n**Example**\n\n```\nFunction Main()\n port = CreateObject(\"roMessagePort\")\n poster = CreateObject(\"roPosterScreen\")\n poster.SetBreadcrumbText(\"[location1]\", \"[location2]\")\n poster.SetMessagePort(port)\n list = CreateObject(\"roArray\", 10, true)\n For i = 0 To 10\n o = CreateObject(\"roAssociativeArray\")\n o.ContentType = \"episode\"\n o.Title = \"[Title]\"\n o.ShortDescriptionLine1 = \"[ShortDescriptionLine1]\"\n o.ShortDescriptionLine2 = \"[ShortDescriptionLine2]\"\n o.Description = \"\"\n o.Description = \"[Description] \"\n o.Rating = \"NR\"\n o.StarRating = \"75\"\n o.ReleaseDate = \"[ Please see the PCRE documentation ([http://www.pcre.org/](http://www.pcre.org/)) for documentation on the PCRE library used for regular expression matching. See the [perlre documentation](http://perldoc.perl.org/perlre.html) for complete documentation of the possible regular expressions this library can parse and match. In general, most Perl compatible regular expressions are supported.\n\nThis object is created with a string that represents the matching-pattern and a string to indicate flags that modify the behavior of the matching operation(s):\n\n`CreateObject(\"roRegex\", \"[a-z]+\", \"i\")`\n\nThe match string (\"\\[a-z\\]+\" in the example above, which matches all lowercase letters) can include most Perl compatible regular expressions documented in the PCRE documentation ([http://www.pcre.org/](http://www.pcre.org/)).\n\nAny combination of the following behavior flags (\"i\" in the example above which modifies to match upper and lowercase letters) is supported:\n\n* \"i\" Case insensitive match\n* \"m\" Multiline mode. The start of line \"^\" and end of line \"$\" constructs match immediately following or before any newline in the subject string as well as the very start and end of the string. Normally, just the start and end of the string would match.\n* \"s\" Sets dot-all mode that includes newline in the \".\\*\" regular expression. This modifier is equivalent to Perl's /s modifier.\n* \"x\" Sets extended mode that ignores whitespace characters except when escaped or inside a character class. Characters between an unescaped # outside a character a character class and the next newline character, inclusive, are also ignored. This modifier is equivalent to Perl's /x modifier.", + "events": [], + "interfaces": [ + { + "name": "ifRegex", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregex.md" + } + ], + "name": "roRegex", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregex.md" + }, + "roregion": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "bitmap", + "type": "Object" + }, + { + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnType": "roRegion" + } + ], + "description": "The roRegion component is used to represent a subsection of a bitmap.\n\nThe region is specified with an x,y, width, and height as well as a time field for use with animated sprites and a wrap field which causes the region to wrap during scrolling. The roRegion is a common parameter used by the drawing functions of [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\"). Wrap and Time are used by [roCompositor](https://developer.roku.com/docs/references/brightscript/components/roCompositor.md\"roCompositor\"). roRegion is also used to specify a pretranslation (x,y) for the draw, rotate, and scale operation. The pretranslation is normally used to specify the center of the region. The scaling operation is controlled by the scalemode specified in the region.\n\nThis object is created with parameters to initialize the x,y coordinates, width, height. If time and wrap are desired, use the SetTime() and SetWrap().\n\n`CreateObject(\"roRegion\", Object bitmap, Integer x, Integer y,Integer width, Integer height)`", + "events": [], + "interfaces": [ + { + "name": "ifRegion", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregion.md" + } + ], + "name": "roRegion", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregion.md" + }, + "roregistry": { + "constructors": [ + { + "params": [], + "returnType": "roRegistry" + } + ], + "description": "The Registry is an area of non-volatile storage where a small number of persistent settings can be stored.\n\nThe Registry provides a means for an application to write and read small amounts of data such as settings, scores, etc. The data persists even if the user exits the application and even if the player is rebooted. Registry data is removed only when the application explicitly removes it, the user uninstalls the application, which remove the registry for the application, or the user performs a factory reset, which removes the registry for all applications.\n\nAccess to the registry is available through the roRegistry object. This object is created with no parameters:\n\n`CreateObject(\"roRegistry\")`\n\nThere is a separate registry for each [developer ID](/docs/developer-program/publishing/packaging-channels.md#step-3-run-the-genkey-utility-to-create-a-signing-key \"developer ID\"). This allows multiple applications to use the registry without being able to read or modify the registry from other applications. If desired, a single registry can be shared across multiple applications by using the same developer ID to package the applications. This is the conventional way that an \"application suite\" with shared preferences and other shared information should work. Each registry is divided into sections which are specified by the developer for organization and grouping of attributes. Methods in ifRegistry are provided to list the sections in the registry and to provide access to the data in each section.\n\n> The maximum size of each zlib-compressed application registry is **16K bytes**. Channels should minimize the amount of data stored in the registry and the frequency in which they update it.\n> \n> Use the **ifRegistry.GetSpaceAvailable()** function to check the number of bytes available in the registry.\n\nThe Registry also supports the use of a special transient registry section. A registry section named \"Transient\" can be used to store attributes that have the lifetime of a single boot. Within a specific boot session, these values will be persistent to the application and stored as any other registry value. Whenever the user reboots the Roku Streaming Player, all \"Transient\" registry sections are removed and the values no longer persist. This technique is useful for caching data to minimize network access, yet still ensuring that this data is always fresh after a system reboot.\n\nThe registry is encrypted, and updates are relatively performance intensive and should be used sparingly. Note that all writes to the registry are delayed, and not committed to non-volatile storage until ifRegistry.Flush() or ifRegistrySection.Flush() is explicitly called. The platform may choose opportune times to flush data on its own, but no application is technically correct unless it explicitly calls Flush() at appropriate times. Flushing the registry is a relatively time-consuming operation, so it should be done as infrequently as possible. The Registry data is stored in a fault tolerant manner by preserving a backup for each write which is automatically rolled-back in the event of a failure.", + "events": [], + "interfaces": [ + { + "name": "ifRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregistry.md" + } + ], + "name": "roRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregistry.md" + }, + "roregistrysection": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "section", + "type": "String" + } + ], + "returnType": "roRegistrySection" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roRegistrySection" + } + ], + "description": "A Registry Section enables the organization of settings within the registry. Different registry sections may have their own keys with the same name. In other words, key names are scoped within the registry section to which they belong.\n\nThis object must be supplied with a \"section\" name on creation. If no such section exists, it will be created. Section names are case sensitive, so sections named \"Settings\" and \"settings\" are two different sections.\n\n`CreateObject(\"roRegistrySection\", section as String)`\n\n**Example: Get and set some user authentication in the registry**\n\n```\nFunction GetAuthData() As Dynamic\n sec = CreateObject(\"roRegistrySection\", \"Authentication\")\n if sec.Exists(\"UserRegistrationToken\")\n return sec.Read(\"UserRegistrationToken\")\n endif\n return invalid\nEnd Function\n\nFunction SetAuthData(userToken As String) As Void\n sec = CreateObject(\"roRegistrySection\", \"Authentication\")\n sec.Write(\"UserRegistrationToken\", userToken)\n sec.Flush()\nEnd Function\n```", + "events": [], + "interfaces": [ + { + "name": "ifRegistrySection", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregistrysection.md" + } + ], + "name": "roRegistrySection", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregistrysection.md" + }, + "rorsa": { + "constructors": [ + { + "params": [], + "returnType": "roRSA" + } + ], + "description": "The RSA component provides an interface to the OpenSSL RSA library of signing algorithms.\n\nThis component can be used to sign/verify using RSA.\n\nTypically, you would use the roEVPDigest component to create a message digest, then use roRSA to sign it.\n\n**Example: RSA signing using SHA1**\n\n```\nba = CreateObject(\"roByteArray\")\n\n' ...populate bytearray...\n\ndigest = CreateObject(\"roEVPDigest\")\ndigest.Setup(\"sha1\")\nhashString = digest.Process(ba)\nhashBA = CreateObject(\"roByteArray\")\nhashBA.FromHexString(hashString)\nrsa = CreateObject(\"roRSA\")\n\n' ... save private key to tmp:/privateKey.txt\n\nrsa.SetPrivateKey(\"tmp:/privateKey.txt\")\nrsa.SetDigestAlgorithm(\"sha1\")\nsignature = rsa.Sign(hashBA)\n```\n\n**Example: RSA verification using SHA1**\n\n```\nrsa = CreateObject(\"roRSA\")\nrsa.SetPublicKey(:tmp:/publicKey.txt\")\nrsa.SetDigestAlgorithm(\"sha1\")\n\n' see hashBA and signature from above example\n\nresult = rsa.Verify(hashBA, signature)\nif (result = 1)\n print \"Verified\"\nelse\n print \"Not verified, result = \" ; result\nend if\n```", + "events": [], + "interfaces": [ + { + "name": "ifRSA", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifrsa.md" + } + ], + "name": "roRSA", + "url": "https://developer.roku.com/docs/references/brightscript/components/rorsa.md" + }, + "roscreen": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "boolean" + }, + { + "isRequired": true, + "name": "param2", + "type": "integer" + }, + { + "isRequired": true, + "name": "param3", + "type": "integer" + } + ], + "returnType": "roScreen" + }, + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "boolean" + } + ], + "returnType": "roScreen" + }, + { + "params": [], + "returnType": "roScreen" + } + ], + "description": "The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from.\n\nYou will need at least one roScreen component in your 2D game application to draw on and get events from. The origin (0,0) is the top left corner of the screen. The pixels are always RGBA 32 bits. Multiple roScreen components stack, and like other screen components only the top screen is viewable and gets events. An roScreen that is not the top most screen can still be drawn to.\n\n> Once an roScreen is created, the display stack enters \"Game Mode\", and other screen components cannot be used. Screensavers will also be disabled and will appear as a black screen in its place. Other screen components cannot be intermixed with roScreens as the roScreen display stack is maintained independently from the main screen component display stack. When the final roScreen component is closed, other screen components can be used again.\n\nWhen the roScreen constructor is called, an optional double buffer flag, and an optional resolution can be passed. If the screen is double buffered, two buffers are fully allocated when CreateObject() succeeds. If the screen is single buffered only one buffer is allocated and the \"front\" and \"back\" buffers in method descriptions below are the same buffer. When a screen is created with a different resolution than the output display, it is scaled automatically to the output display resolution.\n\nTo maintain proper aspect ratio, and take care of the different pixel aspect ratio in HD vs SD; there is a fixed set of bitmap resolutions that are allowed to be created for screens:\n\n**HD mode screensizes**\n\n* 1280x720PAR=1:1 (default for HD)\n* 854x480 PAR=1:1 useful for higher performance HD games, also for 640x480 games\n* 940x480 PAR=1.1:1 used for displaying a RokuSD (720x480) games\n\n**SD mode screensizes**\n\n* 720x480 PAR=1.1:1 (default for SD)\n* 640x480 PAR=1:1 (used for 640x480 games)\n* 854x626 PAR=1:1 (used for 854x480 HD games)\n\nThe screen dimensions correspond to the drawable area that applications see. The dimensions were chosen so that applications do not need to compensate for screen aspect ratio or pixel aspect ratio.\n\nIt's likely that when porting games from other platforms, the active game area may be smaller and correspond to more traditional dimensions. In this case, the application can supply letterbox or pillarbox artwork and use an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") to define the active area. The roRegion will translate and clip graphics to the proper area for the game. Similarly, roRegions are used to describe the left and right pillars for an SD game in HD mode, or the upper and lower letterbox regions for an HD game in SD mode.\n\nGames that require more performance should use smaller dimensions. Games should run in HD and SD mode. The screensizes HD 854x480 paired with SD 854x626 and HD 940x480 paired with SD 720x480 were designed for this purpose.\n\nThe game creates a single active game roRegion to do all graphics operations in. roRegions for pillar or letter boxes are used to fill the rest of the screen area depending on if the app is in HD or SD mode. Please refer to the dfSetupDisplayRegions() function in [v30/bslDefender.brs](https://developer.roku.com/docs/references/brightscript/language/component-architecture.mdv30bslcorebrs \"v30/bslDefender.brs\") for help in setting up the drawable regions in screen scaling.\n\nThere are some useful rules of thumb to be aware of to get the best performance when rendering your games:\n\n* Alpha enabled regions are expensive to render\n\nIt is a requirement that the destination be alpha enabled in order for non-rectangular sprites to be properly rendered with transparency. However the sprite used for a background would typically have all pixels be fully nontransparent. Since alpha blending is expensive, a quick way to blit the background in this scenario is to first disable alpha on the screen, manually draw the background, and then enable alpha for the screen before drawing the rest of the sprites.\n\n* Use smaller resolution images wherever possible. Scaling a large image down at run time is expensive with no benefit to the user\n* Rendering text with DrawText() is expensive\n\nFortunately, many of these calls are redundant and can be eliminated. The static text for a particular level can be drawn on the background once and this newly created background can be used for refreshing the screen. This will eliminate almost all text redraws.\n\nA screen can be created with one of three constructors. If it is created with no parameters, the screen will be single buffered, and its output resolution will match the current display resolution (if the current resolution is specified in the manifest file ui\\_resolutions entry, otherwise the size will be 720p).\n\n`CreateObject(\"roScreen\")`\n\nIf a single parameter is passed, it is a Boolean that indicates if the screen is double buffered or not. See SwapBuffers():\n\n`CreateObject(\"roScreen\", true) ' double buffered screen`\n\nIf four parameters are passed, the last two specify the screen's resolution. The dimensions must be one of the screen sizes specified above:\n\n`CreateObject(\"roScreen\", true, 720, 480) ' db & SD res`\n\n**Example: Display an image**\n\n```\nScreen=CreateObject(\"roScreen\")\ndfDrawImage(screen, \"myphoto.jpg\",0,0)\nScreen.Finish()\n```\n\n**Example: Alpha blending**\n\n```\nwhite=&hFFFFFFFF\nscreen0=CreateObject(\"roScreen\")\nscreen0.SetAlphaEnable(true)\nscreen0.Clear(white)\nscreen0.DrawRect(100,100, screen0.GetWidth()-200, screen0.GetHeight()-200, &h80)\n' &h80 is black with a 50% alpha mix (RGBA)\nscreen0.finish()\n```", + "events": [ + { + "name": "roUniversalControlEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rouniversalcontrolevent.md" + } + ], + "interfaces": [ + { + "name": "ifDraw2D", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdraw2d.md" + }, + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifscreen.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + }, + "rosearchhistory": { + "constructors": [ + { + "params": [], + "returnType": "roSearchHistory" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Search History object implements the system wide storage of search terms for use in implementing the roSearchScreen.\n\nAs the user searches for content, recent searches are placed into the search history. This allows the user to easily re-execute these commands later without typing on the keyboard. The initial list of recent searches is displayed on the roSearchScreen to assist the user in finding content to watch. This history is used system wide, so that the user can find references to their search in multiple types of content.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roSearchHistory\")`\n\n**Example**\n\n```\nhistory = CreateObject(\"roSearchHistory\")\nlist = history.GetAsArray()\nprint \"There are \"; list.Count(); \" items in the history\"\n```", + "events": [], + "interfaces": [ + { + "name": "ifSearchHistory", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsearchhistory.md" + } + ], + "isDeprecated": true, + "name": "roSearchHistory", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosearchhistory.md" + }, + "rosearchscreen": { + "constructors": [ + { + "params": [], + "returnType": "roSearchScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Search Screen provides a standard way to allow users to enter text for searching.\n\nThis screen features a simplified keyboard (a-z, 0-9) designed to provide just the keys necessary to perform case-insensitive searches without punctuation.\n\nIdeally, the user would enter a search string and the backend service would perform that query in a case-insensitive manner ignoring special characters like punctuation. The script is notified as each key is pressed so that a progress disclosure search can be performed if supported by the back-end service. In addition, the script can control the text displayed on the screen and will receive events when the text entry is complete.\n\nIn addition to entering search strings, this screen features a list that can be used to display search results or show the most recent searches. It's desirable for the screen to maintain a list of recent searches for the user to allow them to easily repeat a recent query without typing. In some implementations, it may be desirable to use this list to show a progressive set of results after each character while the user is typing.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roSearchScreen\")`\n\n**Example**\n\n```\nREM ******************************************************\nREM Main routine - example of search screen usage\nREM ******************************************************\nSub Main()\n print \"start\"\n 'toggle the search suggestions vs. search history behavior\n 'this allow you to generate both versions of the example below\n displayHistory = false\n history = CreateObject(\"roArray\", 1, true)\n 'prepopulate the search history with sample results\n history.Push(\"seinfeld\")\n history.Push(\"fraiser\")\n history.Push(\"cheers\")\n port = CreateObject(\"roMessagePort\")\n screen = CreateObject(\"roSearchScreen\")\n 'commenting out SetBreadcrumbText() hides breadcrumb on screen\n screen.SetBreadcrumbText(\"\", \"search\")\n screen.SetMessagePort(port)\n if displayHistory\n screen.SetSearchTermHeaderText(\"Recent Searches:\")\n screen.SetSearchButtonText(\"search\")\n screen.SetClearButtonText(\"clear history\")\n screen.SetClearButtonEnabled(true) 'defaults to true\n screen.SetSearchTerms(history)\n else\n screen.SetSearchTermHeaderText(\"Suggestions:\")\n screen.SetSearchButtonText(\"search\")\n screen.SetClearButtonEnabled(false)\n endif\n print \"Doing show screen...\"\n screen.Show()\n print \"Waiting for a message from the screen...\"\n ' search screen main event loop\n done = false\n while done = false\n msg = wait(0, screen.GetMessagePort())\n if type(msg) = \"roSearchScreenEvent\"\n if msg.isScreenClosed()\n print \"screen closed\"\n done = true\n else if msg.isCleared()\n print \"search terms cleared\"\n history.Clear()\n else if msg.isPartialResult()\n print \"partial search: \"; msg.GetMessage()\n if not displayHistory\n screen.SetSearchTerms(GenerateSearchSuggestions(msg.GetMessage()))\n endif\n else if msg.isFullResult()\n print \"full search: \"; msg.GetMessage()\n history.Push(msg.GetMessage())\n if displayHistory\n screen.AddSearchTerm(msg.GetMessage())\n end if\n 'uncomment to exit the screen after a full search result:\n 'done = true\n else\n print \"Unknown event: \"; msg.GetType(); \" msg: \"; msg.GetMessage()\n endif\n endif\n endwhile\n print \"Exiting...\"\nEnd Sub\n\nFunction GenerateSearchSuggestions(partSearchText As String) As Object\n availableContent = [\n \"ghost in the shell\"\n \"parasite dolls\"\n \"final fantasy\"\n \"ninja scroll\"\n \"space ghost\"\n \"hellboy\"\n \"star wars\"\n \"terminator\"\n \"house of cards\"\n \"dexter\"\n ]\n suggestions = []\n if partSearchText <> \"\"\n partSearchText = LCase(partSearchText)\n for each available in availableContent\n if available.Instr(partSearchText) >= 0\n suggestions.Push(available)\n end if\n end for\n end if\n return suggestions\nEnd Function\n```\n\n**Image: roSearchScreen example results (search suggestions)**\n\n![Image: roSearchScreen example results (search suggestions)](https://image.roku.com/ZHZscHItMTc2/roSearchScreen.png \"roSearchScreen\")", + "events": [ + { + "name": "roSearchScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rosearchscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSearchScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsearchscreen.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + } + ], + "isDeprecated": true, + "name": "roSearchScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosearchscreen.md" + }, + "rosgnode": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roSGNode" + } + ], + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation. To create an roSGNode object for a specific node class, call:\n\n`CreateObject(\"roSGNode\", \"nodetype\")`\n\nWhere nodetype is a string specifying the node class to be created. For example, the following creates an object of the SceneGraph Poster node class:\n\n`CreateObject(\"roSGNode\", \"Poster\")`\n\nReference information on all SceneGraph node classes can be found in [SceneGraph API Reference](https://developer.roku.com/docs/references/scenegraph/node.md.\n\nPrior to creating an roSGScreen object and calling its `show()` function, creating roSGNode objects and using their interfaces is not guaranteed to work correctly. If you need to create some roSGNode objects and/or use roSGNode interfaces prior to calling an roSGScreen object `show()` function, you can use an roSGScreen object `createScene()` function to create an instance of a SceneGraph XML component that does any required setup and initialization prior to the roSGScreen object being displayed.\n\nIn addition, roSGNode implements the ifAssociativeArray interface as a wrapper for ifSGNodeFIeld so that the convenient node.field notation may be using for setting, getting, and observing fields.", + "events": [ + { + "name": "roSGNodeEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rosgnodeevent.md" + } + ], + "interfaces": [ + { + "name": "ifAssociativeArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifassociativearray.md" + }, + { + "name": "ifSGNodeBoundingRect", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodeboundingrect.md" + }, + { + "name": "ifSGNodeChildren", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodechildren.md" + }, + { + "name": "ifSGNodeDict", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodedict.md" + }, + { + "name": "ifSGNodeField", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.md" + }, + { + "name": "ifSGNodeFocus", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefocus.md" + }, + { + "name": "ifSGNodeHttpAgentAccess", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodehttpagentaccess.md" + } + ], + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + }, + "rosgscreen": { + "constructors": [ + { + "params": [], + "returnType": "roSGScreen" + } + ], + "description": "The roSGScreen object is a SceneGraph canvas that displays the contents of a SceneGraph Scene node instance. The object is created by calling:\n\n`CreateObject(\"roSGScreen\")`\n\n`CreateScene()` takes one argument, the name of the scene component. A channel will typically extend Scene to define its own channel specific Scene type (such as MyScene, etc.) This Scene component name is passed to `CreateScene().`\n\n**roSGScreen typical usage example**\n\n```\nscreen = CreateObject(\"roSGScreen\")\nscene = screen.CreateScene(\"Scene\")\nscreen.show()\n```", + "events": [ + { + "name": "roSGScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rosgscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifSgScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgscreen.md" + } + ], + "name": "roSGScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgscreen.md" + }, + "roslideshow": { + "constructors": [ + { + "params": [], + "returnType": "roSlideShow" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Slide Show screen provides the ability to setup a photo slide show to playback a series of images.\n\nImages may be jpg, png or gif files. The developer can control the sequencing and timing of the slideshow. The object is designed to accept an array of [Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\") objects, describing the images and providing URLs for accessing each image. TextOverlayUL, TextOverlayUR, and TextOverlayBody are content meta-data properties used to display a text overlay.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roSlideShow\")`", + "events": [ + { + "name": "roSlideShowEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/roslideshowevent.md" + } + ], + "interfaces": [ + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSlideShow", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifslideshow.md" + } + ], + "isDeprecated": true, + "name": "roSlideShow", + "url": "https://developer.roku.com/docs/references/brightscript/components/roslideshow.md" + }, + "rosocketaddress": { + "constructors": [ + { + "params": [], + "returnType": "roSocketAddress" + } + ], + "description": "The roSocketAddress is used by the roStreamSocket and roDataGramSocket components for TCP and UDP traffic respectively.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roSocketAddress\")`\n\nMethods in [ifSocketAddress](https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketaddress.md\"ifSocketAddress\") are used to assign an IP address to the object. roSocketAddress currently supports only IPV4 addresses.", + "events": [], + "interfaces": [ + { + "name": "ifSocketAddress", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketaddress.md" + } + ], + "name": "roSocketAddress", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosocketaddress.md" + }, + "rospringboardscreen": { + "constructors": [ + { + "params": [], + "returnType": "roSpringboardScreen" + } + ], + "description": "The Springboard Screen shows detailed information about an individual piece of content and provides options for actions that may be taken on that content.\n\nThe detailed description of the content is displayed with poster art for the title. Artwork may be displayed portrait or landscape orientation depending on the ContentType set in the [Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\").\n\nThe caller may add one or more buttons to the screen with actions such as Play, Resume, Purchase or More Info. The script is notified via an event when a button is selected and it is the responsibility of the script writer to handle that event as desired and perform the requested action.\n\n![worddav-button-actions](https://image.roku.com/ZHZscHItMTc2/worddav3570180012b8208f098d035b989f8fa8.png \"worddav3570180012b8208f098d035b989f8fa8\")\n\nThis object is created with no parameters:\n\n`CreateObject(\"roSpringboardScreen\")`\n\n* Orientation for artwork is based on ContentType setting and may be portrait or landscape.\n \n* The audio springboard is capable of adding a progress bar.\n \n* If the ContentType is audio, the album art dimensions are:\n \n\n```\n SD: 124 x 112\n HD: 188 x 188\n```\n\n* If the ContentType is episode, the album art dimensions are:\n\n```\n SD: 180 x 122\n HD: 264 x 198\n```\n\n* If the ContentType is any other value, the album art dimensions are:\n\n```\n SD: 112 x 142\n HD: 148 x 212\n```\n\n* Up to 5 user-defined buttons may be displayed. Buttons are displayed in the order they are added and always appear in a fixed region of the screen\n \n* The description text will be formatted and justified (right and left edges) to fit between the margins. When the maximum length is reached, the text will be clipped and terminated with an ellipsis. The font is variable pitch, so the maximum number of characters is dependent on the text. The spacing is approximately 85 characters per line x 4 lines = 340 characters. The fonts and character spacing for HD and SD are similar, and display approximately the same number of characters, but the relationship is not exactly 1:1.\n \n* The star rating can show either community StarRating (red) or UserStarRating (yellow). If both values are set, the control will display the UserStarRating. If ratings are not desired, it can be removed by calling SetStaticRatingEnabled(false), providing more space to display actor names.\n \n* The Length attribute will display a formatted string or show length. If the value is zero, this field will display 0m, if the attribute is not set/missing then this field will not be displayed.\n \n\n**Example**\n\nThe following example shows the process of creating an roSpringboardScreen, setting up the content meta-data, showing the screen and waiting for an event. This example is simplified for clarity and it's assumed the real-world applications will use techniques like getting data from web services using roUrlTransfer.\n\n![springboard-audio](https://image.roku.com/ZHZscHItMTc2/springboard-audio.png \"springboard-audio\")\n\n```\nFunction Main()\n port = CreateObject(\"roMessagePort\")\n springBoard = CreateObject(\"roSpringboardScreen\")\n springBoard.SetBreadcrumbText(\"[location 1]\", \"[location2]\")\n springBoard.SetMessagePort(port)\n o = CreateObject(\"roAssociativeArray\")\n o.ContentType = \"episode\"\n o.Title = \"[Title]\"\n o.ShortDescriptionLine1 = \"[ShortDescriptionLine1]\"\n o.ShortDescriptionLine2 = \"[ShortDescriptionLine2]\"\n o.Description = \"\"\n For i = 1 To 15\n o.Description = o.Description + \"[Description] \"\n End For\n o.SDPosterUrl = \"\"\n o.HDPosterUrl = \"\"\n o.Rating = \"NR\"\n o.StarRating = \"75\"\n o.ReleaseDate = \"[mm/dd/yyyy]\"\n o.Length = 5400\n o.Categories = CreateObject(\"roArray\", 10, true)\n o.Categories.Push(\"[Category1]\")\n o.Categories.Push(\"[Category2]\")\n o.Categories.Push(\"[Category3]\")\n o.Actors = CreateObject(\"roArray\", 10, true)\n o.Actors.Push(\"[Actor1]\")\n o.Actors.Push(\"[Actor2]\")\n o.Actors.Push(\"[Actor3]\")\n o.Director = \"[Director]\"\n springBoard.SetContent(o)\n springBoard.Show()\n While True\n msg = wait(0, port)\n If msg.isScreenClosed() Then\n Return -1\n Elseif msg.isButtonPressed()\n print \"msg: \"; msg.GetMessage(); \"idx: \"; msg.GetIndex()\n Endif\n End While\nEnd Function\n```\n\nThe following screen is displayed when this code is executed:\n\n![worddav-code-displayed](https://image.roku.com/ZHZscHItMTc2/worddav256ada1e0e0cdc53d79428655ca7702b.png \"worddav256ada1e0e0cdc53d79428655ca7702b\")", + "events": [ + { + "name": "roSpringboardScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rospringboardscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifSpringboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifspringboardscreen.md" + } + ], + "name": "roSpringboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rospringboardscreen.md" + }, + "rosprite": { + "constructors": [], + "description": "The roSprite object cannot be created directly with a CreateObject() call. It must be associated with a managing roCompositor object. This association is implicitly created by creating an roSprite object with the roCompositor methods NewSprite() or NewAnimatedSprite().", + "events": [], + "interfaces": [ + { + "name": "ifSprite", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsprite.md" + } + ], + "name": "roSprite", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosprite.md" + }, + "rostreamsocket": { + "constructors": [ + { + "params": [], + "returnType": "roStreamSocket" + } + ], + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them. The interface is modeled on and works much like standard Berkeley sockets.\n\nThis object is created without any arguments:\n\n`CreateObject(\"roStreamSocket\")`\n\n**Example: Open TCP Connection to Server**\n\n```\nsendAddress = CreateObject(\"roSocketAddress\")\nsendAddress.SetAddress(\"www.google.com:80\")\nsocket = CreateObject(\"roStreamSocket\")\nsocket.setSendToAddress(sendAddress)\nIf socket.Connect()\n Print \"Connected Successfully\"\nEnd If\n```\n\n**Example: Echo Server**\n\n```\nfunction main()\n messagePort = CreateObject(\"roMessagePort\")\n connections = {}\n buffer = CreateObject(\"roByteArray\")\n buffer[512] = 0\n tcpListen = CreateObject(\"roStreamSocket\")\n tcpListen.setMessagePort(messagePort)\n addr = CreateObject(\"roSocketAddress\")\n addr.setPort(54321)\n tcpListen.setAddress(addr)\n tcpListen.notifyReadable(true)\n tcpListen.listen(4)\n if not tcpListen.eOK()\n print \"Error creating listen socket\"\n stop\n end if\n while True\n event = wait(0, messagePort)\n if type(event) = \"roSocketEvent\"\n changedID = event.getSocketID()\n if changedID = tcpListen.getID() and tcpListen.isReadable()\n ' New\n newConnection = tcpListen.accept()\n if newConnection = Invalid\n print \"accept failed\"\n else\n print \"accepted new connection \" newConnection.getID()\n newConnection.notifyReadable(true)\n newConnection.setMessagePort(messagePort)\n connections[Stri(newConnection.getID())] = newConnection\n end if\n else\n ' Activity on an open connection\n connection = connections[Stri(changedID)]\n closed = False\n if connection.isReadable()\n received = connection.receive(buffer, 0, 512)\n print \"received is \" received\n if received > 0\n print \"Echo input: '\"; buffer.ToAsciiString(); \"'\"\n ' If we are unable to send, just drop data for now.\n ' You could use notifywritable and buffer data, but that is\n ' omitted for clarity.\n connection.send(buffer, 0, received)\n else if received=0 ' client closed\n closed = True\n end if\n end if\n if closed or not connection.eOK()\n print \"closing connection \" changedID\n connection.close()\n connections.delete(Stri(changedID))\n end if\n end if\n end if\n end while\n\n print \"Main loop exited\"\n tcpListen.close()\n for each id in connections\n connections[id].close()\n end for\nEnd Function\n```", + "events": [ + { + "name": "roSocketEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rosocketevent.md" + } + ], + "interfaces": [ + { + "name": "ifSocket", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocket.md" + }, + { + "name": "ifSocketAsync", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketasync.md" + }, + { + "name": "ifSocketConnection", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnection.md" + }, + { + "name": "ifSocketConnectionOption", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnectionoption.md" + }, + { + "name": "ifSocketConnectionStatus", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnectionstatus.md" + }, + { + "name": "ifSocketStatus", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketstatus.md" + } + ], + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + }, + "rostring": { + "constructors": [], + "description": "roString is the object equivalent for intrinsic type 'String'.\n\nThis is useful in the following situations:\n\n* When an object is needed, instead of an intrinsic value. For example, \"roList\" maintains a list of objects. If an String is added to roList, it will be automatically wrapped in an roString by the language interpreter. When a function that expects a BrightScript Component as a parameter is passed a string, BrightScript automatically creates the equivalent BrightScript Component.\n \n* If any object exposes the ifString interface, that object can be used in any expression that expects an intrinsic value.", + "events": [], + "interfaces": [ + { + "name": "ifString", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifstring.md" + }, + { + "name": "ifStringOps", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifstringops.md" + }, + { + "name": "ifToStr", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + } + ], + "name": "roString", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostring.md" + }, + "rosystemlog": { + "constructors": [], + "description": "The roSystemLog component enables the application to receive events from the Roku Streaming Player that are intended for reporting errors and trends, rather than trigger a response to a user action.\n\nAll of the log event messages are sent to the roMessagePort that is registered on the [roSystemLogEvent](https://developer.roku.com/docs/references/brightscript/events/rosystemlogevent.md\"roSystemLogEvent\") object. See roSystemLogEvent for details on the messages.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roSystemLog\")`\n\nThe roSystemLog component requires specific Design Patterns in your BrightScript Application. Take care to:\n\n* Use one roMessagePort throughout the application (instead of creating a new roMessagePort for each screen).\n* Create one roSystemLog instance at startup that remains for the entire lifetime of the application.\n* Pass the global roMessagePort referenced in the first bullet point to SetMessagePort() on the roSystemLog component.\n* Enable the desired log types using EnableType().\n* Handle the [roSystemLogEvents](https://developer.roku.com/docs/references/brightscript/events/rosystemlogevent.md\"roSystemLogEvents\") in all message loops.", + "events": [], + "interfaces": [ + { + "name": "ifSystemLog", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsystemlog.md" + } + ], + "name": "roSystemLog", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosystemlog.md" + }, + "rotextscreen": { + "constructors": [], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nroTextScreen provides a way of displaying large amounts of scrollable text.\n\nThis type of screen can be used to display help text, credits, license agreements, or other large amounts of text that require scrolling.\n\nThe interface allows you to set the text and specify zero or more buttons.\n\nIf no buttons are specified, then the user can exit the screen by pressing BACK or OK.", + "events": [ + { + "name": "roTextScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rotextscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifTextScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftextscreen.md" + } + ], + "isDeprecated": true, + "name": "roTextScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotextscreen.md" + }, + "rotexttospeech": { + "constructors": [ + { + "params": [], + "returnType": "roTextToSpeech" + } + ], + "description": "> Please note this component is only available on the following devices: Roku Streaming Stick (3600X), Roku Express (3700X) and Express+ (3710X), Roku Premiere (4620X) and Premiere+ (4630X), Roku Ultra (4640X), and any Roku TV running Roku OS version 7.2 and later.\n\nThe roTextToSpeech component provides text to speech capabilities to applications.\n\nAn roTextToSpeech component object is created with no parameters:\n\n`CreateObject(\"roTextToSpeech\")`", + "events": [ + { + "name": "roTextToSpeechEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rotexttospeechevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexttospeech.md" + } + ], + "name": "roTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md" + }, + "rotexturemanager": { + "constructors": [ + { + "params": [], + "returnType": "roTextureManager" + } + ], + "description": "The Texture Manager provides a set of API's for managing an roBitmap cache.\n\n**Example: Requesting an roBitmap from the roTextureManager**\n\n```\nSub Main()\n mgr = CreateObject(\"roTextureManager\")\n msgport = CreateObject(\"roMessagePort\")\n mgr.SetMessagePort(msgport)\n\n request = CreateObject(\"roTextureRequest\",\"pkg:/assets/comet.jpg\")\n mgr.RequestTexture(request)\n\n msg=wait(0, msgport)\n if type(msg)=\"roTextureRequestEvent\" then\n print \"request id\";msg.GetId()\n print \"request state:\";msg.GetState()\n print \"request URI:\";msg.GetURI()\n state = msg.GetState()\n if state = 3 then\n bitmap = msg.GetBitmap()\n if type(bitmap)<>\"roBitmap\" then\n print \"Unable to create robitmap\"\n stop ' stop exits to the debugger\n end if\n end if\n end if\nEnd Sub\n```", + "events": [ + { + "name": "roTextureRequestEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rotexturerequestevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifTextureManager", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexturemanager.md" + } + ], + "name": "roTextureManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturemanager.md" + }, + "rotexturerequest": { + "constructors": [ + { + "params": [ + { + "isRequired": true, + "name": "param1", + "type": "string" + } + ], + "returnType": "roTextureRequest" + } + ], + "description": "An roTextureRequest is used to make requests to the roTextureManager.\n\nAn roTextureRequest object is created using the CreateObject() method and passing it a URI string:\n\n`CreateObject(\"roTextureRequest\", \"pkg:/assets/comet.jpg\")`\n\n**Example: Requesting a URL from the roTextureManager**\n\n```\nSub Main()\n mgr = CreateObject(\"roTextureManager\")\n msgport = CreateObject(\"roMessagePort\")\n mgr.SetMessagePort(msgport)\n\n request = CreateObject(\"roTextureRequest\",\"http://192.168.1.10/ball.png\")\n mgr.RequestTexture(request)\n\n msg=wait(0, msgport)\n if type(msg)=\"roTextureRequestEvent\" then\n print \"request id\";msg.GetId()\n print \"request state:\";msg.GetState()\n print \"request URI:\";msg.GetURI()\n state = msg.GetState()\n if state = 3 then\n bitmap = msg.GetBitmap()\n if type(bitmap)<>\"roBitmap\" then\n print \"Unable to create robitmap\"\n stop ' stop exits to the debugger\n end if\n end if\n end if\nEnd Sub\n```\n\n**Example: Requesting a scaled image from the roTextureManager**\n\n```\nSub Main()\n mgr = CreateObject(\"roTextureManager\")\n msgport = CreateObject(\"roMessagePort\")\n mgr.SetMessagePort(msgport)\n\n request = CreateObject(\"roTextureRequest\",\"pkg:/assets/ball.png\")\n request.SetSize(100, 100)\n request.SetScaleMode(1)\n mgr.RequestTexture(request)\nEnd Sub\n```\n\n**Example: Making an HTTPS request from the roTextureManager**\n\n```\nSub Main()\n mgr = CreateObject(\"roTextureManager\")\n msgport = CreateObject(\"roMessagePort\")\n mgr.SetMessagePort(msgport)\n\n request = CreateObject(\"roTextureRequest\",\"https://192.168.1.10/ball.png\")\n request.SetCertificatesFile(\"common:/certs/ca-bundle.crt\")\n request.InitClientCertificates()\n\n mgr.RequestTexture(request)\nEnd Sub\n```", + "events": [], + "interfaces": [ + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifTextureRequest", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexturerequest.md" + } + ], + "name": "roTextureRequest", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturerequest.md" + }, + "rotimespan": { + "constructors": [ + { + "params": [], + "returnType": "roTimespan" + } + ], + "description": "The Timespan object provides an interface to a simple timer for tracking the duration of activities. It's useful for tracking how an action has taken or if a specified time has elapsed from some starting event.\n\n**Example: Timing an activity**\n\n```\nREM ******************************************************\nREM Compute the number of millisecs to perform a task\nREM ******************************************************\ntimer = CreateObject(\"roTimespan\")\ntimer.Mark()\nDoTimeConsumingTask()\nPrint \"Task took: \" + timer.TotalMilliseconds().ToStr()\n\nREM ******************************************************\nREM Compute how many seconds until rental expires\nREM ******************************************************\nFunction secondsLeft(String expirationDate) As Integer\n str = expirationDate\n if str = invalid return -1\n ts = CreateObject(\"roTimespan\")\n seconds = ts.GetSecondsToISO8601Date(str)\n print \"Expires: \" + str + \" secs: \" + Stri(seconds)\n return seconds\nEnd Function\n```", + "events": [], + "interfaces": [ + { + "name": "ifTimespan", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftimespan.md" + } + ], + "name": "roTimespan", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotimespan.md" + }, + "rourltransfer": { + "constructors": [ + { + "params": [], + "returnType": "roUrlTransfer" + } + ], + "description": "A roUrlTransfer object transfers data to or from remote servers specified by URLs. It can perform mutual authentication with a web server.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roUrlTransfer\")`\n\nIf using HTTPS, the developer must specify a certificate file by calling SetCertificatesFile() with a .pem file that includes the certificate authority cert (like Verisign, Thawte, etc., or your own with OpenSSL) that signed the web server certificate. This must be called before making a request. The developer can also use the Roku standard cert bundle (which contains certificates for most common signing authorities) stored in common:/certs/ca-bundle.crt; or download the CA certificate [here](https://github.com/rokudev/ca-certificate/blob/master/ca-bundle.crt).\n\nThe web server can authenticate that the requested connection is from a Roku Streaming Player and that the request is from your application by taking the following actions:\n\n* Add the Roku CA certificate to the web server's certificate authorities keychain, download the CA certificate.\n* Configure your web server to reject any connection that does not have a valid client certificate.\n* Check the X-Roku-Reserved-Dev-Id header in the request. It should contain the Developer ID of your application. If it does not, another application on the Roku is attempting to access the server, and the request is rejected.\n\n**Example**\n\nIn order for your web server to perform the steps above to authenticate your Roku Streaming Player, your application needs to call the following functions before performing any https requests:\n\n```\nobject.SetCertificatesFile(\"common:/certs/ca-bundle.crt\")\nobject.AddHeader(\"X-Roku-Reserved-Dev-Id\", \"\")\nobject.InitClientCertificates()\n```", + "events": [ + { + "name": "roUrlEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rourlevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md" + } + ], + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + }, + "rovideoplayer": { + "constructors": [ + { + "params": [], + "returnType": "roVideoPlayer" + } + ], + "description": "The roVideoPlayer component implements a video player with more programmatic control, but less user control than the roVideoScreen component.\n\nThe roVideoPlayer can be used in conjunction with the roImageCanvas to do graphical overlays, windowed video, zoom, and programmatic control of playlists and trick play. When using with the roImageCanvas, you can put the roVideoPlayer is at a lower z-order layer than other imageCanvas layers and implement overlays on top of the playing video.\n\nUnlike the roVideoScreen component roVideoPlayer does not have automatic trick play modes and built in controls to support that trick play. Any trick play requires the developer to build his own controls using buttons on the roImageCanvas.\n\nNote that all the video playback notes under roVideoScreen apply to the roVideoPlayer. The customvideoplayer sample application is a good example of roVideoPlayer usage.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roVideoPlayer\")`", + "events": [ + { + "name": "roVideoPlayerEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rovideoplayerevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifVideoPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoplayer.md" + } + ], + "name": "roVideoPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoplayer.md" + }, + "rovideoscreen": { + "constructors": [ + { + "params": [], + "returnType": "roVideoScreen" + } + ], + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe Video Screen object implements the video playback portion of the user interface.\n\nThis object is created with no parameters:\n\n`CreateObject(\"roVideoScreen\")`\n\nThe API's to the video screen allow the developer to setup a fully featured playback environment with minimal coding. The developer is responsible for initial playback setup and providing the required data (e.g. StreamURLs, SteamsBitrates, etc.) as part of the [Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\"). Once created and displayed, the screen will respond to events and manage the playback experience for the user.\n\nThe roVideoScreen is designed for streaming content. The preferred implementation should provide multiple bitrates (ideally four) of video to provide a high quality user experience under a variety of network conditions. Using the StreamBitrates and StreamURLs provided in the content meta-data for the item, the roVideoScreen will automatically monitor and select the best stream based on the users measured bandwidth. If network performance changes, the system will adapt and rebuffer to stream at a different bandwidth if necessary. Note that the StreamURLs, StreamBitrates, StreamQualities and StreamStickyHttpRedirects are all arrays that are aligned with each other. For example, the first stream listed would be the 0th element of all of these arrays.\n\nThe bitrates should represent the actual bitrate of the stream. The bitrate is used for both the display of the dots as well as the stream selection algorithm. The dots work a follows; If the stream bitrate equals:\n\n* 0 = no dots\n* < 500 Kbps= 1 dot\n* < 800 Kbps = 2 dots\n* <1.1 Mbps = 3 dots\n* > \\= 1.1 Mbps = 4 dots\n \n\nThe dots are displayed automatically based on the bitrate of the stream selected unless there is a single stream and the bitrate is set to zero, then it won't show any dots. The StreamQuality attribute is used to select streams and indicates if a stream is HD or not. If the attribute for HDBranded is set to true and the stream is HD, the HD icon will show beside the quality dots. If the StreamQuality is set to HD, and the user display type is set to SD, the HD stream will not be selected.\n\nThe roVideoScreen automatically provides trick mode for all supported content types. There are two type of trick modes supported; scene based selection and time-based selection. If BIF image files are provided for a title, scene-based trick modes will appear. (See the BIF File format Documentation for more information) The user will be presented with the images and progress bar needed for SEEK, FF, REW within a stream. The following image shows how trick modes are rendered with BIF files:\n\n![IMAGE](https://image.roku.com/ZHZscHItMTc2/worddavdc8a50b63d70082736fbebee19c18eff.png \"worddavdc8a50b63d70082736fbebee19c18eff\")\n\nThe FF/REW functionality provides three speeds; slow, medium and fast. At slower speeds, the system displays the current frame in the center of the screen and additional frames on the side for contextual information. At higher speeds, the side frames disappear and only the center image is displayed. The I-frames within the video do not need to precisely align with the time stamp of the image frames in the BIF file. When the user stops and selects a frame, the video playback begins at the first I-frame less than or equal to the time position of the selected frame.\n\nWhen BIF images are not available, the system will default to a time based trick play behavior. The user control is still the same, but only the progress bar is displayed and the user will not see individual scenes within the video. This mode is the default, so if images are not available for an individual title, the system will always provide this functionality by default.\n\nThe system will only seek to locations on an I-Frame boundary. Window Media (WMA9 or VC-1) uses the simple index object to determine the I-frame locations and H.264 uses the MOOV atom to determine the correct offsets. If the BIF images are at a consistent time intervals which do not align to I-Frame boundaries, the system will use the nearest I-Frame less than or equal to the time of the BIF image. MP4 or Windows Media are the preferred formats.\n\n**Important Notes on Video Playback**\n\n* The dimensions vary on a title-by-title basis depending on the source material and the target aspect ratio for the encode (e.g. 4:3 or 16:9). Content is always encoded at full width and the height is adjusted. For example, a 1.66 aspect ratio source is encoded as a 720x432 video and displayed as letterboxed for a 4:3 display.\n \n* The frame rate used for encoding is dependent on the source material. Film content is generally 23.976 fps, while video content is generally at 29.97.\n \n* For typical streaming video applications, we recommend a range of 384Kbps to 4.5Mbps. For USB playback, we recommend that you stay under 8.0 Mbps. This provides a good balance between quality and support for a wide number of users. In some cases lower and higher bitrates have been used, but this frequently results in poor quality or limits the % of the installed base that can view this encoding.\n \n* It is critical that the StreamURLs, StreamBitrates, StreamQualities and StreamStickyHttpRedirects arrays are all aligned with each other. For example, the first stream listed would be the 0th element of all of these arrays. You may have multiple streams in the arrays and the system will automatically pick the most appropriate stream based on the users available bandwidth and video settings.\n \n* The StreamQualities array identifies streams as either SD or HD. If the user is configured for SD the system will not select HD streams for playback.\n \n* The optional StreamStartTimeOffset is the offset into the stream which is considered to be the new origin of playback.\n \n* Live – declares the video as live and replaces the time remaining text in the progress bar with \"live\".\n \n* HLS Http Live Streaming support is included in the Roku OS (Introduced in Roku OS2.6). We currently support version 3 of the Http Live Streaming protocol (Pantos – Draft submitted to IETF November 19, 2010 [http://tools.ietf.org/html/draft-pantos-http-live-streaming-05](http://tools.ietf.org/html/draft-pantos-http-live-streaming-05) ). When using HLS, the StreamUrls and StreamQualities array should each have exactly one element. If the HLS stream has only a single bitrate stream, the StreamBitrates array should contain one element specifying that bitrate. If the stream contains more than one variant stream at multiple bitrates, the StreamBitrates array should contain one element with a value of zero. Please see the Video Encoding Guide for information about creating HLS .m3u8 files and segmented .ts files from your current h264 encoded video or distributing live video over HLS to the Roku box.\n \n* In addition to the support for version 2 of the HLS Pantos draft spec, the Roku box supports .m3u8 files that are compressed via deflate or gzip.\n \n * The HTTP response for a query that returns a gzip-compressed file must contain the header: Content-Encoding: gzip\n * The HTTP response for a query that returns a deflate-compressed file must contain the header: Content-Encoding: deflate\n* \"Trick Modes\" and seeking work a little differently with HLS streams. There are a couple of ways that seeking works with HLS and they are different than other streams.\n \n One way of seeking uses the \"target duration\" specified in the .m3u8 file. The first segment in an m3u8 file is assigned a time offset:\n \n T = G \\* N\n \n where G is the \"target duration\" value and N is the sequence number of the segment. Each subsequent segment is assigned a time offset equal to T (the time offset of the first segment) plus the duration value of all earlier segments. The duration of a segment is determined by the EXTINF line before that segment.\n \n* Smooth Streaming (since v4.7) and later by setting the StreamFormat to \"ism\" and setting the streamURL to the MANIFEST url.\n \n * The player type (ContentMetaData.StreamFormat) is \"ism\"\n * The stream URL is the URL that points to the manifest\n * Only H.264 and/or AAC encoding formats are currently supported.\n * Only direct PlayReady licensing is supported. Indirect licensing is currently unsupported. That is, for decryption to work, the ProtectionHeader must be available in the manifest and the LA\\_URL should contain a valid URL to an accessible PlayReady license server.\n * If there are multiple audio tracks, a track will be chosen based on the StreamIndex.Language attribute in the manifest. If the StreamIndex.Language attribute is not populated, the audio track will be chosen arbitrarily. To select a specific audio track before playback, set the ContentMetaData.TrackIDAudio field to the desired track's StreamIndex.Name attribute.\n * If there are multiple video tracks, a track will be chosen arbitrarily. To select a specific video track before playback, set the ContentMetaData.TrackIDVideo field to the desired track's StreamIndex.Name attribute.\n* Standard PlayReady SDK 2.0 Direct License Acquisition Over-the-Air (since v4.8) works by reading the Rights Management Protection Header in the Smooth Streaming Manifest Url. The Roku OS retrieves the license from the PlayReady license server at the license acquisition url endpoint in the Protection Header.\n \n\n```\n#EXT-X_TARGETDURATION:10\n#EXT-X-MEDIA-SEQUENCE:37\n#EXTINF:10\nurl1\n#EXTINF:8\nurl2\n#EXTINF:10\nurl3\n```\n\nThe segment url1 has a time offset of 370, url2 is 380, and url3 is 388. Note that if no TARGETDURATION is specified, the default is 1, so the first segment in the file will have a nonzero time offset (equal to the target duration). The PlayStart content-meta data value allows direct seeking to an offset that is valid within the window of data in the current .m3u8 file.\n\nThere is a second way to seek in an HLS stream. If the m3u8 file has #EXT-X-PROGRAM-DATE-TIME entries, you can seek to a particular date/time by passing a value equal to a modified Unix epoch value. The modified epoch is 1/1/2004 rather than the standard Unix epoch of 1/1/1970. A Unix time value can be converted to an HLS seek time by subtracting 1072915200 (the number of seconds between 1/1/1970 and 1/1/2004). Once again, setting the PlayStart content meta data value allows direct seeking to a specific time offset.\n\nFor example, to seek to the segment marked with the date/time of 7/4/2010 11:30, set PlayStart to 205327800. An example shell expression showing this arithmetic is:\n\n```\n% expr `date -d \"7/4/2010Z11:30:00.000\" +%s` - 1072915200\n205327800\n```\n\nIn BrightScript, the same calculation might be:\n\n```\ndt = CreateObject(\"roDateTime\")\ndt.fromISO8601String(\"7/4/2010T11:30:00.000\")\nitemContentMetaData.PlayStart = dt. asSeconds() - 1072915200 '205327800\n```\n\n1. In Roku OS version 2.6, we've introduced support for SRT files. Please see the content meta-data parameter SubtitleUrl for pointing to a matching SRT file for your video content.\n \n2. In Roku OS version 2.7, we've introduced 1080p support. Please see the content meta-data parameter FullHD for specifying 1080p resolution. Playback at 1080p resolution will only occur when the user has set the display type to HDTV 1080p. Another content meta-data parameter, FrameRate, specifies the frames per second of the video. Valid values are 24 and 30. If the user's display type is set to 1080p and FullHD for the content is false or not set, HD playback will be at 720p resolution. If the user's display type is set to HDTV 720p and FullHD content is set to 1080p resolution, the box will downscale the content to 720p resolution.\n \n\n**Example**\n\n```\n'**********************************************************************\n' This example function is passed an associative array representing a ' piece of content (e.g. a TV episode) There are other attributes\n' (title, description, etc.) but this example focuses on showing\n' attributes required for initiating playback. It creates a video\n' screen, sets the content and starts playback by calling Show()\n'**********************************************************************\nFunction showVideoScreen(episode As Object)\n if type(episode) <> \"roAssociativeArray\" then\n print \"invalid data passed to showVideoScreen\"\n return -1\n endif\n port = CreateObject(\"roMessagePort\")\n screen = CreateObject(\"roVideoScreen\")\n ' Note: HDBranded controls whether the \"HD\" logo is displayed for a\n ' title. This is separate from IsHD because its possible to\n' have an HD title where you don't want to show the HD logo\n' branding for the title. Set these two as appropriate for\n' your content\n episode.HDBranded = false\n episode.IsHD = false\n ' Note: The preferred way to specify stream info in v2.6 is to use\n' the Stream roAssociativeArray content meta data parameter.\n\nepisode.Stream = { url:\"http://myserver.mydomain.com/mycontent.mp4\",\nbitrate:2000\nquality:false\ncontentid:\"mycontent-2000\"\n}\nepisode.StreamFormat: \"mp4\"\n ' now just tell the screen about the title to be played, set the\n ' message port for where you will receive events and call show to\n ' begin playback. You should see a buffering screen and then\n ' playback will start immediately when we have enough data buffered.\n screen.SetContent(episode)\n screen.SetMessagePort(port)\n screen.Show()\n ' Wait in a loop on the message port for events to be received.\n ' We will just quit the loop and return to the calling function\n ' when the users terminates playback, but there are other things\n ' you could do here like monitor playback position and see events\n ' from the streaming player. Look for status messages from the video\n ' player for status and failure events that occur during playback\n while true\n msg = wait(0, port)\n\n if type(msg) = \"roVideoScreenEvent\" then\n print \"showVideoScreen | msg = \"; msg.GetMessage() \" | index = \"; msg.GetIndex()\n if msg.isScreenClosed()\n print \"Screen closed\"\n exit while\n else if msg.isStatusMessage()\n print \"status message: \"; msg.GetMessage()\n else if msg.isPlaybackPosition()\n print \"playback position: \"; msg.GetIndex()\n else if msg.isFullResult()\n print \"playback completed\"\n exit while\n else if msg.isPartialResult()\n print \"playback interrupted\"\n exit while\n else if msg.isRequestFailed()\n print \"request failed - error: \"; msg.GetIndex();\" - \"; msg.GetMessage()\n exit while\n end if\n end if\n end while\nEnd Function\n```", + "events": [ + { + "name": "roVideoScreenEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rovideoscreenevent.md" + } + ], + "interfaces": [ + { + "name": "ifGetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + { + "name": "ifHttpAgent", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + { + "name": "ifSetMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + { + "name": "ifVideoScreen", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoscreen.md" + } + ], + "isDeprecated": true, + "name": "roVideoScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoscreen.md" + }, + "roxmlelement": { + "constructors": [], + "description": "roXMLElement is used to contain an XML tree.\n\nFor instance,\n\n```\nthis is some text\n```\n\nWould parse such that:\n\n```\n Name = \"tag1\"\n Attributes = invalid\n Body = roString with \"this is some text\"\n```\n\n**Example**\n\n```\n \n```\n\nWould parse such that:\n\n```\n Name = \"emptytag\"\n Attributes = roAssociativeArray, with one entry { caveman: \"barney\" }\n Body = invalid\n```\n\nIf the tag contains other tags, body will be of type roXMLList.\n\nTo generate XML, create an roXMLElement, then use functions like SetName(), AddAttribute(), SetBody(), AddElementWithBody(), AddElement(), AddBodyElement(), and AddText() functions to build the XML object hierarchy.\n\nThen call GenXML() to return the XML as a string.\n\nGenXML() takes one parameter (boolean) that indicates whether the generated xml should have the tag at the top.\n\n**Example: Subroutine to print out the contents of an roXMLElement tree**\n\n```\nPrintXML(root, 0)\n\nSub PrintXML(element As Object, depth As Integer)\n print tab(depth*3);\"Name: \";element.GetName()\n if not element.GetAttributes().IsEmpty() then\n print tab(depth*3);\"Attributes: \";\n for each a in element.GetAttributes()\n print a;\"=\";left(element.GetAttributes()[a], 20);\n if element.GetAttributes().IsNext() then print \", \";\n end for\n print\n end if\n if element.GetText()<>invalid then\n print tab(depth*3);\"Contains Text: \";left(element.GetText(), 40)\n end if\n if element.GetChildElements()<>invalid\n print tab(depth*3);\"Contains roXMLList:\"\n for each e in element.GetChildElements()\n PrintXML(e, depth+1)\n end for\n end if\n print\nend sub\n```\n\n**Example: Generating XML**\n\n```\nroot.SetName(\"myroot\")\nroot.AddAttribute(\"key1\", \"value1\")\nroot.AddAttribute(\"key2\", \"value2\")\nne = root.AddBodyElement()\nne.SetName(\"sub\")\nne.SetBody(\"this is the sub1 text\")\nne = root.AddBodyElement()\nne.SetName(\"subelement2\")\nne.SetBody(\"more sub text\")\nne.AddAttribute(\"k\", \"v\")\nne = root.AddElement(\"subelement3\")\nne.SetBody(\"more sub text 3\")\nroot.AddElementWithBody(\"sub\", \"another sub (#4)\")\nPrintXML(root, 0)\nprint root.GenXML(false)\n```", + "events": [], + "interfaces": [ + { + "name": "ifXMLElement", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifxmlelement.md" + } + ], + "name": "roXMLElement", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmlelement.md" + }, + "roxmllist": { + "constructors": [], + "description": "Contains a list of roXML objects.\n\nNormally roXMLList objects are not created via CreateObject(), but are returned from various ifXMLElement functions such as GetChildElements() and GetBod", + "events": [], + "interfaces": [ + { + "name": "ifList", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflist.md" + }, + { + "name": "ifListToArray", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflisttoarray.md" + }, + { + "name": "ifXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifxmllist.md" + } + ], + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + }, + "interfaces": { + "AppManagerTheme": { + "implementers": [], + "methods": [], + "name": "AppManagerTheme", + "properties": [ + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Pa Po Se Sp Te. Example: #E0DFDF", + "name": "BackgroundColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Pa Po Se Sp Te. Example: #FF00FF", + "name": "BreadcrumbDelimiter", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Pa Po Se Sp Te. Example: #FF00FF", + "name": "BreadcrumbTextLeft", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Pa Po Se Sp Te. Example: #FF00FF", + "name": "BreadcrumbTextRight", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Di Se Sp. Example: #FF00FF", + "name": "ButtonHighlightColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Di Se Sp. Example: #0033FF", + "name": "ButtonMenuHighlightText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Di Se Sp. Example: #B0B0B0", + "name": "ButtonMenuNormalOverlayText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Di Se Sp. Example: #686868", + "name": "ButtonMenuNormalText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Di Se Sp. Example: #FF00FF", + "name": "ButtonNormalColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Po. Example: #00FF00", + "name": "CounterSeparator", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Po. Example: #FF0000", + "name": "CounterTextLeft", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Po. Example: #0000FF", + "name": "CounterTextRight", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Di. Example: #808080", + "name": "DialogBodyText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Di. Example: #363636", + "name": "DialogTitleText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "EpisodeSynopsisText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "FilterBannerActiveColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD Filter Banner Active/Focus Highlighter. Screen types: Po. Example: pkg:/images/Filter\\_ActiveHint\\_HD.png", + "name": "FilterBannerActiveHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD Filter Banner Active/Focus Highlighter. Screen types: Po. Example: pkg:/images/Filter\\_ActiveHint\\_SD43.png", + "name": "FilterBannerActiveSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "FilterBannerInactiveColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD Filter Banner Inactive Highlighter. Screen types: Po. Example: pkg:/images/Filter\\_InactiveHint\\_HD.png", + "name": "FilterBannerInactiveHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD Filter Banner Inactive Highlighter. Screen types: Po. Example: pkg:/images/Filter\\_ActiveHint\\_SD43.png", + "name": "FilterBannerInactiveSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "FilterBannerSideColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD Filter Banner Background Image. Screen types: Po. Example: pkg:/images/Filter\\_ActiveHint\\_HD.png", + "name": "FilterBannerSliceHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD Filter Banner Background Image. Screen types: Po. Example: pkg:/images/Filter\\_ActiveHint\\_SD43.png", + "name": "FilterBannerSliceSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value Must be a grayscale value. Screen types: Gr. Example: #363636", + "name": "GridScreenBackgroundColor", + "type": "string" + }, + { + "default": "invalid", + "description": "String representing point \"(x, y)\" that is the offset from the upper left corner of the focused HD image. Set to the negative width & height of border. Screen types: Gr. Example: (-25,-25)", + "name": "GridScreenBorderOffsetHD", + "type": "string" + }, + { + "default": "invalid", + "description": "String representing point \"(x, y)\" that is the offset from the upper left corner of the focused SD image. Set to the negative width & height of border. Screen types: Gr. Example: (-20,-20)", + "name": "GridScreenBorderOffsetSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr. Example: #FF005B", + "name": "GridScreenDescriptionDateColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD Description callout background image on Grid. Screen types: Gr. Example: pkg:/images/Description\\_Background\\_HD.ng", + "name": "GridScreenDescriptionImageHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD Description callout background image on Grid. Screen types: Gr. Example: pkg:/images/Description\\_Background\\_SD43.ng", + "name": "GridScreenDescriptionImageSD", + "type": "string" + }, + { + "default": "invalid", + "description": "String representing point \"(x, y)\" that is the offset from the upper left corner of the focused HD image. Negative values have the description above and to the left of the focused image. Screen types: Gr. Example: (190,255)", + "name": "GridScreenDescriptionOffsetHD", + "type": "string" + }, + { + "default": "invalid", + "description": "String representing point \"(x, y)\" that is the offset from the upper left corner of the focused SD image. Negative values have the description above and to the left of the focused image. Screen types: Gr. Example: (125,170)", + "name": "GridScreenDescriptionOffsetSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr. Example: #5B005B", + "name": "GridScreenDescriptionRuntimeColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr. Example: #606000", + "name": "GridScreenDescriptionSynopsisColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr. Example: #00FFFF", + "name": "GridScreenDescriptionTitleColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD Focus image on Active Grid Poster. Screen types: Gr. Example: pkg:/images/Border\\_16x9\\_HD.png", + "name": "GridScreenFocusBorderHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD Focus image on Active Grid Poster. Screen types: Gr. Example: pkg:/images/Border\\_16x9\\_SD43.png", + "name": "GridScreenFocusBorderSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Gr. Example: #FFFFFF", + "name": "GridScreenListNameColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Logo formatted for display in the overhang. Screen types: Gr. Example: pkg:/images/gridlogoHD.png", + "name": "GridScreenLogoHD", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display. Range 0 to 1280. Screen types: Gr. Example: 592", + "name": "GridScreenLogoOffsetHD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display. Range 0 to 720. Screen types: Gr. Example: 31", + "name": "GridScreenLogoOffsetHD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display. Range 0 to 720. Screen types: Gr. Example: 324", + "name": "GridScreenLogoOffsetSD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display. Range 0 to 480. Screen types: Gr. Example: 21", + "name": "GridScreenLogoOffsetSD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Logo formatted for display in the overhang. Screen types: Gr. Example: pkg:/images/gridlogoSD.png", + "name": "GridScreenLogoSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Gr. Example: #808080", + "name": "GridScreenMessageColor", + "type": "string" + }, + { + "default": "invalid", + "description": "The HD overhang height. Default: \"69\". Screen types: Gr. Example: 75", + "name": "GridScreenOverhangHeightHD", + "type": "string" + }, + { + "default": "invalid", + "description": "The SD overhang height. Default: \"49\". Screen types: Gr. Example: 55", + "name": "GridScreenOverhangHeightSD", + "type": "string" + }, + { + "default": "invalid", + "description": "URI for the overhang slice (thin piece of top of screen border). Screen types: Gr. Example: pkg:/images/gridoverhangHD.png", + "name": "GridScreenOverhangSliceHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URI for the overhang slice (thin piece of top of screen border). Screen types: Gr. Example: pkg:/images/gridoverhangSD.png", + "name": "GridScreenOverhangSliceSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Gr. Example: #CCCCCC", + "name": "GridScreenRetrievingColor", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set HD highlight image. Screen types: Gr Li Po. Example: pkg:/images/listitem\\_highlight\\_hd.png", + "name": "ListItemHighlightHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URL to set SD highlight image. Screen types: Gr Li Po. Example: pkg:/images/listitem\\_highlight\\_sd.png", + "name": "ListItemHighlightSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Po. Example: #CCCC00", + "name": "ListItemHighlightText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Gr Li Po. Example: #CCCC00", + "name": "ListItemText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Li. Example: #CCCC00", + "name": "ListScreenDescriptionText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Li. Example: #CC0000", + "name": "ListScreenTitleColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Small application logo formatted for display in overhang top left. Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/co\\_logo\\_sd.png", + "name": "OverhangPrimaryLogoHD", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films.Range 0 to 1280. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 25", + "name": "OverhangPrimaryLogoOffsetHD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films.Range 0 to 720. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 50", + "name": "OverhangPrimaryLogoOffsetHD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films.Range 0 to 720. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 25", + "name": "OverhangPrimaryLogoOffsetSD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films.Range 0 to 480. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 50", + "name": "OverhangPrimaryLogoOffsetSD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Small application logo formatted for display in overhang top left. Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/co\\_logo\\_sd.png", + "name": "OverhangPrimaryLogoSD", + "type": "string" + }, + { + "default": "invalid", + "description": "Small application logo formatted for display in overhang top left. Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/co\\_logo\\_hd.png", + "name": "OverhangSecondaryLogoHD", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films. Range 0 to 1280. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 25", + "name": "OverhangSecondaryLogoOffsetHD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films. Range 0 to 720. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 50", + "name": "OverhangSecondaryLogoOffsetHD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films. Range 0 to 720. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 25", + "name": "OverhangSecondaryLogoOffsetSD\\_X", + "type": "string" + }, + { + "default": "invalid", + "description": "Offset in pixels from the top-left origin of the display films. Range 0 to 480. Screen types: Co Ke Li Pa Po Se Sp Te. Example: 50", + "name": "OverhangSecondaryLogoOffsetSD\\_Y", + "type": "string" + }, + { + "default": "invalid", + "description": "Small application logo formatted for display in overhang top left. Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/co\\_logo\\_sd.png", + "name": "OverhangSecondaryLogoSD", + "type": "string" + }, + { + "default": "invalid", + "description": "URI for the overhang slice (thin piece of border at the top of the screen in HD size). Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/overhang\\_hd.png", + "name": "OverhangSliceHD", + "type": "string" + }, + { + "default": "invalid", + "description": "URI for the overhang slice (thin piece of top of screen border). Screen types: Co Ke Li Pa Po Se Sp Te. Example: pkg:/images/overhang\\_sd.png", + "name": "OverhangSliceSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Co Pa Te. Example: #FF00FF", + "name": "ParagraphBodyText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Co Pa Te. Example: #FF00FF", + "name": "ParagraphHeaderText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "PosterScreenLine1Text", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Po. Example: #FF00FF", + "name": "PosterScreenLine2Text", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Co. Example: #FF00FF", + "name": "RegistrationCodeColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Co. Example: #FF00FF", + "name": "RegistrationFocalColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Co. Example: #10FF80", + "name": "RegistrationFocalRectColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Position and size of the HD focal rectangle. Four integer: (x,y,width,height). Screen types: Co. Example: (228,360,120,82)", + "name": "RegistrationFocalRectHD", + "type": "string" + }, + { + "default": "invalid", + "description": "Position and size of the SD focal rectangle. Four integer: (x,y,width,height). Screen types: Co. Example: (172,220,90,76)", + "name": "RegistrationFocalRectSD", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardActorColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardAlbumColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Album Label. Screen types: Sp. Example: on", + "name": "SpringboardAlbumLabel", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardAlbumLabelColor", + "type": "string" + }, + { + "default": "invalid", + "description": "boolean string. Screen types: Sp. Example: true", + "name": "SpringboardAllow6Buttons", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardArtistColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Artist Label. Screen types: Sp. Example: by", + "name": "SpringboardArtistLabel", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardArtistLabelColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardDirectorColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardDirectorLabelColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardDirectorPrefixText", + "type": "string" + }, + { + "default": "invalid", + "description": "Director Label. Screen types: Sp. Example: Written by", + "name": "SpringboardDirectorText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardGenreColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardRuntimeColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardSynopsisColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Sp. Example: #FF00FF", + "name": "SpringboardTitleText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Must be a grayscale value. Screen types: Te. Example: #808080", + "name": "TextScreenBodyBackgroundColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Te. Example: #363636", + "name": "TextScreenBodyText", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Te. Example: #CC0000", + "name": "TextScreenScrollBarColor", + "type": "string" + }, + { + "default": "invalid", + "description": "HTML HEX Color Value. Screen types: Te. Example: #00CC00", + "name": "TextScreenScrollThumbColor", + "type": "string" + }, + { + "default": "invalid", + "description": "Theme type. Generic-dark is the only valid value. Otherwise the default theme applies. Screen types: . Example: generic-dark", + "name": "ThemeType", + "type": "string" + } + ] + }, + "ifappinfo": { + "implementers": [ + { + "description": "Returns information about the application", + "name": "roAppInfo", + "url": "https://developer.roku.com/docs/references/brightscript/components/roappinfo.md" + } + ], + "methods": [ + { + "description": "Returns the app's developer ID, or the keyed developer ID, if the application is sideloaded.", + "name": "GetDevID", + "params": [], + "returnDescription": "Channel's Developer ID", + "returnType": "String" + }, + { + "description": "Returns the app's channel ID.", + "name": "GetID", + "params": [], + "returnDescription": "Channel ID; e.g., \"12345\" or \"dev\"", + "returnType": "String" + }, + { + "description": "Returns the subtitle value from the manifest.", + "name": "GetSubtitle", + "params": [], + "returnDescription": "Possible subtitle configuration", + "returnType": "String" + }, + { + "description": "Returns the title value from the manifest.", + "name": "GetTitle", + "params": [], + "returnDescription": "Title of the channel", + "returnType": "String" + }, + { + "description": "Returns the named manifest value, or an empty string if the entry is does not exist.", + "name": "GetValue", + "params": [ + { + "default": null, + "description": "The manifest value to be returned.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "Manifest value; empty string", + "returnType": "String" + }, + { + "description": "Returns the conglomerate version number from the manifest, as formatted major\\_version + minor\\_version + build\\_version.", + "name": "GetVersion", + "params": [], + "returnDescription": "Channel version number. e.g. \"1.2.3\"", + "returnType": "String" + }, + { + "description": "Returns true if the application is sideloaded, i.e. the channel ID is \"dev\".", + "name": "IsDev", + "params": [], + "returnDescription": "True/ False", + "returnType": "Boolean" + } + ], + "name": "ifAppInfo", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappinfo.md" + }, + "ifappmanager": { + "implementers": [ + { + "description": "Returns information about the application", + "name": "roAppManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/roappmanager.md" + } + ], + "methods": [ + { + "description": "Clears a previously set attribute and reverts to its default value.", + "name": "ClearThemeAttribute", + "params": [ + { + "default": null, + "description": "The theme attribute to be cleared.", + "isRequired": true, + "name": "attributeName", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Returns the user's screensaver wait time setting in number of minutes, or zero if the screensaver is disabled.", + "name": "GetScreensaverTimeout", + "params": [], + "returnDescription": "The number of minutes set for the screensaver wait time.", + "returnType": "Integer" + }, + { + "description": "Returns an [roTimespan](https://developer.roku.com/docs/references/brightscript/components/rotimespan.md\"roTimespan\") object, which is \"marked\" when the user clicked on the application button on the home screen.Calling the TotalMilliseconds() method on the returned roTimespan object returns the total number of milliseconds since the application started.", + "name": "GetUptime", + "params": [], + "returnDescription": "An [roTimespan](/docs/references/brightscript/components/rotimespan.md \"roTimespan\") object.", + "returnType": "Object" + }, + { + "description": "This method returns true if a channel with the specified channelID and the minimum version required is installed.", + "name": "IsAppInstalled", + "params": [ + { + "default": null, + "description": "The unique id of the channel.", + "isRequired": true, + "name": "channelID", + "type": "String" + }, + { + "default": null, + "description": "The minimum version number of the channel to be used for the query.", + "isRequired": true, + "name": "version", + "type": "String" + } + ], + "returnDescription": "A boolean indicating whether the specified channel is installed.", + "returnType": "Boolean" + }, + { + "description": "Launches the channel with the specified channelID and the specified version, with the playback experience upon launching the channel based on the provided params.", + "name": "LaunchApp", + "params": [ + { + "default": null, + "description": "The unique ID of the channel to be launched.", + "isRequired": true, + "name": "channelID", + "type": "String" + }, + { + "default": null, + "description": "The minimum version of the channel required to launch the channel. If the specified version (or later) is not being used, the channel is not launched. To skip the version check, pass an empty string.", + "isRequired": true, + "name": "version", + "type": "String" + }, + { + "default": null, + "description": "Key-value pairs specifying the playback experience upon launching the channel. For example, if deep linking parameters are passed into the method, the content ID specifies the content to be played upon launching the channel, and the mediaType determines whether the content is launched directly into playback, launched into playback using bookmarks, or an episode selection screen for the content is displayed). To launch the channel store springboard of a channel, use the **ShowChannelStoreSpringboard()** method. You can also do this by passing a channelID of \"11\" is passed into this method (`LaunchApp(\"11\", \"\", params)`). In this case, the params field would include a content ID (`params = {contentID: \"myAwesomeMovie_123\"}`).", + "isRequired": true, + "name": "params", + "type": "roAssociative Array" + } + ], + "returnType": "Void" + }, + { + "description": "Enables or disables automatic Audio Guide and override any manifest setting.This is useful for channels that want to temporarily turn off automatic Audio Guide for specific screens.", + "name": "SetAutomaticAudioGuideEnabled", + "params": [ + { + "default": null, + "description": "A flag indicating whether to enable or disable the automatic audio guide.", + "isRequired": true, + "name": "enabled", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Updates video or audio [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md) during playback. This method takes a subset of content metadata parameters to be updated. These values override any previously ones sent to the Roku Media Player, and they are used until this function is called again or until the [**roAppManager**](https://developer.roku.com/docs/references/brightscript/components/roappmanager.md instance is deleted.", + "name": "SetNowPlayingContentMetaData", + "params": [ + { + "default": null, + "description": "The video or audio [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md) parameters to be updated (for example, the title and contentType)", + "isRequired": true, + "name": "contentMetaData", + "type": "roAssociativeArray" + } + ], + "returnType": "Void" + }, + { + "description": "Sets a group of theme attributes for the application.", + "name": "SetTheme", + "params": [ + { + "default": null, + "description": "The attributeArray is an [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") of attribute/value pairs. The program may create the roAssociativeArray at runtime or read it from an XML file using the [roXMLElement](https://developer.roku.com/docs/references/brightscript/components/roxmlelement.md\"roXMLElement\") object. Existing values for attributes will be overwritten by the values provided. Any values set by a previous SetTheme or SetThemeAttribute call, but not included in the array currently provided by with the subsequent call will remain unchanged. See [roAppManager](https://developer.roku.com/docs/references/brightscript/components/roappmanager.md\"roAppManager\") the list of valid attributes.", + "isRequired": true, + "name": "attributeArray", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Set an individual theme attribute for the application.", + "name": "SetThemeAttribute", + "params": [ + { + "default": null, + "description": "The attributeName is the name of one of the settable theme attributes and the value is the desired setting. If the attributeName is not valid, no action is performed.", + "isRequired": true, + "name": "attributeName", + "type": "String" + }, + { + "default": null, + "description": "This attributeValue will override the default value for that attribute or modify the value provided by a previous SetTheme or SetThemeAttribute call to the new value provided.", + "isRequired": true, + "name": "attributeValue", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "This method allows a channel to tell Roku when the user is signed in or signed out of the channelIf the channel is removed, the Roku OS will call SetUserSignedIn(false) on the channel's behalf.", + "name": "SetUserSignedIn", + "params": [ + { + "default": null, + "description": "Set to true to indicate that the user is signed in; set to false to indicate the user is signed out.", + "isRequired": true, + "name": "signedIn", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Launches the channel store springboard of the specified channel. The channel store springboard contains detailed information about the channel, including ratings, version, date of last update, developer name, and a description.", + "name": "ShowChannelStoreSpringboard", + "params": [ + { + "default": null, + "description": "The unique ID of the channel to be launched.", + "isRequired": true, + "name": "channelID", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifAppManager", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappmanager.md" + }, + "ifarray": { + "implementers": [ + { + "description": "An array stores an indexed collection of BrightScript objects. Each entry of an array can be a different type, or they may all of the same type", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + }, + { + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes", + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + }, + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Appends the entries in one **roArray** to another. If the passed array contains entries that have not been set to a value, they are not appended.", + "name": "Append", + "params": [ + { + "default": null, + "description": "The **roArray** to be appended to the target array.", + "isRequired": true, + "name": "array", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Deletes all the entries in the array.", + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the length of the array, which is one more than the index of highest entry.", + "name": "Count", + "params": [], + "returnDescription": "The length of the array.", + "returnType": "Integer" + }, + { + "description": "Deletes the indicated array entry, and shifts all entries up. This decreases the array length by one.", + "name": "Delete", + "params": [ + { + "default": null, + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the specified array entry has been removed. If the entry was successfully deleted, returns true. If index is out of range, returns false and does not change the array.", + "returnType": "Boolean" + }, + { + "description": "Returns the last (highest index) array entry without removing it. If the array is empty, returns invalid", + "name": "Peek", + "params": [], + "returnDescription": "Invalid", + "returnType": "Dynamic" + }, + { + "description": "Returns the last entry (highest index) from the array and removes it from the array. If the array is empty, returns invalid and does not change the array.", + "name": "Pop", + "params": [], + "returnDescription": "The last (highest index) array entry.", + "returnType": "Dynamic" + }, + { + "description": "Adds the specified value to the end of the array.", + "name": "Push", + "params": [ + { + "default": null, + "description": "The value to be added to the beginning of the array.", + "isRequired": true, + "name": "tvalue", + "type": "Dynamic" + } + ], + "returnType": "Void" + }, + { + "description": "Removes the first entry (zero index) from the beginning of the array and shifts the other entries up. This method is similar to the [Pop method](#pushtvalue-as-dynamic-as-void), but removes the first entry in the array instead of the last.", + "name": "Shift", + "params": [], + "returnDescription": "The first entry (zero index) removed from the array.", + "returnType": "Dynamic" + }, + { + "description": "Adds the specified value to the beginning of the array (at the zero index) and shifts the other entries down. This method is similar to the [Push method](#push-as-dynamic), but removes the first entry in the array instead of the last.", + "name": "Unshift", + "params": [ + { + "default": null, + "description": "The value to be added to the beginning of the array.", + "isRequired": true, + "name": "tvalue", + "type": "Dynamic" + } + ], + "returnType": "Void" + } + ], + "name": "ifArray", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarray.md" + }, + "ifarrayget": { + "description": "The ifArrayGet interface supports the array indexing operator \\[ \\]\n\n(See [Array Operator](https://developer.roku.com/docs/references/brightscript/language/expressions-variables-types.mdeffects-of-type-conversions-on-accuracy \"Array Operator\"))", + "implementers": [ + { + "description": "An array stores an indexed collection of BrightScript objects. Each entry of an array can be a different type, or they may all of the same type", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + }, + { + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes", + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + }, + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Returns an array entry based on the provided index.", + "name": "GetEntry", + "params": [ + { + "default": null, + "description": "The index of the array entry to be returned.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "The array entry corresponding to the provided index, or invalid if the entry has not been set.", + "returnType": "Dynamic" + } + ], + "name": "ifArrayGet", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayget.md" + }, + "ifarrayjoin": { + "implementers": [ + { + "description": "Returns information about the application", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + } + ], + "methods": [ + { + "description": "Creates a string by joining all array elements together separated by the specified separator. All elements must be of type string; otherwise, an empty string is returned", + "name": "Join", + "params": [ + { + "default": null, + "description": "The string used to separate elements in an array.", + "isRequired": true, + "name": "separator", + "type": "String" + } + ], + "returnDescription": "A String containing the array elements.", + "returnType": "String" + } + ], + "name": "ifArrayJoin", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayjoin.md" + }, + "ifarrayset": { + "description": "The ifArraySet interface supports the array indexing operator \\[\\].\n\n(See ArrayOperator)", + "implementers": [ + { + "description": "An array stores an indexed collection of BrightScript objects. Each entry of an array can be a different type, or they may all of the same type", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + }, + { + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes", + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + }, + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Sets an entry at a given index to the passed value. If index is beyond the bounds of the array, the array is expanded to accommodate it.", + "name": "SetEntry", + "params": [ + { + "default": null, + "description": "The entry to be updated.", + "isRequired": true, + "name": "index", + "type": "Integer" + }, + { + "default": null, + "description": "The new value for the specified entry.", + "isRequired": true, + "name": "tvalue", + "type": "Dynamic" + } + ], + "returnType": "Void" + } + ], + "name": "ifArraySet", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayset.md" + }, + "ifarraysort": { + "implementers": [ + { + "description": "Returns information about the application", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + } + ], + "methods": [ + { + "description": "Reverses the order of elements in an array.", + "name": "Reverse", + "params": [], + "returnType": "Void" + }, + { + "description": "Performs a stable sort on an array.", + "name": "Sort", + "params": [ + { + "default": null, + "description": "Items are arbitrarily grouped by comparable type of number or string, and are sorted within the group with a logical comparison. If \"r\" is included in flags, a reverse sort is performed. If \"i\" is included in flags, a case-insensitive sort is performed. If invalid flags are specified, the sort is not performed.", + "isRequired": true, + "name": "flags", + "type": "Dynamic" + } + ] + }, + { + "description": "Performs a stable sort of an array of associative arrays by value of a common field.", + "name": "SortBy", + "params": [ + { + "default": null, + "description": "The field to be used for sorting.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "Items are arbitrarily grouped by comparable type of number or string, and are sorted within the group with a logical comparison. If \"r\" is included in flags, a reverse sort is performed. If \"i\" is included in flags, a case-insensitive sort is performed. If invalid flags are specified, the sort is not performed.", + "isRequired": true, + "name": "flags", + "type": "Dynamic" + } + ] + } + ], + "name": "ifArraySort", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifarraysort.md" + }, + "ifassociativearray": { + "implementers": [ + { + "description": "An associative array allows objects to be associated with string keys", + "name": "roAssociativeArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md" + }, + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Adds a new entry to the array associating the supplied value with the supplied key string. Only one value may be associated with a key. If the key is already associated with a value, the existing value is discarded.", + "name": "AddReplace", + "params": [ + { + "default": null, + "description": "The key to be added to the associative array.", + "isRequired": true, + "name": "key", + "type": "String" + }, + { + "default": null, + "description": "The value of the key to be added to the associative array.", + "isRequired": true, + "name": "value", + "type": "Dynamic" + } + ], + "returnType": "Void" + }, + { + "description": "Appends an associative array to this calling object. If any key in the **aa** parameter is already associated with a value in the calling object, the current value is discarded and is replaced with the value provided in the **aa** parameter.", + "name": "Append", + "params": [ + { + "default": null, + "description": "The associative array to be appended to the calling object.", + "isRequired": true, + "name": "aa", + "type": "Object" + } + ] + }, + { + "description": "Remove all key/values from the associative array.", + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the number of keys in the associative array.", + "name": "Count", + "params": [], + "returnDescription": "The number of keys in the associative array.", + "returnType": "Integer" + }, + { + "description": "Deletes an entry from an associative array based on the key.", + "name": "Delete", + "params": [ + { + "default": null, + "description": "The key associated with the entry to be deleted.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether an entry is associated with the specified key exists. If there is no associated object then false is returned. If there is such an object then true is returned.", + "returnType": "Boolean" + }, + { + "description": "Looks for an entry in the associative array associated with the specified key.", + "name": "DoesExist", + "params": [ + { + "default": null, + "description": "The key associated with the entry to be checked.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether an entry is associated with the specified key exists. If there is no associated object then false is returned. If there is such an object then true is returned.", + "returnType": "Boolean" + }, + { + "description": "Returns an array containing the associative array key/value pairs in lexicographical order of key.", + "name": "Items", + "params": [], + "returnDescription": "An array of associative array keys/value pairs.", + "returnType": "Object" + }, + { + "description": "Returns an array containing the associative array keys in lexicographical order.", + "name": "Keys", + "params": [], + "returnDescription": "An array of associative array keys.", + "returnType": "Object" + }, + { + "description": "Returns the value in the array associated with the specified key. The key comparison is case-insensitive, unless the **SetModeCaseSensitive()** method has been called.", + "name": "Lookup", + "params": [ + { + "default": null, + "description": "The key associated with the value to be retrieved from the associative array.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "Returns the value in the array associated with the specified key. If there is no value associated with the key, the type \"invalid\" is returned.", + "returnType": "Dynamic" + }, + { + "description": "Same as the [Lookup()](#lookupkey-as-string-as-dynamic) method except that the key comparison is always case insensitive, regardless of the case mode.", + "name": "LookupCI", + "params": [ + { + "default": null, + "description": "The key (case-insensitive) associated with the value to be retrieved from the associative array.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "Returns the value in the array associated with the specified key. If there is no value associated with the key, the type \"invalid\" is returned.", + "returnType": "Dynamic" + }, + { + "description": "Makes all subsequent associative array lookups case sensitive (by default, lookups are case insensitive).", + "name": "SetModeCaseSensitive", + "params": [], + "returnType": "Void" + } + ], + "name": "ifAssociativeArray", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifassociativearray.md" + }, + "ifaudioguide": { + "description": "> Please note this component is only available on the following devices: Roku Streaming Stick (3600X), Roku Express (3700X) and Express+ (3710X), Roku Premiere (4620X) and Premiere+ (4630X), Roku Ultra (4640X), and any Roku TV running Roku OS version 7.5 and later.", + "implementers": [ + { + "description": "Returns information about the application", + "name": "roAudioGuide", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioguide.md" + } + ], + "methods": [ + { + "description": "Interrupts and stops any current text to speech spoken string, to be used when the application does not want the text to speech to continue.", + "name": "Flush", + "params": [] + }, + { + "description": "Returns an ID for the spoken string to notify observer callbacks about a specific spoken string. This ID can be used with the [roTextToSpeechEvent](https://developer.roku.com/docs/references/brightscript/events/rotexttospeechevent.md\"roTextToSpeechEvent\").This method will automatically split up text to reduce lag. Due to this automatic splitting, the roTextToSpeechEvent 0 (\"Started speech\") event for the returned ID may not be sent until later than expected. The roTextToSpeechEvents 1 (\"Speech has completed\") and 2 (\"Speech has been flushed\") events are sent at the expected times.", + "name": "Say", + "params": [ + { + "default": null, + "description": "The string to be spoken.", + "isRequired": true, + "name": "text", + "type": "String" + }, + { + "default": null, + "description": "Set to true to make the Audio Guide immediately stop speaking any other speech before speaking. Set to false to make the Audio Guide wait until any current speech is done before speaking.", + "isRequired": true, + "name": "flushSpeech", + "type": "Boolean" + }, + { + "default": null, + "description": "Set to true to ignore calls to the say() method with the same text. Set to false to speak when calls to the say() method are sent with the same text.", + "isRequired": true, + "name": "dontRepeat", + "type": "Boolean" + } + ], + "returnDescription": "An ID associated with the spoken string to be used to notify observer callbacks.", + "returnType": "Integer" + }, + { + "description": "If Audio Guide is enabled, causes text to speech to continue to suppress any application background sound for the amount of time specified by duration (in milliseconds).This can be used to add clarity for longer spoken text that may have pauses that might otherwise allow application background sound to be heard. This method does nothing if Audio Guide is currently disabled.", + "name": "Silence", + "params": [ + { + "default": null, + "description": "The number of milliseconds to suppress application background sounds.", + "isRequired": true, + "name": "duration", + "type": "Integer" + } + ], + "returnDescription": "The number of milliseconds that the background sound has been silenced.", + "returnType": "Integer" + } + ], + "name": "ifAudioGuide", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioguide.md" + }, + "ifaudiometadata": { + "implementers": [ + { + "description": "This component provides developers access to audio file metadata included in many audio files", + "name": "roAudioMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudiometadata.md" + } + ], + "methods": [ + { + "description": "Returns an associative array with a simple set of audio properties.", + "name": "GetAudioProperties", + "params": [], + "returnDescription": "An associative array that may be set to one of the following values (these are values that may involve reading a larger portion of the file and thus may take longer to retrieve than tags):", + "returnType": "Object" + }, + { + "description": "Returns the cover art, if available.", + "name": "GetCoverArt", + "params": [], + "returnDescription": "An associative array with two entries: \"bytes\" and \"type\".", + "returnType": "Object" + }, + { + "description": "Returns an associative array that contains a simple set of tags that are common to most audio formats.", + "name": "GetTags", + "params": [], + "returnDescription": "An associative array that may be set to one of the following values:", + "returnType": "Object" + }, + { + "description": "Sets the URL to the audio file. Only file URLs are initially supported", + "name": "SetUrl", + "params": [ + { + "default": null, + "description": "The URL of the audio file.", + "isRequired": true, + "name": "url", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifAudioMetaData", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudiometadata.md" + }, + "ifaudioplayer": { + "implementers": [ + { + "description": "The Audio Player object provides the ability to setup the playing of a series of audio streams", + "name": "roAudioMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudiometadata.md" + } + ], + "methods": [ + { + "description": "Adds a new ContentMetaData item to the end of the content list for the Audio Player.", + "name": "AddContent", + "params": [ + { + "default": null, + "description": "The new ContentMetaData item to be added to the content list.", + "isRequired": true, + "name": "contentItem", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Clears the content list.", + "name": "ClearContent", + "params": [], + "returnType": "Void" + }, + { + "description": "Puts the Audio Player into pause mode. It is an error to Pause if player is not in play mode.", + "name": "Pause", + "params": [], + "returnDescription": "A flag indicating whether the Audio Player was successfully set to pause mode.", + "returnType": "Boolean" + }, + { + "description": "Puts the Audio Player into play mode starting at the current item in the Content List. This will stop any currently playing content.", + "name": "Play", + "params": [], + "returnDescription": "A flag indicating whether the Audio Player was successfully set to play mode.", + "returnType": "Boolean" + }, + { + "description": "Puts the Audio Player into play mode starting from the pause point. It is an error to Resume if the player is not in pause mode.", + "name": "Resume", + "params": [], + "returnDescription": "A flag indicating whether the Audio Player was successfully set to play mode.", + "returnType": "Boolean" + }, + { + "description": "Set the start point of playback for the current item to offsetMs milliseconds.", + "name": "Seek", + "params": [ + { + "default": null, + "description": "The offset to be used to determine the start point of the current content item.", + "isRequired": true, + "name": "offsetMs", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the Audio Player was successfully set to the specified offset.", + "returnType": "Boolean" + }, + { + "description": "Sets the content list to be played by the Audio Player.", + "name": "SetContentList", + "params": [ + { + "default": null, + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Enables/disables the automatic replaying of the Content List.", + "name": "SetLoop", + "params": [ + { + "default": null, + "description": "Set to true to have the Audio Player automatically begin playing the first item in the content list after playing the last item. Set to false to have the Audio Player stop after playing the last item in the content list.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the next item in the Content List to be played.", + "name": "SetNext", + "params": [ + { + "default": null, + "description": "Item is the zero-based index of the item in the content list. This item will be played after the currently playing item finishes.", + "isRequired": true, + "name": "item", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the timedMetaData keys that the channel is interested in receiving from the timedMetaData event.", + "name": "SetTimedMetaDataForKeys", + "params": [ + { + "default": null, + "isRequired": true, + "name": "keys" + } + ] + }, + { + "description": "Stops the Audio Player from playing or pausing and cleanup.", + "name": "Stop", + "params": [], + "returnDescription": "A flag indicating whether the Audio Player was successfully stopped.", + "returnType": "Boolean" + } + ], + "name": "ifAudioPlayer", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioplayer.md" + }, + "ifaudioresource": { + "implementers": [ + { + "description": "The roAudioResouce allows .wav files to be cached to memory and quickly played at any time", + "name": "roAudioResource", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioresource.md" + } + ], + "methods": [ + { + "description": "Returns an [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") array containing the indicated metadata parameters about the audio resource.", + "name": "GetMetaData", + "params": [], + "returnDescription": "An associative array with the following integer values:", + "returnType": "Object" + }, + { + "description": "Checks whether this audio resource is currently playing.", + "name": "IsPlaying", + "params": [], + "returnDescription": "A flag indicating whether the calling audio resource is playing.", + "returnType": "Boolean" + }, + { + "description": "Returns the device-dependent maximum number of audio streams that can be mixed together and presented simultaneously.", + "name": "MaxSimulStreams", + "params": [], + "returnDescription": "Typically, 1-2.", + "returnType": "Integer" + }, + { + "description": "Stops playing the audio resource. If the resource is not currently playing, has no effect.", + "name": "Stop", + "params": [], + "returnType": "Void" + }, + { + "description": "This method triggers the start of the audio resource sound playback. The effect of Trigger(volume) is identical to Trigger(volume, 0).", + "name": "Trigger", + "params": [ + { + "default": null, + "description": "The volume is a number between 0 and 100 (percentage of full volume). A value of 50 should be used for normal volume.", + "isRequired": true, + "name": "volume", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Triggers the start of the audio resource sound playback. This method will interrupt any playing sound when the index is the same. It will mix with any playing sound if the index is different.", + "name": "Trigger", + "params": [ + { + "default": null, + "description": "The volume is a number between 0 and 100 (percentage of full volume). 50 should be used for normal volume.", + "isRequired": true, + "name": "volume", + "type": "Integer" + }, + { + "default": null, + "description": "The index is a value between 0 and [MaxSimulStreams()](#maxsimulstreams-as-integer) allowing for multiple sounds to be mixed.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnType": "Void" + } + ], + "name": "ifAudioResource", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifaudioresource.md" + }, + "ifboolean": { + "description": "Interface equivalent for intrinsic type Boolean.", + "implementers": [ + { + "description": "Object equivalent for intrinsic type Boolean", + "name": "roBoolean", + "url": "https://developer.roku.com/docs/references/brightscript/components/roboolean.md" + } + ], + "methods": [ + { + "description": "Gets the boolean value stored in the calling boolean object.", + "name": "GetBoolean", + "params": [], + "returnDescription": "The boolean value stored in the calling boolean object.", + "returnType": "Boolean" + }, + { + "description": "Sets the calling boolean object to the specified true/false value.", + "name": "SetBoolean", + "params": [ + { + "default": null, + "description": "True/false.", + "isRequired": true, + "name": "value", + "type": "Boolean" + } + ], + "returnType": "Void" + } + ], + "name": "ifBoolean", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifboolean.md" + }, + "ifbytearray": { + "implementers": [ + { + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes", + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + } + ], + "methods": [ + { + "description": "Appends the contents of the Byte Array to the specified file.", + "name": "AppendFile", + "params": [ + { + "default": null, + "description": "The path to the file to be appended to the ByteArray.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the file was successfully appended to the calling ByteArray.", + "returnType": "Boolean" + }, + { + "description": "Appends the contents of the Byte Array to the specified file.", + "name": "AppendFile", + "params": [ + { + "default": null, + "description": "The path to the file to be appended to the Byte Array.", + "isRequired": true, + "name": "path", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "start" + } + ], + "returnDescription": "A flag indicating whether the file was successfully appended to the calling ByteArray." + }, + { + "description": "Sets the contents of the Byte Array to the specified string using UTF-8 encoding. Any data currently in the Byte Array is discarded.", + "name": "FromAsciiString", + "params": [ + { + "default": null, + "description": "The string to which the ByteArray is to be set.", + "isRequired": true, + "name": "s", + "type": "String" + } + ] + }, + { + "description": "Sets the contents of the Byte Array to the specified value. Any data currently in the Byte Array is discarded.", + "name": "FromBase64String", + "params": [ + { + "default": null, + "description": "A valid base-64 encoding", + "isRequired": true, + "name": "s", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the contents of the Byte Array to the specified value. Any data currently in the Byte Array is discarded.", + "name": "FromHexString", + "params": [ + { + "default": null, + "description": "An even number of hexadecimal digits. The string must contain valid hexadecimal digits, or the result is undefined", + "isRequired": true, + "name": "hexstring", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Calculates a CRC-32 of the contents of the Byte Array.", + "name": "GetCRC32", + "params": [], + "returnDescription": "The calculated CRC-32 checksum.", + "returnType": "Integer" + }, + { + "description": "Calculates a CRC-32 of a subset of bytes within the Byte Array.", + "name": "GetCRC32", + "params": [ + { + "default": null, + "description": "The starting index of the subset of bytes to be used in the CRC-32 calculation.", + "isRequired": true, + "name": "start", + "type": "Integer" + }, + { + "default": null, + "description": "The length of the bytes to be included.", + "isRequired": true, + "name": "length", + "type": "Integer" + } + ], + "returnDescription": "The calculated CRC-32 checksum.", + "returnType": "Integer" + }, + { + "description": "Returns the signed byte at the specified zero-based index in the Byte ArrayUse the [ifArrayGet.GetEntry()](https://developer.roku.com/docs/references/brightscript/interfaces/ifarrayget.mdgetentryindex-as-integer-as-dynamic) method or the \\[ \\] array operator to read an unsigned byte in the Byte Array.", + "name": "GetSignedByte", + "params": [ + { + "default": null, + "description": "The index of the signed byte to be returned.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "The signed byte at the specified zero-based index in the Byte Array.", + "returnType": "Integer" + }, + { + "description": "Returns the signed long (four bytes) starting at the specified zero-based index in the Byte Array.", + "name": "GetSignedLong", + "params": [ + { + "default": null, + "description": "The index of the ByteArray from which to start retrieving the signed long.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "A signed long.", + "returnType": "Integer" + }, + { + "description": "Returns true if the CPU architecture is little-endian.", + "name": "IsLittleEndianCPU", + "params": [], + "returnDescription": "A flag indicating whether the CPU architecture is little-endian.", + "returnType": "Boolean" + }, + { + "description": "Reads the specified file into the Byte Array. Any data currently in the Byte Array is discarded.", + "name": "ReadFile", + "params": [ + { + "default": null, + "description": "The path to the file to be read.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the bytes were successfully read into the Byte Array.", + "returnType": "Boolean" + }, + { + "description": "Reads the specified file into the Byte Array. Any data currently in the Byte Array is discarded.", + "name": "ReadFile", + "params": [ + { + "default": null, + "description": "The path to the file to be read.", + "isRequired": true, + "name": "path", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "start" + } + ], + "returnDescription": "A flag indicating whether the bytes were successfully read into the Byte Array." + }, + { + "description": "If the size of the Byte Array is less than min\\_size, expands the Byte Array to min\\_size. Also sets the auto-resize attribute of the Byte Array to the specified value.", + "name": "SetResize", + "params": [ + { + "default": null, + "isRequired": true, + "name": "min" + } + ] + }, + { + "description": "Returns the contents of the Byte Array as a string. The contents must be valid UTF-8 (or ASCII subset), or the result is undefined", + "name": "ToAsciiString", + "params": [], + "returnDescription": "A String containing the contents of the ByteArray.", + "returnType": "String" + }, + { + "description": "Returns a base-64 string representing the contents of the Byte Array.", + "name": "ToBase64String", + "params": [], + "returnDescription": "A base-64 string representing the contents of the Byte Array.", + "returnType": "String" + }, + { + "description": "Returns a hexadecimal string representing the contents of the Byte Array, two digits per byte.", + "name": "ToHexString", + "params": [], + "returnDescription": "A hexadecimal string.", + "returnType": "String" + }, + { + "description": "Writes the bytes contained in the Byte Array to the specified file.", + "name": "WriteFile", + "params": [ + { + "default": null, + "description": "The path to the file to which the bytes are to be written.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the bytes were successfully written to the file.", + "returnType": "Boolean" + }, + { + "description": "Writes a subset of the bytes contained in the Byte Array to the specified file.", + "name": "WriteFile", + "params": [ + { + "default": null, + "description": "The path to the file to which the bytes are to be written.", + "isRequired": true, + "name": "path", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "start" + } + ], + "returnDescription": "A flag indicating whether the bytes were successfully written to the file." + } + ], + "name": "ifByteArray", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifbytearray.md" + }, + "ifcaptionrenderer": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "Returns information about the application", + "name": "roCaptionRenderer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocaptionrenderer.md" + } + ], + "methods": [ + { + "description": "The ChangeSubtitleTrack function is used to change the caption source after playback has begun.", + "name": "ChangeSubtitleTrack", + "params": [ + { + "default": null, + "description": "One of the 608 channels or ttml text tracks to be selected. The 608 channels are specified as 'eia608/' where is 1, 2, 3, or 4. The ttml text tracks are specified as 'ism/'.", + "isRequired": true, + "name": "track", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Returns an roArray of roAssociativeArrays for each closed caption track found in the stream. This function can also be used to retrieve caption URLs for external (sideloaded) caption files.", + "name": "GetSubtitleTracks", + "params": [], + "returnDescription": "Each associative array in the returned array contains track information similar to that specified in the SubtitleTracks content metadata structure:", + "returnType": "Object" + }, + { + "description": "Sets the [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md that should receive [roCaptionRendererEvents](https://developer.roku.com/docs/references/brightscript/events/rocaptionrendererevent.md from the roCaptionRenderer.", + "name": "SetMessagePort", + "params": [ + { + "default": null, + "isRequired": true, + "name": "port", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the rendering mode for the [roCaptionRenderer](https://developer.roku.com/docs/references/brightscript/components/rocaptionrenderer.md\"roCaptionRenderer\").", + "name": "SetMode", + "params": [ + { + "default": null, + "isRequired": true, + "name": "mode", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the [roScreen](https://developer.roku.com/docs/references/brightscript/components/roscreen.md or [roImageCanvas](https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md instance associated with this caption renderer.This function only needs to be called if the roCaptionRenderer mode is set to 1 (the default value). In this case, the Roku OS is responsible for all of the closed caption text rendering, and thus must know what screen to draw on.If the mode is set to 2, the BrightScript channel is responsible for all of the caption drawing, and thus the Roku OS does not need to be informed as to what screen is being rendered on.", + "name": "SetScreen", + "params": [ + { + "default": null, + "description": "The [roScreen](https://developer.roku.com/docs/references/brightscript/components/roscreen.md or [roImageCanvas](https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md instance to be associated with this caption renderer.", + "isRequired": true, + "name": "screen", + "type": "roScreen or roImageCanvas" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies whether the roCaptionRenderer displays captions. This function behaves the same as [ifVideoScreen.ShowSubtitle()](https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoscreen.md\"ifVideoScreen.ShowSubtitle/(/)\").", + "name": "ShowSubtitle", + "params": [ + { + "default": null, + "description": "A flag indicating whether the roCaptionRenderer displays captions.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "For roCaptionRenderer instances with the mode set to 1, this method tells the Roku OS to render the current caption. If the mode is 2, this function does nothing", + "name": "UpdateCaption", + "params": [], + "returnType": "Void" + } + ], + "name": "ifCaptionRenderer", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.md" + }, + "ifchannelstore": { + "implementers": [ + { + "description": "allows the application to perform a purchase of an In-Channel Product or upgrade a channel", + "name": "roChannelStore", + "url": "https://developer.roku.com/docs/references/brightscript/components/rochannelstore.md" + } + ], + "methods": [ + { + "name": "ClearOrder", + "params": [], + "returnType": "Void" + }, + { + "description": "Applies a change in quantity to one item in the current Order (shopping cart).", + "name": "DeltaOrder", + "params": [ + { + "default": null, + "description": "The product identifier.", + "isRequired": true, + "name": "code", + "type": "String" + }, + { + "default": null, + "description": "The quantity purchased. This may be a negative number.", + "isRequired": true, + "name": "qty", + "type": "Integer" + } + ], + "returnType": "Integer" + }, + { + "description": "Displays the Roku Channel Store Product Purchase Screen populated with information from the current Order.", + "name": "DoOrder", + "params": [], + "returnDescription": "A flag indicating whether the user approved the order (true if the order was approved; false otherwise).", + "returnType": "Boolean" + }, + { + "description": "This test mode short circuits communication to the Roku Channel store. It makes other methods get their responses to async queries and operations from configuration files, rather than actual server communication.", + "name": "FakeServer", + "params": [ + { + "default": null, + "description": "If enable is true, enables a test mode for the roChannelStore component.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Requests the list of In-Channel products that are linked to the running channel.", + "name": "GetCatalog", + "params": [], + "returnType": "Void" + }, + { + "description": "Retrieves a Roku Partner Unique Customer Identifier (roku\\_pucid), or retrieves an access token, oAuth token, or other authentication artifact (channel\\_data).", + "name": "GetChannelCred", + "params": [], + "returnDescription": "An associative array that contains the following fields:", + "returnType": "Object" + }, + { + "description": "Returns a unique number for this object that can be used to identify whether a roChannelStoreEvent event originated from this object, by comparing with the roChannelStoreEvent object's GetSourceIdentity() value.", + "name": "GetIdentity", + "params": [], + "returnDescription": "The unique number generated for the object.", + "returnType": "Integer" + }, + { + "description": "Retrieves the current Order.", + "name": "GetOrder", + "params": [], + "returnDescription": "The returned object is an roList of roAssociativeArray items, where each item contains the following parameter names with specified value type:", + "returnType": "Object" + }, + { + "description": "This function works like GetUserData(), but allows the caller to specify which user data elements to return. The specified values are also displayed in the user data dialog screen.", + "name": "GetPartialUserData", + "params": [ + { + "default": null, + "description": "A comma-separated list of the attribute names to be returned. For example, to return only the email address and first name of the user's account, you would call GetPartialUserData(\"email, firstname\"). The full set of user account properties that can be queried with the function is: * firstname * lastname * email * street * city * state * zip * country * phone * birth (_Available since Roku OS 10.0_) * gender (_Available since Roku OS 10.0_)", + "isRequired": true, + "name": "properties", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "requestInfo", + "type": "Object" + } + ], + "returnDescription": "An roAssociativeArray containing the Roku account information passed in the method.", + "returnType": "Object" + }, + { + "description": "Requests the list of purchases associated with the current user account.", + "name": "GetPurchases", + "params": [], + "returnType": "Void" + }, + { + "description": "Requests the list of globally available In-Channel products, which are available to all channels.", + "name": "GetStoreCatalog", + "params": [], + "returnType": "Void" + }, + { + "description": "The GetUserData() function provides a way to request user authorization to share the user’s account information with the calling channel.", + "name": "GetUserData", + "params": [], + "returnDescription": "When called, the method presents a dialog screen containing the user’s account information, along with two buttons labeled Share and Don’t Share.", + "returnType": "Object" + }, + { + "description": "Retrieves the state, zip code, and country associated with the customer's Roku account. The location information returned by this command can be used to determine a customer's eligibility for regional-specific subscription products and content.", + "name": "GetUserRegionData", + "params": [], + "returnDescription": "An associative array that contains the following fields:", + "returnType": "Object" + }, + { + "description": "Sets the current Order (shopping cart) to the elements specified in the parameter, which must be an roList of roAssociativeArray items.Passing an empty roList clears the Order, like calling ClearOrder().", + "name": "SetOrder", + "params": [ + { + "default": null, + "description": "Each roAssociativeArray in the roList contains the following fields:
NameTypeDescription
codeStringThe product identifier
qtyIntegerThe quantity purchased
", + "isRequired": true, + "name": "order", + "type": "roList of roAssociativeArray items" + }, + { + "default": null, + "description": "_Available since Roku OS 9.3_ This parameter is used for subscription upgrades and downgrades. If it is not specified, the action is a product purchase. It contains the following fields:
NameTypeDescription
actionStringThe action to be performed, which may be one of the following:
  • \"Upgrade\": The order is an upgrade from one subscription product to another.
  • \"Downgrade\": The order is a subscription downgrade.
**Example** ``` m.store = CreateObject(\"roChannelStore\")​ ' Populate myOrderItems myOrderInfo.action = \"Upgrade\" m.store.setOrder(myOrderItems, myOrderInfo) ``` See [On-device upgrade and downgrade](/docs/developer-program/roku-pay/implementation/on-device-upgrade-downgrade.md#calling-the-roku-web-service-validate-transaction-api) for how to implement Roku Pay web services for upgrades/downgrades.", + "isRequired": true, + "name": "orderInfo", + "type": "roAssociativeArray" + } + ], + "returnType": "Void" + }, + { + "description": "Stores an access token, oAuth token, or other authentication artifact that can be retrieved by calling the [GetChannelCred()](https://developer.roku.com/docs/references/brightscript/interfaces/ifchannelstore.mdgetchannelcred-as-object)method. This data is stored securely in the Roku cloud and can be retrieved by other devices linked to the same Roku account. This method can be used to store an authentication artifact with Roku for a signed in user, associating that user with a particular Roku account. For more information, see [Automatic Account Link](/docs/developer-program/authentication/universal-authentication-protocol-for-single-sign-on.md).", + "name": "StoreChannelCredData", + "params": [ + { + "default": null, + "description": "An OAuth token, custom token, or other custom data to be stored.", + "isRequired": true, + "name": "data", + "type": "String" + } + ], + "returnDescription": "This command returns an roAssociativeArray with the following values:", + "returnType": "Object" + } + ], + "name": "ifChannelStore", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifchannelstore.md" + }, + "ifcoderegistrationscreen": { + "deprecatedDescription": "This component is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Code Registration Screen is designed to present the user a registration code, and the information required to instruct the user on how to register with a service provider", + "name": "roCodeRegistrationScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocoderegistrationscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Adds a button to the screen identified by the title and ID provided. The buttons are at the bottom of the screen and appear in the order added", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID of the button.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "When the button is pressed, the script will receive an event from the application indicating the ID of the button pressed.", + "returnType": "Boolean" + }, + { + "description": "Adds high visibility focal text to the screen to be placed above the registration code.", + "name": "AddFocalText", + "params": [ + { + "default": null, + "description": "This text is intended to provide the user important instructions on where to use the registration code. It is generally a few words of instruction followed by the URL for the registration site on the web.", + "isRequired": true, + "name": "text", + "type": "String" + }, + { + "default": null, + "description": "Multiple lines of text may be added and the spacing between each is controlled by specifying the spacing format as one of the following: spacing-dense, spacing–normal or spacing-sparse.", + "isRequired": true, + "name": "spacingFormat", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Add a string of bold, high visibility text to the screen as a header to introduce the subsequent paragraph(s).", + "name": "AddHeaderText", + "params": [ + { + "default": null, + "description": "The text to be displayed in the header of the code registration screen.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Adds a paragraph of text to the screen. A paragraph is specified as a single string and are ordered on the screen in the same order as they are added. The roCodeRegistrationScreen handles all text formatting and justification. Spacing is automatically inserted between paragraphs for readability.", + "name": "AddParagraph", + "params": [ + { + "default": null, + "description": "The paragraph of text to be displayed on the code registration screen.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Breadcrumbs allow the application to display a two-part navigational title which shows the current and the previous locations in the application hierarchy (e.g. TV – Friends). If both location values are set, the application will display the title in breadcrumb format. If only the first location is set, the application will display the specified text in the title area like the SetTitle API call", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The second location value.", + "isRequired": true, + "name": "String", + "type": "location2" + } + ] + }, + { + "description": "Sets the registration code (e.g. XM3RT) or text (e.g. retrieving…) to be displayed on the screen.", + "name": "SetRegistrationCode", + "params": [ + { + "default": null, + "isRequired": true, + "name": "regCode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title for the screen to the specified string.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The title to be displayed on the code registration screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen is displayed (returns true if the screen is displayed, otherwise false).", + "returnType": "Boolean" + } + ], + "name": "ifCodeRegistrationScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifcoderegistrationscreen.md" + }, + "ifcompositor": { + "implementers": [ + { + "description": "The roCompositor allows the composition and animation of multiple roBitmaps and roRegions", + "name": "roCompositor", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocompositor.md" + } + ], + "methods": [ + { + "description": "Moves all animated sprites. Sprites will not animate unless you call this function regularly.", + "name": "AnimationTick", + "params": [ + { + "default": null, + "description": "The number of ms since the last call.", + "isRequired": true, + "name": "duration", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Provides a global search and replace of sprite [roRegions](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegions\"). Replaces regions that match oldRegion with newRegion", + "name": "ChangeMatchingRegions", + "params": [ + { + "default": null, + "description": "The sprite roRegion to be replaced.", + "isRequired": true, + "name": "oldRegion", + "type": "Object" + }, + { + "default": null, + "description": "The new sprite roRegion to be used.", + "isRequired": true, + "name": "newRegion", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Draws any dirty sprites (that is, whatever is new or has changed since the last Draw). No compositor or sprite operations will be reflected on the display until Draw() is called. After calling Draw(), you must call Finish() (if single buffered) or SwapBuffers() (if double buffered) before the changes will be user visible", + "name": "Draw", + "params": [], + "returnType": "Void" + }, + { + "description": "Redraws all sprites even if not dirty. After calling Draw(), you must call Finish() (if single buffered) or SwapBuffers() (if double buffered) before the changes will be user visible", + "name": "DrawAll", + "params": [], + "returnType": "Void" + }, + { + "description": "Creates a new sprite that consists of a sequence of frames to be animated. The frames are defined by the regionArray which is an [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roRegions](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegions\"). Position the sprite at coordinate x,y. If z is provided, position the sprite in front of all other sprites with equal or lower z value", + "name": "NewAnimatedSprite", + "params": [ + { + "default": null, + "description": "The x-coordinate of the sprite.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the sprite.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The frames to be animated.", + "isRequired": true, + "name": "regionArray", + "type": "Object" + }, + { + "default": null, + "description": "The z-coordinate of the sprite.", + "isRequired": true, + "name": "z", + "type": "Integer" + } + ], + "returnDescription": "Returns an [roSprite](/docs/references/brightscript/components/rosprite.md \"roSprite\") object.", + "returnType": "Object" + }, + { + "description": "Creates a new sprite, using an roRegion to define the sprite's bitmap. Position the sprite at coordinate x,y. If z is provided, position the sprite in front of all other sprites with equal or lower z value. Sprites with negative z values are not rendered or displayed on the screen.", + "name": "NewSprite", + "params": [ + { + "default": null, + "description": "The x-coordinate of the sprite.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the sprite.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The region to be used to define the sprite's bitmap.", + "isRequired": true, + "name": "region", + "type": "Object" + }, + { + "default": null, + "description": "The z-coordinate of the sprite.", + "isRequired": true, + "name": "z", + "type": "Integer" + } + ], + "returnDescription": "Returns an [roSprite](/docs/references/brightscript/components/rosprite.md \"roSprite\") object.", + "returnType": "Object" + }, + { + "description": "Sets the destBitmap ([roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or [roScreen](https://developer.roku.com/docs/references/brightscript/components/roscreen.md\"roScreen\")) and the background color.", + "name": "SetDrawTo", + "params": [ + { + "default": null, + "description": "The bitmap to be drawn.", + "isRequired": true, + "name": "destBitmap", + "type": "Object" + }, + { + "default": null, + "description": "The background color to be used.", + "isRequired": true, + "name": "rgbaBackground", + "type": "Integer" + } + ], + "returnType": "Void" + } + ], + "name": "ifCompositor", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifcompositor.md" + }, + "ifdatetime": { + "implementers": [ + { + "description": "The roDateTime provides an interface to obtain the current date/time for the player and manipulate date/times", + "name": "roDateTime", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatetime.md" + } + ], + "methods": [ + { + "description": "Returns the date/time formatted string.", + "name": "AsDateString", + "params": [ + { + "default": null, + "description": "
Format
long-date
short-weekday
no-weekday
short-month
short-month-short-weekday
short-month-no-weekday
short-date
short-date-dashes
", + "isRequired": true, + "name": "format", + "type": "String" + } + ], + "returnDescription": "A dateString corresponding to the specified format.", + "returnType": "String" + }, + { + "description": "Returns the date/time in long-date format.", + "name": "AsDateStringNoParam", + "params": [], + "returnDescription": "A date/time string in long-date format (for example, Tuesday October 9, 2012)", + "returnType": "String" + }, + { + "description": "Returns the date/time as the number of seconds from the Unix epoch (00:00:00 1/1/1970 GMT).", + "name": "AsSeconds", + "params": [], + "returnDescription": "Number of seconds as Integer.", + "returnType": "Integer" + }, + { + "description": "Sets the date/time using a string in the ISO 8601 format. For example \"YYYY-MM-DD HH:MM:SS\" e.g \"2009-01-01 01:00:00.000\" or \"2009-01-01T01:00:00.000\".", + "name": "FromISO8601String", + "params": [ + { + "default": null, + "description": "The ISO-8601 string to be used to set the date and time.", + "isRequired": true, + "name": "dateString", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the date/time value using the number of seconds from the Unix epoch.", + "name": "FromSeconds", + "params": [ + { + "default": null, + "description": "The number of seconds from the Unix epoch.", + "isRequired": true, + "name": "numSeconds", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Returns the date/time value's day of the month.", + "name": "GetDayOfMonth", + "params": [], + "returnDescription": "Month value as an Integer (1-31).", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's day of week.", + "name": "GetDayOfWeek", + "params": [], + "returnDescription": "Day value as an Integer (Sunday=0, Monday=1, ..., Saturday=6).", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's hour within the day.", + "name": "GetHours", + "params": [], + "returnDescription": "Hour value as an Integer (0-23)", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's last day of the month.", + "name": "GetLastDayOfMonth", + "params": [], + "returnDescription": "Day as an Integer (28-31)", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's millisecond within the second.", + "name": "GetMilliseconds", + "params": [], + "returnDescription": "Millisecond value as an Integer (0-999).", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's minute within the hour.", + "name": "GetMinutes", + "params": [], + "returnDescription": "Minute value as an Integer (0-59)", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's month.", + "name": "GetMonth", + "params": [], + "returnDescription": "Month value as an Integer (1=Jan, 12=Dec).", + "returnType": "Integer" + }, + { + "description": "Returns the date/time value's second within the minute.", + "name": "GetSeconds", + "params": [], + "returnDescription": "Second value as an Integer (0-59).", + "returnType": "Integer" + }, + { + "description": "Returns the offset in minutes from the system time zone to UTC. For example, if the system time zone is in PDT / UTC-7 the value returned would be 420.", + "name": "GetTimeZoneOffset", + "params": [], + "returnDescription": "Minutes of offset as Integer.", + "returnType": "Integer" + }, + { + "description": "Returns the day of the week.", + "name": "GetWeekday", + "params": [], + "returnDescription": "Week value as a String (e.g. \"Monday\").", + "returnType": "String" + }, + { + "description": "Return Value", + "name": "GetYear", + "params": [], + "returnType": "Integer" + }, + { + "description": "Sets the date/time value to the current UTC date and time.", + "name": "Mark", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns an ISO 8601 representation of the date/time value. As of Roku OS 10.0, this now resolves to milliseconds.", + "name": "ToISOString", + "params": [], + "returnDescription": "ISO 8601 as String, e.g. \"2021-03-25T18:53:03+0000\"", + "returnType": "String" + }, + { + "description": "Offsets the date/time value from an assumed UTC date/time to a local date/time using the system time zone setting. This function is not idempotent, and multiple calls will do multiple timezone adjustments to the time yielding an incorrect result.", + "name": "ToLocalTime", + "params": [], + "returnType": "Void" + } + ], + "name": "ifDateTime", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdatetime.md" + }, + "ifdevicecrypto": { + "implementers": [ + { + "description": "Encrypts and decrypts data on a device using a key that is unique per channel, device, or model.", + "name": "roDeviceCrypto", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodevicecrypto.md" + } + ], + "methods": [], + "name": "ifDeviceCrypto", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdevicecrypto.md" + }, + "ifdeviceinfo": { + "implementers": [ + { + "description": "The roDeviceInfo component provides an interface to obtain attributes about the device", + "name": "roDeviceInfo", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodeviceinfo.md" + } + ], + "methods": [ + { + "description": "Checks if the device can decode and play the specified audio format.", + "name": "CanDecodeAudio", + "params": [ + { + "default": null, + "isRequired": true, + "name": "audio" + } + ], + "returnDescription": "An associative array that includes a flag indicating whether the audio format can be played, and the closest audio format supported by the device." + }, + { + "description": "Checks whether the device can decode and play the specified video format.", + "name": "CanDecodeVideo", + "params": [ + { + "default": null, + "isRequired": true, + "name": "video" + } + ], + "returnDescription": "An associative array that includes a flag indicating whether the video format can be played, and the closest video format supported by the device." + }, + { + "description": "Notifies the channel when a system overlay event (such as the [confirm partner button HUD](/docs/developer-program/getting-started/architecture/channel-manifest.md#special-purpose-attributes) or the caption control overlay) is displayed. This notification gives the channel the opportunity to do any processing they may want to when the channel loses or regains focus.", + "name": "EnableAppFocusEvent", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable/disable system overlay event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the system overlay event notifications are enabled (true) or disabled (false).", + "returnType": "Dynamic" + }, + { + "description": "Notifies the channel when the audio guide changes. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md when the audio guide changes. To receive events, you must have first called [SetMessagePort](https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md on the roDeviceInfo object specifying the message port that is to receive the events", + "name": "EnableAudioGuideChangedEvent", + "params": [ + { + "default": null, + "description": "A flag indicating whether to enable/disable audio guide change event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether audio guide change event notifications are enabled (true) or disabled (false).", + "returnType": "Dynamic" + }, + { + "description": "Notifies the channel when the audio or video codec changes. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md when the codec changes. To receive events, you must have first called [SetMessagePort](https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md on the roDeviceInfo object specifying the message port that is to receive the events", + "name": "EnableCodecCapChangedEvent", + "params": [ + { + "default": null, + "description": "A flag indicating whether to enable/disable codec change event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether codec change event notifications are enabled (true) or disabled (false)." + }, + { + "description": "Notifies the channel when an internet connection status event occurs. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md) when the network connection status changes, as indicated by `roDeviceInfoEvent.internetStatus`. To receive events, the channel must have first called [SetMessagePort](https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md) on the roDeviceInfo object specifying the message port that is to receive the events.", + "name": "EnableInternetStatusEvent", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable/disable network connection status event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether network connection status event notifications are enabled (true) or disabled (false).", + "returnType": "Boolean" + }, + { + "description": "Notifies the channel when a network connection status event occurs. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md when the network connection status changes. To receive events, you must have first called [SetMessagePort](https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md on the roDeviceInfo object specifying the message port that is to receive the events", + "name": "EnableLinkStatusEvent", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable/disable network connection status event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether network connection status event notifications are enabled (true) or disabled (false).", + "returnType": "Boolean" + }, + { + "description": "Notifies the channel when a lowGeneralMemoryLevel event occurs. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md when a low general memory event occurs.", + "name": "EnableLowGeneralMemoryEvent", + "params": [ + { + "default": null, + "isRequired": true, + "name": "enabled", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether lowGeneralMemoryLevel event notifications are enabled (true) or disabled (false).", + "returnType": "Dynamic" + }, + { + "description": "Notifies the channel when a screensaver exit event occurs. This function enables the sending of an [roDeviceInfoEvent](https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md when a user has exited the screensaver.", + "name": "EnableScreensaverExitedEvent", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable/disable screensaver exit event notifications.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether screensaver exit event notifications are enabled (true) or disabled (false).", + "returnType": "Dynamic" + }, + { + "description": "Forces a new internet connection check. A new check will only be initiated if the cached internet status is older than 10 seconds.", + "name": "ForceInternetStatusCheck", + "params": [], + "returnDescription": "True indicates only that a new internet check has been initiated; otherwise, false. To get the actual internet connection status, use the [**GetInternetStatus()**](getinternetstatus-as-boolean) method.", + "returnType": "Boolean" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use the [GetRIDA()](/docs/references/brightscript/interfaces/ifdeviceinfo.md#getrida-as-string) method to get the unique identifier.\n", + "description": "Returns a unique identifier for the device. This identifier is persistent but can be reset by the user from the device's Settings menu or by performing a factory reset on the device.", + "isDeprecated": true, + "name": "GetAdvertisingId", + "params": [], + "returnDescription": "A Universally Unique Identifier (UUID) as specified in IETF-RFC 4122 with 36 characters (32 alphanumeric characters and four hyphens). The characters are grouped in the form 8-4-4-4-12, for example \"123e4567-e89b-12d3-a456-426655440000\"", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nChannel developers should use the [CanDecodeAudio()](#candecodeaudioaudio_format-as-object-as-object) function instead.\n", + "description": "Lists each audio decoder supported by the device, with up to four numbers describing the decoder from the EDID SAD (Short Audio Descriptor). Each value is of the form \"::::\"", + "isDeprecated": true, + "name": "GetAudioDecodeInfo", + "params": [], + "returnDescription": "An associative array with EDID (EIA.2FCEA-861) audio decoder information for the device connected to the HDMI port (or the device itself for a Roku TV).", + "returnType": "Object" + }, + { + "description": "Checks for the type of audio output.", + "name": "GetAudioOutputChannel", + "params": [], + "returnDescription": "The selected audio output, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Determines whether global captions are turned on or off, or are in instant replay mode.", + "name": "GetCaptionsMode", + "params": [], + "returnDescription": "The current global setting for the Mode property, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks the current value of the specified global setting property.", + "name": "GetCaptionsOption", + "params": [ + { + "default": null, + "description": "The global setting property to be checked, which may be one of the following values: * Mode * Text/Font * Text/Effect * Text/Size * Text/Color * Text/Opacity * Background/Color * Background/Opacity * Window/Color * Window/Opacity * Track * Track\\_Composite * Track\\_Analog * Muted", + "isRequired": true, + "name": "option", + "type": "String" + } + ], + "returnDescription": "The value of the specified global setting property, which may be as follows:", + "returnType": "String" + }, + { + "description": "Returns a unique identifier for the device. The ID is persistent and cannot be reset. This value can be used to manage or identify devices linked to the channel’s content services.", + "name": "GetChannelClientId", + "params": [], + "returnDescription": "A unique device identifier. This identifier is different across channels so each channel will get a different identifier when calling this function", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use the [GetChannelClientId](/docs/references/brightscript/interfaces/ifdeviceinfo.md#getchannelclientid-as-string) method to get the unique identifier.\n", + "description": "Returns a unique identifier for the device.", + "isDeprecated": true, + "name": "GetClientTrackingId", + "params": [], + "returnDescription": "A unique device identifier. This identifier is different across channels so each channel will get a different identifier when calling this function", + "returnType": "String" + }, + { + "description": "Checks whether the system settings for Time (**Setting > System > Time**) is set to a 12 or 24-hour format.", + "name": "GetClockFormat", + "params": [], + "returnDescription": "The time format:", + "returnType": "String" + }, + { + "description": "Checks for the information associated with the hardware's connection", + "name": "GetConnectionInfo", + "params": [], + "returnDescription": "An associative array with the following key-value pairs:", + "returnType": "Object" + }, + { + "description": "Checks whether the device has a WiFi or wired connection, or if it is not connected through any type of network.", + "name": "GetConnectionType", + "params": [], + "returnDescription": "The type of internet connection the device is using. This may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks for the country code of the channel.", + "name": "GetCountryCode", + "params": [], + "returnDescription": "A value that indicates the Roku Channel Store associated with a user’s Roku account. Typically, the value returned will be an ISO 3166-1 (2-letter) country code representing the country. Alternatively, if the channel owner entered into an additional agreement to have the channel published to a curated [Roku Powered Channel Store](https://www.roku.com/roku-powered) instead of the user country, then a Roku Powered Channel Store Identifier will instead be returned. This may be one of the following values:", + "returnType": "String" + }, + { + "description": "Gets the current locale value based on the user's language setting.", + "name": "GetCurrentLocale", + "params": [], + "returnDescription": "A string representing the current locale based on the user's language setting. The string is an ISO 639-1 (2-letter) language code followed by an underscore and a ISO 3166-1 (2-letter) country code. This may be one of the following values:", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use the 32-character alphanumeric unique identifier returned by [GetChannelClientId()](/docs/references/brightscript/interfaces/ifdeviceinfo.md#getchannelclientid-as-string).\n", + "description": "Returns a string of 12 zeroes (it no longer returns the unique identifier for the channel on a device).", + "isDeprecated": true, + "name": "GetDeviceUniqueId", + "params": [], + "returnDescription": "A string of 12 zeros (\"000000000000\")", + "returnType": "String" + }, + { + "description": "Checks the aspect ratio for the display screen.", + "name": "GetDisplayAspectRatio", + "params": [], + "returnDescription": "The aspect ratio, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks the UI resolution of the device.", + "name": "GetDisplayMode", + "params": [], + "returnDescription": "The configured graphics layer resolution, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks for the display properties of the screen.", + "name": "GetDisplayProperties", + "params": [], + "returnDescription": "An associative array with the following key/value pairs for the display properties of the screen:", + "returnType": "Object" + }, + { + "description": "Checks the display size of a screen.", + "name": "GetDisplaySize", + "params": [], + "returnDescription": "An associative array with the screen width and height. Specifically, the keys \"w\" and \"h\" contain the values for the screen width and height respectively, either 720 and 480, or 1280 and 720", + "returnType": "Object" + }, + { + "description": "Gets the text corresponding to the button selection in the Player Info Settings/Display Type page.", + "name": "GetDisplayType", + "params": [], + "returnDescription": "The display type, which may be one of the following values:", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use the replacement API [GetDrmInfoEx()](/docs/references/brightscript/interfaces/ifdeviceinfo.md#getdrminfoex-as-object) to return the supported DRM system and features.\n", + "description": "Checks for the supported DRM system and its features.", + "isDeprecated": true, + "name": "GetDrmInfo", + "params": [], + "returnDescription": "An associative array with the supported DRM system and features. For example, a device that supports PlayReady inside a trusted environment with secure stop returns:", + "returnType": "Object" + }, + { + "name": "GetDrmInfoEx", + "params": [], + "returnType": "Object" + }, + { + "description": "Checks the IP address assigned to the device by your internet service provider (ISP). This IP address is visible to the internet and all other computers outside your local network.", + "name": "GetExternalIp", + "params": [], + "returnDescription": "The external IP address assigned to the device.", + "returnType": "String" + }, + { + "description": "Returns a string describing the device that may be used for network device selection. The string is subject to change and should not be used as a persistent key or ID", + "name": "GetFriendlyName", + "params": [], + "returnDescription": "A user-assigned device name or a description of the device such as model name and/or serial number.", + "returnType": "String" + }, + { + "description": "Checks the general memory levels of the channel.", + "name": "GetGeneralMemoryLevel", + "params": [], + "returnDescription": "Returns the general memory levels of the channel, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks the graphics platform of the device.", + "name": "GetGraphicsPlatform", + "params": [], + "returnDescription": "The device's graphics platform, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Checks the internet connection status of the device.", + "name": "GetInternetStatus", + "params": [], + "returnDescription": "True if the cached internet status shows a connection; false, otherwise.", + "returnType": "Boolean" + }, + { + "description": "Checks the local IP address of the device. This can be used in conjunction with the ECP (see the External Control Protocol Guide) \"launch\" command (or the \"install\" command for uninstalled channels) to start a different channel from the current channel.", + "name": "GetIPAddrs", + "params": [], + "returnDescription": "An associative array, where each key is the name of a network interface and the value is the IP-address of the interface. Typically, the associative array only contains a single interface.", + "returnType": "Object" + }, + { + "description": "Checks if the device has an active connection.", + "name": "GetLinkStatus", + "params": [], + "returnDescription": "A flag indicating whether the device has an active connection.", + "returnType": "Boolean" + }, + { + "description": "Returns the model name of the Roku device. See the [Hardware Specification](/docs/specs/hardware.md) for the list of the current, updatable, and legacy Roku models.", + "name": "GetModel", + "params": [], + "returnDescription": "A five-character alphanumeric string (for example, \"3050X\") .", + "returnType": "String" + }, + { + "description": "Returns detailed information about the device model.", + "name": "GetModelDetails", + "params": [], + "returnDescription": "An associative array containing the following information about the device model:", + "returnType": "Object" + }, + { + "description": "Returns the model display name of the Roku device.", + "name": "GetModelDisplayName", + "params": [], + "returnDescription": "The model display name (for example, \"Roku 2 XD\")", + "returnType": "String" + }, + { + "description": "Returns a string describing the type of device. For future compatibility, the caller should by default assume \"STB\" when anything other than described value is returned", + "name": "GetModelType", + "params": [], + "returnDescription": "The device type, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Returns an roAssociativeArray containing the **major**, **minor**, **revision**, and **build** numbers of the Roku OS running on the device.", + "name": "GetOSVersion", + "params": [], + "returnDescription": "An roAssociativeArray containing the following fields:", + "returnType": "Object" + }, + { + "description": "Checks the three-letter ISO 639-2 language terminology code of the preferred caption language set on the Roku device.", + "name": "GetPreferredCaptionLanguage", + "params": [], + "returnDescription": "The three-letter ISO 639-2 language terminology code, which may be one of the following values:", + "returnType": "String" + }, + { + "description": "Returns a randomly generated unique identifier. Each time this function is called, a different identifier is returned", + "name": "GetRandomUUID", + "params": [], + "returnDescription": "A Universally Unique Identifier (UUID) version 4 as specified in IETF-RFC 4122 with 36 characters (32 alphanumeric characters and four hyphens). The characters are grouped in the form 8-4-4-4-12, for example \"123e4567-e89b-12d3-a456-426655440000\"", + "returnType": "String" + }, + { + "description": "Returns a unique identifier for the device.", + "name": "GetRIDA", + "params": [], + "returnDescription": "A Universally Unique Identifier (UUID). This identifier is persistent, but it can be reset by the user from the device's **Settings** menu or by performing a factory reset on the device", + "returnType": "String" + }, + { + "description": "Checks for the user interface sound effects volume level.", + "name": "GetSoundEffectsVolume", + "params": [], + "returnDescription": "The UI sounds effects volume as a percentage. A return value of 0 indicates that UI sound effects are muted, and a value of 100 indicates that they are set to the maximum volume level", + "returnType": "Integer" + }, + { + "description": "Checks the supported graphics resolutions.", + "name": "GetSupportedGraphicsResolutions", + "params": [], + "returnDescription": "A list of associative arrays. Each associative array contains the following key/value pairs for the graphics resolutions:", + "returnType": "Object" + }, + { + "description": "Checks for the user's current system time zone setting.", + "name": "GetTimeZone", + "params": [], + "returnDescription": "A string representing the user's current system time zone setting. For example, this method may return values such as:", + "returnType": "String" + }, + { + "description": "Checks for the UI resolution of the screen.", + "name": "GetUIResolution", + "params": [], + "returnDescription": "An associative array with the following key-value pairs describing the current UI resolution:", + "returnType": "Object" + }, + { + "description": "Returns the ISO 3166-1 (2-letter) country code associated with the user's Roku account.", + "name": "GetUserCountryCode", + "params": [], + "returnDescription": "An ISO 3166-1 (2-letter) country code.", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use [GetOSVersion()](/docs/references/brightscript/interfaces/ifdeviceinfo.md#getosversion-as-object) method to get the current Roku OS version running on a device.\n", + "description": "Returns the version number of the device.", + "isDeprecated": true, + "name": "GetVersion", + "params": [], + "returnDescription": "A 13-character string (for example \"034.08E01185A\"). The third through sixth characters are the major/minor version number (\"4.08\") and the ninth through twelfth are the build number (\"1185\")", + "returnType": "String" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nChannel developers should use the [CanDecodeVideo()](#candecodevideovideo_format-as-object-as-object) function instead.\n", + "description": "See [http://en.wikipedia.org/wiki/Extended\\_display\\_identification\\_data#EIA.2FCEA-861\\_extension\\_block](http://en.wikipedia.org/wiki/Extended_display_identification_data#EIA.2FCEA-861_extension_block) for an explanation of the information returned.", + "isDeprecated": true, + "name": "GetVideoDecodeInfo", + "params": [], + "returnDescription": "An associative array with the EDID (EIA.2FCEA-861) information describing the video display", + "returnType": "Object" + }, + { + "description": "Checks the video playback resolution.", + "name": "GetVideoMode", + "params": [], + "returnDescription": "The video playback resolution, which maybe one of the following values:", + "returnType": "String" + }, + { + "description": "Checks if the current device/firmware supports the passed in feature string.", + "name": "HasFeature", + "params": [ + { + "default": null, + "description": "The feature to be checked, which may be one of the following values: * \"5.1\\_surround\\_sound\" * \"can\\_output\\_5.1\\_surround\\_sound\" * \"sd\\_only\\_hardware\" * \"usb\\_hardware\" * \"sdcard\\_hardware\" * \"ethernet\\_hardware\" * \"gaming\\_hardware\" * \"energy\\_star\\_compliant\" * \"soundbar\\_hardware\"", + "isRequired": true, + "name": "feature", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the current device/firmware supports the passed in feature string.", + "returnType": "Boolean" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nDevelopers must update their channels to use [IsRIDADisabled()](/docs/references/brightscript/interfaces/ifdeviceinfo.md#isridadisabled-as-boolean) to get the Ad Id tracking status.\n", + "description": "If Ad Id tracking is disabled, the identifier returned by GetAdvertisingId() should not be used for Ad targeting", + "isDeprecated": true, + "name": "IsAdIdTrackingDisabled", + "params": [], + "returnDescription": "Returns true if the user has disabled Ad Id tracking by selecting \"Limit ad tracking\" from the Roku Settings menu, false otherwise.", + "returnType": "Boolean" + }, + { + "description": "Checks if the audio guide is enabled.", + "name": "IsAudioGuideEnabled", + "params": [], + "returnDescription": "A flag indicating whether the audio guide is enabled.", + "returnType": "Dynamic" + }, + { + "deprecatedDescription": "**This method is deprecated**.\n\nChannel developers must use the [ifHdmiStatus](/docs/references/brightscript/interfaces/ifhdmistatus.md) interface functions instead.\n", + "description": "Checks for an HDMI connection.", + "isDeprecated": true, + "name": "IsHDMIConnected", + "params": [], + "returnDescription": "A flag indicating whether an HDMI connection to a TV has been detected.", + "returnType": "Boolean" + }, + { + "description": "Indicates whether tracking via Roku's ID for Advertisers (RIDA) is disabled on the device.", + "name": "IsRIDADisabled", + "params": [], + "returnDescription": "A flag indicating whether RIDA tracking is disabled on the device (RIDA tracking can be disabled by selecting \"Limit ad tracking\" from the **Settings>Privacy>Advertising** menu). If RIDA tracking is disabled, this returns true; false otherwise.", + "returnType": "Boolean" + }, + { + "description": "Checks whether the device is in demo mode.", + "name": "IsStoreDemoMode", + "params": [], + "returnDescription": "A flag indicating whether the device is in demo mode.", + "returnType": "Boolean" + }, + { + "description": "Sets the current global setting for the Mode property.", + "name": "SetCaptionsMode", + "params": [ + { + "default": null, + "description": "The current global setting for the Mode property, which may be one of the following values: * On * Off * Instant replay * When mute (Roku TVs only)", + "isRequired": true, + "name": "mode", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the Mode property was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Checks for the number of seconds passed since the last remote keypress.", + "name": "TimeSinceLastKeypress", + "params": [], + "returnDescription": "The number of seconds since the last remote keypress was received.", + "returnType": "Integer" + } + ], + "name": "ifDeviceInfo", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdeviceinfo.md" + }, + "ifdouble": { + "description": "> Interface equivalent for intrinsic type Double", + "implementers": [ + { + "description": "roDouble is a legacy object name, corresponding to the intrinsic Double object", + "name": "roDouble", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodouble.md" + } + ], + "methods": [ + { + "description": "Gets the double value stored in the calling Double object.", + "name": "GetDouble", + "params": [], + "returnDescription": "The double value stored in the calling Double object.", + "returnType": "Double" + }, + { + "description": "Sets the calling Double object to the specified double value.", + "name": "SetDouble", + "params": [ + { + "default": null, + "description": "The double value to be set on the calling Double object.", + "isRequired": true, + "name": "value", + "type": "Double" + } + ], + "returnType": "Void" + } + ], + "name": "ifDouble", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdouble.md" + }, + "ifdraw2d": { + "description": "Coordinates (x,y) for this interface are based on an origin (0,0) at the top, left. (This is common for 2D drawing APIs, but is different than OpenGL's default coordinate system).\n\nBitmap pixel values and color values are always represented as 32-bit integer RGBA color values. That is, red is in the most significant byte and alpha is in the least significant byte.", + "implementers": [ + { + "description": "The roBitmap component contains image data and provides an interface (ifDraw2D) for drawing", + "name": "roBitmap", + "url": "https://developer.roku.com/docs/references/brightscript/components/robitmap.md" + }, + { + "description": "The roRegion component is used to represent a subsection of a bitmap", + "name": "roRegion", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregion.md" + }, + { + "description": "The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from", + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + } + ], + "methods": [ + { + "description": "Clears the bitmap, and fills it with the specified RGBA color.", + "name": "Clear", + "params": [ + { + "default": null, + "description": "The RGBA color to be used to fill the bitmap.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Draws a line from (xStart, yStart) to (xEnd, yEnd) with RGBA color.", + "name": "DrawLine", + "params": [ + { + "default": null, + "description": "The x-coordinate of the line's start point.", + "isRequired": true, + "name": "xStart", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the line's start point.", + "isRequired": true, + "name": "yStart", + "type": "Integer" + }, + { + "default": null, + "description": "The x-coordinate of the line's end point.", + "isRequired": true, + "name": "xEnd", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the line's end point.", + "isRequired": true, + "name": "yEnd", + "type": "Integer" + }, + { + "default": null, + "description": "The RGBA color of the line.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Draws the source object, where src is an [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object, at position x,y.", + "name": "DrawObject", + "params": [ + { + "default": null, + "description": "The x-coordinate of the source object.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the source object.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object to be drawn.", + "isRequired": true, + "name": "src", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the object was successfully drawn.", + "returnType": "Boolean" + }, + { + "description": "Draws a point at (x,y) with the given size and RGBA color.", + "name": "DrawPoint", + "params": [ + { + "default": null, + "description": "The x-coordinate of the point.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the point.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The size of the point.", + "isRequired": true, + "name": "size", + "type": "Float" + }, + { + "default": null, + "description": "The RGBA color of the point.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Fills the specified rectangle from left (x), top (y) to right (x + width), bottom (y + height) with the RGBA color.", + "name": "DrawRect", + "params": [ + { + "default": null, + "description": "The x-coordinate of the rectangle.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the rectangle.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the rectangle.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the rectangle.", + "isRequired": true, + "name": "height", + "type": "Integer" + }, + { + "default": null, + "description": "The RGBA color to be used to fill the rectangle.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Draws the source object, where src is an [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object, at position x,y rotated by angle theta degrees.", + "name": "DrawRotatedObject", + "params": [ + { + "default": null, + "description": "The x-coordinate of the source object.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the source object.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The position which to rotate the source object. This may be 0, 90, 180, and 270 degrees.", + "isRequired": true, + "name": "theta", + "type": "Float" + }, + { + "default": null, + "description": "The [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object to be drawn.", + "isRequired": true, + "name": "src", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the object was successfully drawn.", + "returnType": "Boolean" + }, + { + "description": "Draws the source object, where src is an [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object, at position x,y, scaled in the x direction by scaleX and in the y direction by scaleY. scaleX and scaleY should each be greater than zero and less than one to reduce the object size, or greater than one to increase the object size", + "name": "DrawScaledObject", + "params": [ + { + "default": null, + "description": "The x-coordinate of the source object.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the source object.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The x direction in which the source object is to be scaled.", + "isRequired": true, + "name": "scaleX", + "type": "Float" + }, + { + "default": null, + "description": "The y direction in which the source object is to be scaled.", + "isRequired": true, + "name": "scaleY", + "type": "Float" + }, + { + "default": null, + "description": "The [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object to be drawn.", + "isRequired": true, + "name": "src", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the object was successfully drawn.", + "returnType": "Boolean" + }, + { + "description": "Draws the source object, where src is an [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object, at position x,y, scaled in the x direction by scaleX and in the y direction by scaleY. scaleX and scaleY should each be greater than zero and less than one to reduce the object size, or greater than one to increase the object size.", + "name": "DrawScaledObject", + "params": [ + { + "default": null, + "description": "The x-coordinate of the source object.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the source object.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The x direction in which the source object is to be scaled.", + "isRequired": true, + "name": "scaleX", + "type": "Float" + }, + { + "default": null, + "description": "The y direction in which the source object is to be scaled.", + "isRequired": true, + "name": "scaleY", + "type": "Float" + }, + { + "default": null, + "description": "The [roBitmap](https://developer.roku.com/docs/references/brightscript/components/robitmap.md\"roBitmap\") or an [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object to be drawn.", + "isRequired": true, + "name": "src", + "type": "Object" + }, + { + "default": null, + "description": "The RGBA color of the source object.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the object was successfully drawn.", + "returnType": "Boolean" + }, + { + "description": "Draws the text at position (x,y) using the specified RGBA color and [roFont](https://developer.roku.com/docs/references/brightscript/components/rofont.md\"roFont\") font object. Text is drawn anti-aliased. The background image/color behind the text will show through the spaces and holes in the text. To have the text erase the background, make a call to [DrawRect()](#drawrectx-as-integer-y-as-integer-width-as-integer-height-as-integer-rgba-as-integer-as-void) before calling DrawText(). The size, bold, and italic attributes are specified when creating the [roFont](https://developer.roku.com/docs/references/brightscript/components/rofont.md\"roFont\").", + "name": "DrawText", + "params": [ + { + "default": null, + "description": "The text to be drawn.", + "isRequired": true, + "name": "text", + "type": "String" + }, + { + "default": null, + "description": "The x-coordinate of the source object.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the source object.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The color of the text.", + "isRequired": true, + "name": "rgba", + "type": "Integer" + }, + { + "default": null, + "description": "The [roFont](https://developer.roku.com/docs/references/brightscript/components/rofont.md\"roFont\") object to be used for the text.", + "isRequired": true, + "name": "font", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the object was successfully drawn.", + "returnType": "Boolean" + }, + { + "description": "Realizes the bitmap by finishing all queued draw calls. Until Finish() is called, prior graphics operations may not be user visible. For example, they may be in the graphics display pipeline, or in a server queue.", + "name": "Finish", + "params": [], + "returnType": "Void" + }, + { + "description": "Checks if the alpha blending is enabled.", + "name": "GetAlphaEnable", + "params": [], + "returnDescription": "A flag indicating whether alpha blending is enabled.", + "returnType": "Boolean" + }, + { + "description": "Gets the RGBA pixel values for the specified rectangle.", + "name": "GetByteArray", + "params": [ + { + "default": null, + "description": "The x-coordinate of the rectangle.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the rectangle.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the rectangle.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the rectangle.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnDescription": "An roByteArray representing the RGBA pixel values for the specified rectangle.", + "returnType": "Object" + }, + { + "description": "Gets the height of the bitmap in pixels.", + "name": "GetHeight", + "params": [], + "returnDescription": "The height of the bitmap in pixels.", + "returnType": "Integer" + }, + { + "description": "Gets PNG image data for the specified area of the bitmap. The PNG is in 32-bit RGBA format.", + "name": "GetPng", + "params": [ + { + "default": null, + "description": "The x-coordinate of the rectangle.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the rectangle.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the rectangle.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the rectangle.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnDescription": "An roByteArray object containing PNG image data for the specified area of the bitmap. If the coordinates are out of bounds, or the PNG conversion fails for any reason, then invalid is returned", + "returnType": "Object" + }, + { + "description": "Gets the width of the bitmap.", + "name": "GetWidth", + "params": [], + "returnDescription": "The width of the bitmap in pixels.", + "returnType": "Integer" + }, + { + "description": "Enables alpha blending when the source bitmap is the destination. The setting of the source bitmap's alpha enable is ignored.", + "name": "SetAlphaEnable", + "params": [ + { + "default": null, + "description": "A flag specifying whether alpha blending is enabled.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + } + ], + "name": "ifDraw2D", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifdraw2d.md" + }, + "ifenum": { + "implementers": [ + { + "description": "An array stores an indexed collection of BrightScript objects", + "name": "roArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roarray.md" + }, + { + "description": "An associative array (also known as a map, dictionary or hash table) allows objects to be associated with string keys", + "name": "roAssociativeArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md" + }, + { + "description": "The byte array component is used to contain and manipulate an arbitrary array of bytes", + "name": "roByteArray", + "url": "https://developer.roku.com/docs/references/brightscript/components/robytearray.md" + }, + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "A Message Port is the place messages (events) are sent", + "name": "roMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessageport.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Checks whether the enumeration contains no elements.", + "name": "IsEmpty", + "params": [], + "returnDescription": "A flag indicating whether the enumeration contains no elements (true), or contains elements (false).", + "returnType": "Boolean" + }, + { + "description": "Checks whether the current position is not past the end of the enumeration.", + "name": "IsNext", + "params": [], + "returnDescription": "A flag indicating whether the current position is not past the end (true), or is past the end (false).", + "returnType": "Boolean" + }, + { + "description": "Increments the position of an enumeration. If the last element of the enumeration is returned, this method sets the current position to indicate that it is now past the end.", + "name": "Next", + "params": [], + "returnDescription": "The value at the current position of the enumeration. If the current position is already past the end (that is, the last element has already been returned by a previous call to this method), \"invalid\" is returned.", + "returnType": "Dynamic" + }, + { + "description": "Resets the current position to the first element of the enumeration.", + "name": "Reset", + "params": [], + "returnType": "Void" + } + ], + "name": "ifEnum", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifenum.md" + }, + "ifevpcipher": { + "implementers": [ + { + "description": "The EVP Cipher component provides an interface to the OpenSSL EVP library of symmetric cipher commands", + "name": "roEVPCipher", + "url": "https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md" + } + ], + "methods": [ + { + "description": "Signals that all data has been submitted by previous calls to Update().", + "name": "Final", + "params": [], + "returnDescription": "The last remaining encrypted or decrypted bytes.", + "returnType": "Object" + }, + { + "description": "Processes the included [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing encrypted/decrypted data.", + "name": "Process", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing data that is encrypted or decrypted.", + "isRequired": true, + "name": "bytes", + "type": "Object" + } + ], + "returnDescription": "An [roByteArray](/docs/references/brightscript/components/robytearray.md \"roByteArray\") containing the result.", + "returnType": "Object" + }, + { + "description": "Reinitializes an existing cipher context. This can be called to reuse an existing [roEVPCipher](https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md\"roEVPCipher\") object to encrypt new data", + "name": "Reinit", + "params": [], + "returnDescription": "Returns 0 on success or non-zero on failure.", + "returnType": "Integer" + }, + { + "description": "Configures and initializes a new cipher context.", + "name": "Setup", + "params": [ + { + "default": null, + "description": "True for encryption; false for decryption", + "isRequired": true, + "name": "encrypt", + "type": "Boolean" + }, + { + "default": null, + "description": "Cipher format string, from openssl, listed at roEVPCipher", + "isRequired": true, + "name": "format", + "type": "String" + }, + { + "default": null, + "description": "A hex-encoded key", + "isRequired": true, + "name": "key", + "type": "String" + }, + { + "default": null, + "description": "A hex-encoded initialization vector, which can be an empty string", + "isRequired": true, + "name": "iv", + "type": "String" + }, + { + "default": null, + "description": "1 to use standard padding; 0 for no padding)", + "isRequired": true, + "name": "padding", + "type": "Integer" + } + ], + "returnDescription": "Returns 0 on success or non-zero on failure.", + "returnType": "Integer" + }, + { + "description": "Updates the included [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing encrypted/decrypted data.", + "name": "Update", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing data that is encrypted or decrypted.", + "isRequired": true, + "name": "bytes", + "type": "Object" + } + ], + "returnDescription": "An [roByteArray](/docs/references/brightscript/components/robytearray.md \"roByteArray\") containing a subset of the result. Some or all of the result may not be returned until the next call to Update().", + "returnType": "Object" + } + ], + "name": "ifEVPCipher", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifevpcipher.md" + }, + "ifevpdigest": { + "implementers": [ + { + "description": "The EVP Digest component provides an interface to the OpenSSL EVP library of message digest algorithms", + "name": "roEVPDigest", + "url": "https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md" + } + ], + "methods": [ + { + "description": "Returns the digest of data passed in by previous calls to [Update()](#updatebytes-as-object-as-void) as a hex string.", + "name": "Final", + "params": [], + "returnDescription": "Hex string (digest of data)", + "returnType": "String" + }, + { + "description": "Digests the provided data.", + "name": "Process", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing digested data", + "isRequired": true, + "name": "bytes", + "type": "Object" + } + ], + "returnDescription": "A Hex string (Digested array data).", + "returnType": "String" + }, + { + "description": "Re-initializes an existing message digest context. This can be called to reuse an existing [roEVPDigest](https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md\"roEVPDigest\") object to digest new data.", + "name": "Reinit", + "params": [], + "returnDescription": "Returns 0 on success or non-zero on failure.", + "returnType": "Integer" + }, + { + "description": "Initializes a new message digest context.", + "name": "Setup", + "params": [ + { + "default": null, + "description": "The supported digest algorithm from openssl, listed at roEVPDigest.", + "isRequired": true, + "name": "digestType", + "type": "String" + } + ], + "returnDescription": "Returns 0 on success or non-zero on failure.", + "returnType": "Integer" + }, + { + "description": "Adds more data to be digested.", + "name": "Update", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md containing data to be added to the current digest", + "isRequired": true, + "name": "bytes", + "type": "Object" + } + ], + "returnType": "Void" + } + ], + "name": "ifEVPDigest", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifevpdigest.md" + }, + "iffilesystem": { + "implementers": [ + { + "description": "The roFilesystem component implements common filesystem inspection and modification routines", + "name": "roFile System", + "url": "https://developer.roku.com/docs/developer-program/getting-started/architecture/file-system.md" + } + ], + "methods": [ + { + "description": "Copies the files from one directory to another.", + "name": "CopyFile", + "params": [ + { + "default": null, + "description": "The source path containing the files to be copied.", + "isRequired": true, + "name": "fromPath", + "type": "String" + }, + { + "default": null, + "description": "The target path to which files are to be copied.", + "isRequired": true, + "name": "toPath", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the files were successfully copied.", + "returnType": "Boolean" + }, + { + "description": "Creates the directory specified by the path parameter. All directories in path except the last one must already exist; that is, only one directory can be created.", + "name": "CreateDirectory", + "params": [ + { + "default": null, + "description": "The path of the directory to be created.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the path was successfully created.", + "returnType": "Boolean" + }, + { + "description": "Permanently removes the file or directory specified by the path parameter. If path is a directory, its contents are recursively removed.", + "name": "Delete", + "params": [ + { + "default": null, + "description": "The path of the directory to be deleted.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the path was successfully deleted.", + "returnType": "Boolean" + }, + { + "description": "Checks if the specified directory path exists on the device.", + "name": "Exists", + "params": [ + { + "default": null, + "description": "The directory path to be checked.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the specified path directory exists on the device.", + "returnType": "Boolean" + }, + { + "description": "Returns the file names in the specified directory path matching the provided regex.", + "name": "Find", + "params": [ + { + "default": null, + "description": "The directory path from which to get a list of file names.", + "isRequired": true, + "name": "dirPath", + "type": "String" + }, + { + "default": null, + "description": "The regex to be used to search for files.", + "isRequired": true, + "name": "regEx", + "type": "String" + } + ], + "returnDescription": "An [roList](/docs/references/brightscript/components/rolist.md \"roList\") of Strings representing the directory listing of names in dirPath that match the regex.", + "returnType": "Object" + }, + { + "description": "Returns the file names in the specified directory path and any sudirectories matching the provided regex.", + "name": "FindRecurse", + "params": [ + { + "default": null, + "description": "The directory path from which to get a list of file names.", + "isRequired": true, + "name": "dirPath", + "type": "String" + }, + { + "default": null, + "description": "The regex to be used to search for files.", + "isRequired": true, + "name": "regEx", + "type": "String" + } + ], + "returnDescription": "An [roList](/docs/references/brightscript/components/rolist.md \"roList\") of Strings representing the directory listing of names in dirPath that match the regex. Each item in the list is the name of the file relative to dirPath.", + "returnType": "Object" + }, + { + "description": "Returns the file names in the specified directory path.", + "name": "GetDirectoryListing", + "params": [ + { + "default": null, + "description": "The directory path from which to get a list of file names.", + "isRequired": true, + "name": "dirPath", + "type": "String" + } + ], + "returnDescription": "An [roList](/docs/references/brightscript/components/rolist.md \"roList\") of strings representing the directory listing of names in dirPath.", + "returnType": "Object" + }, + { + "description": "Returns information about the specified volume. The function can only be called on external volumes; internal volumes do not return meaningful information.", + "name": "GetVolumeInfo", + "params": [ + { + "default": null, + "description": "The external volume for which to get information. This should be specified as the volume name plus a directory separator (for example, \"ext1:/\").", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "An roAssociativeArray containing the following key-value pairs about the specified external volume:", + "returnType": "Object" + }, + { + "description": "Returns the available volumes on the device.", + "name": "GetVolumeList", + "params": [], + "returnDescription": "An [roList](/docs/references/brightscript/components/rolist.md \"roList\") containing strings representing the available volumes.", + "returnType": "Object" + }, + { + "description": "Returns the file names in the specified directory path matching the provided shell-like pattern. This method is similar to the [Find()](#finddirpath-as-string-regex-as-string-as-object) method except that it uses shell-like pattern matching rather than regular expression matching.", + "name": "Match", + "params": [ + { + "default": null, + "description": "The directory path from which to get a list of file names.", + "isRequired": true, + "name": "path", + "type": "String" + }, + { + "default": null, + "description": "The shell-like pattern to be used to search for files. The pattern may contain wildcards such as `*`and `?`.", + "isRequired": true, + "name": "pattern", + "type": "String" + } + ], + "returnDescription": "An [roList](/docs/references/brightscript/components/rolist.md \"roList\") of Strings representing the directory listing of names in dirPath that match the shell-like pattern.", + "returnType": "Object" + }, + { + "description": "Renames the directory.", + "name": "Rename", + "params": [ + { + "default": null, + "description": "The current name of the path to be renamed.", + "isRequired": true, + "name": "fromPath", + "type": "String" + }, + { + "default": null, + "description": "The new name of the path.", + "isRequired": true, + "name": "toPath", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the directory was successfully renamed. If the provided target directory (**toPath**) exists, it is not overwritten; instead the operation fails and this method returns false", + "returnType": "Boolean" + }, + { + "description": "Returns the keys in the specified directory path.", + "name": "Stat", + "params": [ + { + "default": null, + "description": "The directory path to be checked.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "An [roAssociativeArray](/docs/references/brightscript/components/roassociativearray.md \"roAssociativeArray\") containing the following key-value pairs for the specified path:", + "returnType": "Object" + } + ], + "name": "ifFileSystem", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffilesystem.md" + }, + "iffloat": { + "description": "Interface equivalent for intrinsic type Float", + "implementers": [ + { + "description": "Object equivalent for intrinsic type 'Float'", + "name": "roFloat", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofloat.md" + } + ], + "methods": [ + { + "description": "Gets the float value stored in the calling Float object.", + "name": "GetFloat", + "params": [], + "returnDescription": "The float value stored in the calling Float object.", + "returnType": "Float" + }, + { + "description": "Sets the calling Float object to the specified float value.", + "name": "SetFloat", + "params": [ + { + "default": null, + "description": "The float value to be set on the calling Float object.", + "isRequired": true, + "name": "value", + "type": "Float" + } + ], + "returnType": "Void" + } + ], + "name": "ifFloat", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffloat.md" + }, + "iffont": { + "implementers": [ + { + "description": "roFont represents a particular font, from a font-family (eg. Arial), with a particular pixel size (e.g 20), and a particular boldness or italicness", + "name": "roFont", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofont.md" + } + ], + "methods": [ + { + "description": "Returns the number of pixels of the font ascent.", + "name": "GetAscent", + "params": [], + "returnDescription": "The number of pixels.", + "returnType": "Integer" + }, + { + "description": "Returns the number of pixels of the font descent.", + "name": "GetDescent", + "params": [], + "returnDescription": "The number of pixels.", + "returnType": "Integer" + }, + { + "description": "Returns the font maximum advance width in pixels.", + "name": "GetMaxAdvance", + "params": [], + "returnDescription": "The number of pixels.", + "returnType": "Integer" + }, + { + "description": "Returns the number of pixels from one line to the next when drawing with this font.", + "name": "GetOneLineHeight", + "params": [], + "returnDescription": "The number of pixels.", + "returnType": "Integer" + }, + { + "description": "Returns the number of pixels from one line to the next when drawing with this font. Each glyph and the needed spacing between glyphs is measured.", + "name": "GetOneLineWidth", + "params": [ + { + "default": null, + "description": "The subject text.", + "isRequired": true, + "name": "text", + "type": "String" + }, + { + "default": null, + "description": "Generally, the amount of pixels available for rendering on this line.", + "isRequired": true, + "name": "MaxWidth", + "type": "Integer" + } + ], + "returnDescription": "The number of pixels. This will be less than provided MaxWidth.", + "returnType": "Integer" + } + ], + "name": "ifFont", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffont.md" + }, + "iffontmetrics": { + "deprecatedDescription": "This interface is deprecated. Developers should use [roFont](/docs/references/brightscript/components/rofont.md \"roFont\") methods (GetOneLineHeight and GetOneLineWidth).\n", + "description": "> This interface is deprecated. Developers should use [roFont](https://developer.roku.com/docs/references/brightscript/components/rofont.md\"roFont\") methods (GetOneLineHeight and GetOneLineWidth).", + "implementers": [ + { + "description": "The roFontMetrics object allows you to get display size information for a specific font returned by the roFontRegistry.Get() method", + "name": "roFontMetrics", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofontmetrics.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Returns the width and height of the **stringToDraw** parameter rendered in the font passed on the CreateObject() call.", + "name": "Size", + "params": [ + { + "default": null, + "description": "The string to be drawn.", + "isRequired": true, + "name": "stringToDraw", + "type": "String" + } + ], + "returnDescription": "An [roAssociativeArray](/docs/references/brightscript/components/roassociativearray.md \"roAssociativeArray\") with width and height parameters. The following example demonstrates this:", + "returnType": "Object" + } + ], + "name": "ifFontMetrics", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffontmetrics.md" + }, + "iffontregistry": { + "implementers": [ + { + "description": "The roFontRegistry object allows you to create roFont objects, either using the default font or using fonts in TrueType or OpenType files packaged with your application", + "name": "roFontRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofontregistry.md" + } + ], + "methods": [ + { + "description": "Returns a valid font string that can be used as the value of the Font content meta-data parameter recognized by the [roImageCanvas](https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md\"roImageCanvas\") method.", + "name": "Get", + "params": [ + { + "default": null, + "description": "The font family name.", + "isRequired": true, + "name": "family", + "type": "String" + }, + { + "default": null, + "description": "The requested font size, in pixels, not points.", + "isRequired": true, + "name": "size", + "type": "Integer" + }, + { + "default": null, + "description": "\"bold\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "bold", + "type": "Boolean" + }, + { + "default": null, + "description": "\"italic\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "italic", + "type": "Boolean" + } + ], + "returnDescription": "A valid font string.", + "returnType": "String" + }, + { + "description": "Returns the system font at its default size. Calling this method is the same as calling the [GetDefaultFont()](#getdefaultfontsize-as-integer-bold-as-boolean-italic-as-boolean-as-object) method with the following syntax: `reg.GetDefaultFont(reg.GetDefaultFontSize(), false, false)`.", + "name": "GetDefaultFont", + "params": [], + "returnDescription": "The system font as its default size.", + "returnType": "Object" + }, + { + "description": "Returns the system font. The system font is always available, even if the [Register()](#registerpath-as-string-as-boolean) method has not been called", + "name": "GetDefaultFont", + "params": [ + { + "default": null, + "description": "The requested font size, in pixels, not points.", + "isRequired": true, + "name": "size", + "type": "Integer" + }, + { + "default": null, + "description": "\"bold\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "bold", + "type": "Boolean" + }, + { + "default": null, + "description": "\"italic\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "italic", + "type": "Boolean" + } + ], + "returnDescription": "An roFont object representing the system font.", + "returnType": "Object" + }, + { + "description": "Returns the default font size.", + "name": "GetDefaultFontSize", + "params": [], + "returnDescription": "The default font size.", + "returnType": "Integer" + }, + { + "description": "Returns the names of the font families that have been registered via the [Register()](#registerpath-as-string-as-boolean) method. Each name can be passed as the first parameter to the [GetFont()](#getfontfamily-as-string-size-as-integer-bold-as-boolean-italic-as-boolean-as-object) method.", + "name": "GetFamilies", + "params": [], + "returnDescription": "An [roArray](/docs/references/brightscript/components/roarray.md \"roArray\") of strings that represent the names of the font families that have been registered.", + "returnType": "Object" + }, + { + "description": "Returns a font from the specified family, selected from the fonts previously registered via the [Register()](#registerpath-as-string-as-boolean) method.", + "name": "GetFont", + "params": [ + { + "default": null, + "description": "The font family name.", + "isRequired": true, + "name": "family", + "type": "String" + }, + { + "default": null, + "description": "The requested font size, in pixels, not points.", + "isRequired": true, + "name": "size", + "type": "Integer" + }, + { + "default": null, + "description": "\"bold\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "bold", + "type": "Boolean" + }, + { + "default": null, + "description": "\"italic\" specifies a font variant that may be (but is not always) supported by the font file.", + "isRequired": true, + "name": "italic", + "type": "Boolean" + } + ], + "returnDescription": "An [roFont](/docs/references/brightscript/components/rofont.md \"roFont\") object representing a font from the specified family.", + "returnType": "Object" + }, + { + "description": "Registers a font file (.ttf or .otf format). Each font file defines one or more font families (usually one).", + "name": "Register", + "params": [ + { + "default": null, + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the fonts in the specified file were successfully installed.", + "returnType": "Boolean" + } + ], + "name": "ifFontRegistry", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffontregistry.md" + }, + "iffunction": { + "description": "Interface equivalent for intrinsic type Function.", + "implementers": [ + { + "description": "Object equivalent for intrinsic type Function", + "name": "roFunction", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofunction.md" + } + ], + "methods": [ + { + "name": "GetSub", + "params": [], + "returnType": "Function" + }, + { + "name": "SetSub", + "params": [ + { + "default": null, + "isRequired": true, + "name": "value", + "type": "Function" + } + ], + "returnType": "Void" + } + ], + "name": "ifFunction", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iffunction.md" + }, + "ifgetmessageport": { + "implementers": [ + { + "description": "The HDMI status component provides an interface to the current HDMI operational status", + "name": "roHdmiStatus", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohdmistatus.md" + }, + { + "description": "The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from", + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + }, + { + "description": "The roTextToSpeech component provides text to speech capabilities to applications", + "name": "roTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md" + }, + { + "description": "A roUrlTransfer object transfers data to or from remote servers specified by URLs", + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + } + ], + "methods": [ + { + "description": "Returns the message port (if any) currently associated with the object", + "name": "GetMessagePort", + "params": [], + "returnDescription": "The message port.", + "returnType": "Object" + } + ], + "name": "ifGetMessagePort", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgetmessageport.md" + }, + "ifgridscreen": { + "deprecatedDescription": "This interface is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Grid Screen provides a graphical display of poster art from multiple content categories from within a single screen", + "name": "roGridScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rogridscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "ClearMessage", + "params": [], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Enables the breadcrumb text specified with [SetBreadcrumbText()](#setbreadcrumbtextlocation1-as-string-location2-as-string-as-void) method to be displayed.", + "name": "SetBreadcrumbEnabled", + "params": [ + { + "default": null, + "description": "Set this flag to true in order to display the breadcrumb text.", + "isRequired": true, + "name": "enabled", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "If SetBreadcrumbEnabled() is true, display Location1 and Location2 in the right of the overhang.", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The first location value.", + "isRequired": true, + "name": "Location1", + "type": "String" + }, + { + "default": null, + "description": "The second location value.", + "isRequired": true, + "name": "Location2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the content list for the specified row.", + "name": "SetContentList", + "params": [ + { + "default": null, + "description": "The row to be updated with the provided content list.", + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "description": "An roArray of items, where each item is an roAssociativeArray containing the [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\") used to display the item (for example, Title, Description, SDPosterUrl, and HDPosterUrl). See [Content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Metadata\") for details.", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Set or update items in the content list for the specified row. Enables performance improvements when dealing with your backend services by asking only for only a screenful worth of data at a time, and then update the rest of the content list content in the background.", + "name": "SetContentListSubset", + "params": [ + { + "default": null, + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "contentList", + "type": "Object" + }, + { + "default": null, + "isRequired": true, + "name": "offset", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "length", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies whether the item counter should be displayed. The item counter shows the focused item number and count of items in the row.", + "name": "SetCounterVisible", + "params": [ + { + "default": null, + "description": "Set this flag to true in order to display the item counter.", + "isRequired": true, + "name": "visible", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Description", + "name": "SetDescriptionVisible", + "params": [ + { + "default": null, + "description": "Set this flag to true in order to display the description box for the focused item.", + "isRequired": true, + "name": "visible", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the mode for displaying images in the grid screen. This allows images to be either scaled to completely fill the poster frame (scale-to-fill) or scaled to fit inside the poster frame (scale-to-fit) while maintaining aspect ratio.", + "name": "SetDisplayMode", + "params": [ + { + "default": null, + "description": "The mode to be used for displaying images, which maybe one of the following values: * \"scale-to-fill\": scales image to completely fill the rectangle of the bounding frame (Default) * \"scale-to-fit\": scale image to fit horizontally or vertically as appropriate while still maintaining aspect ratio. Note that scale-to-fit may result in pillar-box or letter-box display of images. * \"zoom-to-fill\": scales and crops image to maintain aspect ratio and completely fill the rectangle of the bounding frame. * \"photo-fit\": Uses several methods to fit the image with a different aspect ratio to the screen. First, it will asymmetrically scale up to a maximum of 5%. Second, for landscape images, if vertical cropping is necessary, it will remove two lines off the bottom for every one line off the top up to a maximum of 30% of the image. For all images, if horizontal cropping is necessary it will crop an equal amount from both sides.", + "isRequired": true, + "name": "displayMode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the default images to be shown in the grid when the grid screen content fails to load.", + "name": "SetErrorPoster", + "params": [ + { + "default": null, + "description": "The URL of the standard definition poster image to be shown.", + "isRequired": true, + "name": "sdPosterUrl", + "type": "String" + }, + { + "default": null, + "description": "The URL of the high definition poster image to be shown.", + "isRequired": true, + "name": "hdPosterUrl", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Set the focus to the specified row and item.", + "name": "SetFocusedListItem", + "params": [ + { + "default": null, + "description": "The row to be updated.", + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "description": "The items within the specified row to be updated.", + "isRequired": true, + "name": "itemIndex", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies whether the focus ring is displayed for the focused item.", + "name": "SetFocusRingVisible", + "params": [ + { + "default": null, + "description": "Set this flag to true in order to display the focus ring for the focused item.", + "isRequired": true, + "name": "visible", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the style or theme for displaying images in the grid screen. This allows different appearances of the overall grid for different sized images.", + "name": "SetGridStyle", + "params": [ + { + "default": null, + "description": "The image size, which may be one of the following values: * flat-movie – movie posters as seen in the Netflix channel (Default) * Image sizes: SD 110x150 HD 210x270 * SD 5 posters across, by 2 rows * HD 5 posters across, by 2 rows", + "isRequired": true, + "name": "style", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the list of row titles. The first name is displayed for the first row, the second name for the second row, and so on. The list should contain as many titles as the number of rows in the grid.", + "name": "SetListName", + "params": [ + { + "default": null, + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the list of row titles. The first name is displayed for the first row, the second name for the second row, etc. The list should contain as many titles as the number of rows in the grid", + "name": "SetListNames", + "params": [ + { + "default": null, + "description": "An roArray of strings containing the titles to be displayed in the grid.", + "isRequired": true, + "name": "names", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Changes the current position (both row and column) of the downloading thread in populating image posters of the grid.", + "name": "SetListOffset", + "params": [ + { + "default": null, + "description": "The row to be updated.", + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "description": "The items within the specified row to be updated.", + "isRequired": true, + "name": "itemIndex", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the images size and orientation for each row in the grid. This method is used in conjunction with the mixed-apect-ratio grid style.", + "name": "SetListPosterStyles", + "params": [ + { + "default": null, + "description": "The style to be used for the row, which may be one of the following values: landscape, portrait, or square", + "isRequired": true, + "name": "styles", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the visibility of the specified row.", + "name": "SetListVisible", + "params": [ + { + "default": null, + "description": "The row to be updated.", + "isRequired": true, + "name": "rowIndex", + "type": "Integer" + }, + { + "default": null, + "description": "A flag indicating whether the row is to be made visible.", + "isRequired": true, + "name": "visible", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the default images to be shown in the grid while the grid screen content is loading.", + "name": "SetLoadingPoster", + "params": [ + { + "default": null, + "description": "The URL of the standard definition poster image to be shown.", + "isRequired": true, + "name": "sdPosterUrl", + "type": "String" + }, + { + "default": null, + "description": "The URL of the high definition poster image to be shown.", + "isRequired": true, + "name": "hdPosterUrl", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Controls how the remote \"up\" key behaves when it is pressed with the top row selected.", + "name": "SetUpBehaviorAtTopRow", + "params": [ + { + "default": null, + "description": "Specifies the behavior of the \"up\" key. This may be one of the following values: * \"stop\": stop scrolling up, and stay on the roGridScreen (default behavior) * \"exit\": exit the roGridScreen", + "isRequired": true, + "name": "behavior", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the number of rows in the grid. The number of rows must be set before calling other functions to populate grid data.", + "name": "SetupLists", + "params": [ + { + "default": null, + "description": "The number of rows to be used in the grid.", + "isRequired": true, + "name": "count", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after the initial creation or the state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen is displayed (true).", + "returnType": "Boolean" + }, + { + "description": "Displays a semi-transparent popup message box to the user in the center of the screen over the poster screen. This is generally used for error messages.", + "name": "ShowMessage", + "params": [ + { + "default": null, + "description": "The text to be displayed in the popup message box.", + "isRequired": true, + "name": "message", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifGridScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifgridscreen.md" + }, + "ifhdmistatus": { + "implementers": [ + { + "description": "The HDMI status component provides an interface to the current HDMI operational status", + "name": "roHdmiStatus", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohdmistatus.md" + } + ], + "methods": [ + { + "description": "Returns the version number of the currently established HDCP link.", + "name": "GetHdcpVersion", + "params": [], + "returnDescription": "The version number of the HDCP link: 1.4 or 2.2.", + "returnType": "String" + }, + { + "description": "Checks whether the HDMI or MHL output is connected to an HDMI device.", + "name": "IsConnected", + "params": [], + "returnDescription": "A flag indicating whether the HDMI or MHL output is connected to an HDMI device.", + "returnType": "Boolean" + }, + { + "description": "Checks if the current established HDCP link is the specified version or higher", + "name": "IsHdcpActive", + "params": [ + { + "default": null, + "description": "The HDCP link version to be checked (for example, \"1.4\" or \"2.2\").", + "isRequired": true, + "name": "version", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the current established HDCP link is the specified `version`.", + "returnType": "Boolean" + } + ], + "name": "ifHdmiStatus", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhdmistatus.md" + }, + "ifhmac": { + "implementers": [ + { + "description": "The HMAC component provides an interface to the OpenSSL HMAC functions", + "name": "roHMAC", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohmac.md" + } + ], + "methods": [ + { + "description": "Returns an [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md\"roByteArray\") containing the final MAC.", + "name": "Final", + "params": [], + "returnDescription": "The final MAC.", + "returnType": "Object" + }, + { + "description": "Digests the data in an array generates a MAC. Calling this method is the same as making the following calls:", + "name": "Process", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md\"roByteArray\") with the data to be digested.", + "isRequired": true, + "name": "message", + "type": "Object" + } + ], + "returnDescription": "An [roByteArray](/docs/references/brightscript/components/robytearray.md \"roByteArray\") containing the generated MAC.", + "returnType": "Object" + }, + { + "description": "Re-initializes an existing HMAC context. This can be called to reuse an existing roHMAC object to authenticate new data.", + "name": "Reinit", + "params": [], + "returnDescription": "An integer indicating whether the function succeeded (0) or failed (1).", + "returnType": "Integer" + }, + { + "description": "Initializes new HMAC context.", + "name": "Setup", + "params": [ + { + "default": null, + "description": "Selects one of the supported digest algorithms, as documented in [roEVPDigest](https://developer.roku.com/docs/references/brightscript/components/roevpdigest.md\"roEVPDigest\").", + "isRequired": true, + "name": "digestType", + "type": "String" + }, + { + "default": null, + "description": "An roByteArray containing the key for the MAC.", + "isRequired": true, + "name": "key", + "type": "Object" + } + ], + "returnDescription": "An integer indicating whether the function succeeded (0) or failed (1).", + "returnType": "Integer" + }, + { + "description": "Adds more data to be digested. The data in the array is added to the current digest.", + "name": "Update", + "params": [ + { + "default": null, + "description": "An [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md\"roByteArray\") with the additional data to be digested.", + "isRequired": true, + "name": "partialMesssage", + "type": "Object" + } + ], + "returnType": "Void" + } + ], + "name": "ifHMAC", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhmac.md" + }, + "ifhttpagent": { + "description": "The ifHttpAgent methods modify the way that URLs are accessed", + "implementers": [ + { + "description": "The Application Manager APIs set application level attributes, which mostly affect the look-and-feel of the application", + "name": "roAppManager", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifappmanager.md" + }, + { + "description": "The Audio Player object provides the ability to setup the playing of a series of audio streams", + "name": "roAudioPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioplayer.md" + }, + { + "description": "The Grid Screen provides a graphical display of poster art from multiple content categories from within a single screen", + "name": "roGridScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rogridscreen.md" + }, + { + "description": "The roImageCanvas component provides an interface to render graphic elements at specific spots on the screen", + "name": "roImageCanvas", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md" + }, + { + "description": "The List Screen provides a graphical display of content in a vertical list within a single screen", + "name": "roListScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolistscreen.md" + }, + { + "description": "The Paragraph Screen provides a way to display text and selection choices to the user", + "name": "roParagraphScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roparagraphscreen.md" + }, + { + "description": "The Poster Screen provides a graphical display of poster art for content selection or can be used as a submenu to provide hierarchical structure to the application", + "name": "roPosterScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roposterscreen.md" + }, + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + }, + { + "description": "The Slide Show screen provides the ability to setup a photo slide show to playback a series of images", + "name": "roSlideShow", + "url": "https://developer.roku.com/docs/references/brightscript/components/roslideshow.md" + }, + { + "description": "The Springboard Screen shows detailed information about an individual piece of content and provides options for actions that may be taken on that content", + "name": "roSpringboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rospringboardscreen.md" + }, + { + "description": "roTextScreen provides a way of displaying large amounts of scrollable text", + "name": "roTextScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotextscreen.md" + }, + { + "description": "The Texture Manager provides a set of API's for managing an roBitmap cache.", + "name": "roTextureManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturemanager.md" + }, + { + "description": "An roTextureRequest is used to make requests to the roTextureManager", + "name": "roTextureRequest", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturerequest.md" + }, + { + "description": "A roUrlTransfer object transfers data to or from remote servers specified by URLs", + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + }, + { + "description": "The roVideoPlayer component implements a video player with more programmatic control, but less user control than the roVideoScreen component", + "name": "roVideoPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoplayer.md" + }, + { + "description": "The Video Screen object implements the video playback portion of the user interface", + "name": "roVideoScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoscreen.md" + } + ], + "methods": [ + { + "description": "Adds the specified cookies to the cookie cache.", + "name": "AddCookies", + "params": [ + { + "default": null, + "description": "An roArray of roAssociativeArrays, where each associative array represents a cookie to be added. Each associative array must contain the following key-value pairs:
NameTypeDescription
VersionIntegerCookie version number
DomainStringDomain to which cookie applies
PathStringPath to which cookie applies
NameStringName of the cookie
ValueStringValue of the cookie
ExpiresroDateTimeCookie expiration date, if any
", + "isRequired": true, + "name": "cookies", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the cookies were successfully added to the cache.", + "returnType": "Boolean" + }, + { + "name": "AddHeader", + "params": [ + { + "default": null, + "description": "The name of the HTTP header to be added to the list of headers. If \"x-roku-reserved-dev-id\" is passed as the name, the value parameter is ignored and in its place, the devid of the currently running channel is used as the value. This allows the developer's server to know which client app is talking to it. Any other headers with names beginning with \"x-roku-reserved-\" are reserved and may not be set.", + "isRequired": true, + "name": "name", + "type": "String" + }, + { + "default": null, + "description": "The value of the HTTP header being added.", + "isRequired": true, + "name": "value", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the HTTP header was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Removes all cookies from the cookie cache.", + "name": "ClearCookies", + "params": [], + "returnType": "Void" + }, + { + "description": "Enables any Set-Cookie headers returned from the request to be interpreted and the resulting cookies to be added to the cookie cache.", + "name": "EnableCookies", + "params": [], + "returnType": "Void" + }, + { + "name": "GetCookies", + "params": [ + { + "default": null, + "description": "The domain of the cookies to be retrieved. To match all domains, provide an empty string.", + "isRequired": true, + "name": "domain", + "type": "String" + }, + { + "default": null, + "description": "The path of the cookies to be retrieved.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "An roArray of roAssociativeArrays, where each associative array represents a cookie. The roAssociativeArrays contain the following key-value pairs:", + "returnType": "Object" + }, + { + "description": "Initializes the object to be sent to the Roku client certificate.", + "name": "InitClientCertificates", + "params": [], + "returnDescription": "A flag indicating whether the object sent to to the Roku client certificate was successfully initialized.", + "returnType": "Boolean" + }, + { + "description": "Sets the maximum depth of the certificate chain that will be accepted.", + "name": "SetCertificatesDepth", + "params": [ + { + "default": null, + "description": "The maximum depth to be used.", + "isRequired": true, + "name": "depth", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "name": "SetCertificatesFile", + "params": [ + { + "default": null, + "description": "The directory path of the .pem file to be used.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the certificate was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the HTTP headers to be sent in the HTTP request.", + "name": "SetHeaders", + "params": [ + { + "default": null, + "description": "An associative array containing the HTTP headers and values to be included in the HTTP request. If \"x-roku-reserved-dev-id\" is passed as a key, the value parameter is ignored and in its place, the devid of the currently running channel is used as the value. This allows the developer's server to know which client app is talking to it. Any other headers with names beginning with \"x-roku-reserved-\" are reserved and may not be set.", + "isRequired": true, + "name": "nameValueMap", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the HTTP header was successfully set.", + "returnType": "Boolean" + } + ], + "name": "ifHttpAgent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifhttpagent.md" + }, + "ifimagecanvas": { + "deprecatedDescription": "This interface is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The roImageCanvas component provides an interface to render graphic elements at specific spots on the screen", + "name": "roImageCanvas", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Adds a button to the specified screen.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be added to the screen.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button to be added to the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Turns updates on or off. Surrounding changes to several layers with AllowUpdates(false) and AllowUpdates(true) calls makes complex display modification atomic. This gives the application protection against the image canvas trying to render a partial update", + "name": "AllowUpdates", + "params": [ + { + "default": null, + "description": "Set this flag to true in order to enable updates.", + "isRequired": true, + "name": "updatesEnabled", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Clears all of the buttons from the screen and resets the array of buttons back to default with no buttons set.", + "name": "ClearButtons", + "params": [], + "returnType": "Void" + }, + { + "description": "Clears all content from a layer (see [SetLayer](#setlayerzorder-as-integer-contentmetadata-as-object-as-void) for the layer definition).", + "name": "ClearLayer", + "params": [ + { + "default": null, + "description": "The layer to be cleared.", + "isRequired": true, + "name": "zOrder", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This method is useful for avoiding screen flicker when the display order of your screens does not resemble a stack", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the width and height of the image canvas.", + "name": "GetCanvasRect", + "params": [], + "returnDescription": "An roAssociativeArray with names w for width and h for height.", + "returnType": "Object" + }, + { + "description": "Purges the internal cache of all images related to URLs in the current content list. If the content list is empty, then calling this method has no affect.", + "name": "PurgeCachedImages", + "params": [], + "returnType": "Void" + }, + { + "description": "Redraws the screen. Each call to this method replaces the previous content metadata that previously existed at that z-order layer.", + "name": "SetLayer", + "params": [ + { + "default": null, + "description": "A z-order specifier with higher z-orders closer to the viewer. Negative z-orders are \"behind the display\" and are thus invisible.", + "isRequired": true, + "name": "zOrder", + "type": "Integer" + }, + { + "default": null, + "description": "An [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") ([Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") objects) representing the information for each image to be displayed on the [roImageCanvas](https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md\"roImageCanvas\")", + "isRequired": true, + "name": "contentMetaData", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "This method is similar to the [SetLayer()](#setlayerzorder-as-integer-contentlist-as-object-as-void) method, except it takes an [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roAssociativeArrays](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArrays\") instead of just a single [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") of [Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\").", + "name": "SetLayer", + "params": [ + { + "default": null, + "description": "A z-order specifier with higher z-orders closer to the viewer. Negative z-orders are \"behind the display\" and are thus invisible.", + "isRequired": true, + "name": "zOrder", + "type": "Integer" + }, + { + "default": null, + "description": "An [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roAssociativeArrays](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArrays\") representing the information for each image to be displayed on the [roImageCanvas](https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md\"roImageCanvas\").", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Waits to draw the screen until all images in the content-meta-data array are downloaded, decoded, and loaded into memory. There is a large performance penalty for setting this to true", + "name": "SetRequireAllImagesToDraw", + "params": [ + { + "default": null, + "description": "A flag enabling the **requireAllImages** feature. It is recommended that this flag be disabled (false) for better performance.", + "isRequired": true, + "name": "requireAllImages", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after initial creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen is displayed.", + "returnType": "Boolean" + }, + { + "description": "Swaps the content metadata stored in one zOrder with another. This enables changing the layer that is displayed \"On Top\". This method can be used to swap empty (unspecified) layers", + "name": "SwapLayers", + "params": [ + { + "default": null, + "description": "The current zOrder that is displayed on top.", + "isRequired": true, + "name": "zOrderA", + "type": "Integer" + }, + { + "default": null, + "description": "The new zOrder to be displayed on top.", + "isRequired": true, + "name": "zOrderB", + "type": "Integer" + } + ], + "returnType": "Void" + } + ], + "name": "ifImageCanvas", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifimagecanvas.md" + }, + "ifimagemetadata": { + "implementers": [ + { + "description": "The roImageMetadata component provides developers access to image file metadata included in many .jpg EXIF headers", + "name": "roImageMetadata", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagemetadata.md" + } + ], + "methods": [ + { + "description": "Returns a set of simple and common image metadata", + "name": "GetMetadata", + "params": [], + "returnDescription": "An associative array containing the following key-value pairs with image metadata:", + "returnType": "Object" + }, + { + "description": "Returns all of the raw EXIF metadata.", + "name": "GetRawExif", + "params": [], + "returnDescription": "An associative array with all of the raw EXIF metadata. See the [EXIF section](/docs/references/brightscript/components/roimagemetadata.md#exif-background) for details about EXIF metadata.", + "returnType": "Object" + }, + { + "description": "Returns the raw data for an Exif tag. The method provides direct access to a specific raw EXIF tag", + "name": "GetRawExifTag", + "params": [ + { + "default": null, + "description": "The ifd of the Exif tag.", + "isRequired": true, + "name": "ifd", + "type": "Integer" + }, + { + "default": null, + "description": "The tag number of the Exif tag.", + "isRequired": true, + "name": "tagnum", + "type": "Integer" + } + ], + "returnDescription": "The raw data of an Exif tag. It the Exif tag doesn't exist it returns invalid.", + "returnType": "Dynamic" + }, + { + "description": "Returns a thumbnail image if one is embedded in the image metadata and the corresponding associative array with image data. This only generates a thumbnail if one exists.", + "name": "GetThumbnail", + "params": [], + "returnDescription": "An associative array that with **bytes** and **type** keys with the image data:", + "returnType": "Object" + }, + { + "description": "Sets the URL to the image. Only file URLs are supported", + "name": "SetUrl", + "params": [ + { + "default": null, + "description": "The URL of the image.", + "isRequired": true, + "name": "url", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifImageMetaData", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifimagemetadata.md" + }, + "ifinput": { + "implementers": [ + { + "description": "An roInput object can be used to receive events sent from a network client using the External Control Protocol (ECP), as described in External Control API", + "name": "roInput", + "url": "https://developer.roku.com/docs/references/brightscript/components/roinput.md" + } + ], + "methods": [ + { + "description": "Registers a channel to receive `roInput transport` events, which are voice commands sent via the Roku remote control, Roku mobile app, or a virtual assistant such as Amazon Alexa or Google Assistant.", + "name": "EnableTransportEvents", + "params": [], + "returnDescription": "A flag indicating whether transport event notifications were successfully registered.", + "returnType": "Boolean" + }, + { + "description": "Marks a transport command as handled, unhandled, or handled with an error.", + "name": "EventResponse", + "params": [ + { + "default": null, + "isRequired": true, + "name": "roAssociativeArray" + } + ], + "returnDescription": "A flag indicating whether the event response operation was successful." + }, + { + "description": "Returns the message port (if any) currently associated with the object.", + "name": "GetMessagePort", + "params": [], + "returnDescription": "The message port value.", + "returnType": "Object" + }, + { + "description": "Sets the roMessagePort to be used to receive events.", + "name": "SetMessagePort", + "params": [ + { + "default": null, + "description": "The port to be used to receive events.", + "isRequired": true, + "name": "port", + "type": "Object" + } + ], + "returnType": "Void" + } + ], + "name": "ifInput", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifinput.md" + }, + "ifint": { + "description": "> Interface equivalent for intrinsic type 'Integer'", + "implementers": [ + { + "description": "Object equivalent for intrinsic type Integer", + "name": "roInt", + "url": "https://developer.roku.com/docs/references/brightscript/components/roint.md" + } + ], + "methods": [ + { + "description": "Gets the integer value stored in the calling Integer object.", + "name": "GetInt", + "params": [], + "returnDescription": "The integer value stored in the calling Integer object.", + "returnType": "Integer" + }, + { + "description": "Sets the calling Integer object to the specified integer value.", + "name": "SetInt", + "params": [ + { + "default": null, + "description": "The integer value to be set on the calling Integer object.", + "isRequired": true, + "name": "value", + "type": "Integer" + } + ], + "returnType": "Void" + } + ], + "name": "ifInt", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifint.md" + }, + "ifintops": { + "implementers": [ + { + "description": "Object equivalent for intrinsic type Integer", + "name": "roInt", + "url": "https://developer.roku.com/docs/references/brightscript/components/roint.md" + } + ], + "methods": [ + { + "description": "Returns the integer value formatted as a decimal string. No leading space is appended for non-negative numbers.", + "name": "ToStr", + "params": [], + "returnDescription": "A decimal string.", + "returnType": "String" + } + ], + "name": "ifIntOps", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifintops.md" + }, + "ifkeyboardscreen": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Keyboard Screen is designed to allow the user to enter an alpha-numeric string for searching, username/password registration or other purposes", + "name": "roKeyboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rokeyboardscreen.md" + } + ], + "methods": [ + { + "description": "Adds a button to the screen identified by the title and ID provided. The buttons are displayed at the bottom of the screen and appear in the order added. When the button is pressed, the script will receive an event from the application indicating the ID of the button pressed.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be added to the keyboard screen.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button to be added to the keyboard screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Removes all buttons from the keyboard screen.", + "name": "ClearButtons", + "params": [], + "returnType": "Void" + }, + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the current value in the keyboard text entry field.", + "name": "GetText", + "params": [], + "returnDescription": "Current value string.", + "returnType": "String" + }, + { + "description": "Sets the descriptive text displayed to the user for prompting regarding the required entry.", + "name": "SetDisplayText", + "params": [ + { + "default": null, + "description": "The descriptive text to be displayed on the keyboard screen.", + "isRequired": true, + "name": "displayText", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the maximum length for text entry. The default is 20 characters.", + "name": "SetMaxLength", + "params": [ + { + "default": null, + "description": "The maximum number of characters for text entered into the keyboard screen.", + "isRequired": true, + "name": "maxLen", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Displays user-entered text with bullet characters.", + "name": "SetSecureText", + "params": [ + { + "default": null, + "description": "A flag indicating whether to obfuscate the text entered into the keyboard screen. If **isSecure** is set to true, the user-entered text is obscured with bullet characters. If false, it is treated as plain text. The default value is false.", + "isRequired": true, + "name": "isSecure", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "name": "SetText", + "params": [ + { + "default": null, + "description": "The default text to be displayed in the keyboard entry field.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "The default text that was set for the keyboard entry field.", + "returnType": "String" + }, + { + "description": "Sets the title for the screen to the specified string.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The title to be displayed on the keyboard screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was displayed.", + "returnType": "Boolean" + } + ], + "name": "ifKeyboardScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifkeyboardscreen.md" + }, + "iflist": { + "implementers": [ + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Adds an element to the head of the list.", + "name": "AddHead", + "params": [ + { + "default": null, + "isRequired": true, + "name": "tval", + "type": "Dynamic" + } + ], + "returnType": "Void" + }, + { + "description": "Adds an element to the tail of the list.", + "name": "AddTail", + "params": [ + { + "default": null, + "isRequired": true, + "name": "tval", + "type": "Dynamic" + } + ], + "returnType": "Void" + }, + { + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the number of elements in the list.", + "name": "Count", + "params": [], + "returnDescription": "The number of elements in the list.", + "returnType": "Integer" + }, + { + "description": "Retrieves the entry at the head of the list.", + "name": "GetHead", + "params": [], + "returnDescription": "The entry retrieved from the head of the list.", + "returnType": "Dynamic" + }, + { + "description": "Gets the entry at current index or position from the list and increments the index or position in the list.", + "name": "GetIndex", + "params": [], + "returnDescription": "The entry retrieved from the list. This method returns invalid if the end of the list is reached.", + "returnType": "Dynamic" + }, + { + "description": "Retrieves the entry at the tail of the list.", + "name": "GetTail", + "params": [], + "returnDescription": "The entry retrieved from the tail of the list.", + "returnType": "Dynamic" + }, + { + "description": "Removes the entry at the head of the list.", + "name": "RemoveHead", + "params": [], + "returnDescription": "The entry removed from the head of the list.", + "returnType": "Dynamic" + }, + { + "description": "Removes the entry at the current index or position from the list and increments the index or position in the list.", + "name": "RemoveIndex", + "params": [], + "returnDescription": "The entry removed from the list. This method returns invalid if the end of the list is reached.", + "returnType": "Dynamic" + }, + { + "description": "Removes the entry at the tail of the list.", + "name": "RemoveTail", + "params": [], + "returnDescription": "The entry removed from the tail of the list.", + "returnType": "Dynamic" + }, + { + "description": "Resets the current index or position in list to the head element.", + "name": "ResetIndex", + "params": [], + "returnDescription": "A flag indicating whether the index has been reset.", + "returnType": "Boolean" + } + ], + "name": "ifList", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflist.md" + }, + "iflistscreen": { + "deprecatedDescription": "This interface is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The List Screen provides a graphical display of content in a vertical list within a single screen", + "name": "roListScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolistscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Adds an item to the end of the list to be displayed on screen.", + "name": "AddContent", + "params": [ + { + "default": null, + "isRequired": true, + "name": "item", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Clears all the content in the list and displays an empty screen.", + "name": "ClearContent", + "params": [], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Removes an item from the list. The screen refreshes immediately if the item is visible", + "name": "RemoveContent", + "params": [ + { + "default": null, + "description": "The index of the item to be removed from the list. If the index is not within the range of 0 – list size, this function has no effect.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "isRequired": true, + "name": "breadcrumb1", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "breadcrumb2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the list of content to be displayed on the screen. See “Content Meta-Data” for details on the attributes for each element. The screen is responsible for fetching the poster art from the URL’s specified.", + "name": "SetContent", + "params": [ + { + "default": null, + "description": "An roArray of roAssociativeArrays (Content Meta-Data objects) representing the information for each item to be displayed on screen.", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the focused list item to the given index. If the item is not visible, focus jumps to the item and item becomes visible.", + "name": "SetFocusedListItem", + "params": [ + { + "default": null, + "description": "The item in the list to be given focus. If the index is not within the range of 0 – list size, this function has no effect.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the header text of the screen. This can be used as an information text for the screen.", + "name": "SetHeader", + "params": [ + { + "default": null, + "description": "A one-line text (string) displayed on top of the screen below the overhang.", + "isRequired": true, + "name": "header", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Updates the content of an item in the list.", + "name": "SetItem", + "params": [ + { + "default": null, + "description": "An roAssociativeArray (Content Meta-Data object) representing the information of the item to be updated.", + "isRequired": true, + "name": "item", + "type": "Object" + }, + { + "default": null, + "description": "The index of the item to be updated. If the index is not within the range of 0 – list size, this function has no effect.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title of the screen. The title is displayed on the top right corner of the screen on the overhang.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The title to be displayed on the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Controls how the remote \"up\" key behaves when it is pressed with the top row selected.", + "name": "SetupBehaviorAtTopRow", + "params": [ + { + "default": null, + "description": "Specifies the behavior of the \"up\" key. This may be one of the following values: * \"stop\": stop scrolling up, and stay on the list screen (default behavior) * \"exit\": exit the list screen", + "isRequired": true, + "name": "behavior", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after initial creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was displayed.", + "returnType": "Boolean" + } + ], + "name": "ifListScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflistscreen.md" + }, + "iflisttoarray": { + "implementers": [ + { + "description": "The list object implements the interfaces: ifList, ifArray, ifEnum and therefore can behave like an array that can dynamically add members", + "name": "roList", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolist.md" + }, + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "Returns an roArray containing the same elements as the list.", + "name": "ToArray", + "params": [], + "returnDescription": "An element list as an array.", + "returnType": "Object" + } + ], + "name": "ifListToArray", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflisttoarray.md" + }, + "iflocalization": { + "implementers": [ + { + "description": "The roLocalization object provides functions to assist in localization", + "name": "roLocalization", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolocalization.md" + } + ], + "methods": [ + { + "name": "GetLocalizedAsset", + "params": [ + { + "default": null, + "description": "The name of a subdirectory in the directory pkg:/locale/XX\\_YY/ where XX\\_YY is the current language setting.", + "isRequired": true, + "name": "dirName", + "type": "String" + }, + { + "default": null, + "description": "The name of the file.", + "isRequired": true, + "name": "fileName", + "type": "String" + } + ], + "returnDescription": "An asset path.", + "returnType": "String" + }, + { + "description": "Replaces \"^n\" in pluralString with count and returns the result.", + "name": "GetPluralString", + "params": [ + { + "default": null, + "isRequired": true, + "name": "count", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "zeroString", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "oneString", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "pluralString", + "type": "String" + } + ], + "returnDescription": "The result of the operation. If count is 0, this returns zeroString. If count is 1, it returns oneString.", + "returnType": "String" + } + ], + "name": "ifLocalization", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflocalization.md" + }, + "iflongint": { + "implementers": [ + { + "description": "Object equivalent for intrinsic type LongInteger", + "name": "roLongInteger", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolonginteger.md" + } + ], + "methods": [ + { + "description": "Gets the longinteger value stored in the calling Longinteger object.", + "name": "GetLongInt", + "params": [], + "returnDescription": "The longinteger value stored in the calling Loninteger object.", + "returnType": "LongInteger" + }, + { + "description": "Sets the calling Longinteger object to the specified longinteger value.", + "name": "SetLongInt", + "params": [ + { + "default": null, + "description": "The longinteger value to be set on the calling Longinteger object.", + "isRequired": true, + "name": "value", + "type": "Longinteger" + } + ], + "returnType": "Void" + } + ], + "name": "ifLongInt", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iflongint.md" + }, + "ifmessagedialog": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Message Dialog component is used to display a formatted, multi-line text message to the user", + "name": "roMessageDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessagedialog.md" + } + ], + "methods": [ + { + "description": "Adds a button to the right-justified side of the dialog box. The buttons are at the bottom of the dialog and appear in the order added. When the button is pressed, the script will receive an event from the application indicating the ID of the button pressed.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be added to the screen.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button to be added to the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added to the screen.", + "returnType": "Boolean" + }, + { + "description": "Adds horizontal line separating buttons into sections on the dialog.", + "name": "AddButtonSeparator", + "params": [], + "returnDescription": "A flag indicating whether the horizontal line was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds a button to the left-justified side of the dialog box. The buttons are at the bottom of the dialog and appear in the order added. When the button is pressed, the script will receive an event from the application indicating the ID of the button pressed.", + "name": "AddLeftButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be added to the screen.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button to be added to the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added to the screen.", + "returnType": "Boolean" + }, + { + "description": "Adds a star rating button to the left-justified side of the dialog. The rating is specified as an integer 1-100 which indicates the number of stars (1 to 5) to be displayed, or 0 if unrated. Think of this as a percentage value (for example, <20% = 1 star).", + "name": "AddLeftRatingButton", + "params": [ + { + "default": null, + "description": "The ID of the rating button to be added to the dialog.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The value the user rated the title, while the aggregate Rating represents the total for all users. The userRating takes precedence and determines the color of the buttons if set.", + "isRequired": true, + "name": "userRating", + "type": "Integer" + }, + { + "default": null, + "description": "The overall rating of the title. The aggregateRating may display half-stars", + "isRequired": true, + "name": "aggregateRating", + "type": "Integer" + }, + { + "default": null, + "description": "The tip associated with the rating button.", + "isRequired": true, + "name": "tip", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the star rating button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds a star rating button to the right-justified side of the dialog. The rating is specified as an integer 1-100 which indicates the number of stars (1 to 5) to be displayed, or 0 if unrated. Think of this as a percentage value (for example, <20% = 1 star).", + "name": "AddRatingButton", + "params": [ + { + "default": null, + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "userRating", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "aggregateRating", + "type": "Integer" + }, + { + "default": null, + "isRequired": true, + "name": "tip", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the star rating button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Appends a static paragraph with the specified text. This static text does not get changed by subsequent calls to the [SetText()](#settexttext-as-string-as-void) or [UpdateText()](#updatetexttext-as-string-as-void) methods.", + "name": "AddStaticText", + "params": [ + { + "default": null, + "description": "The text to be included in the static paragraph.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This method is useful for avoiding screen flicker when the display order of your screens does not resemble a stack", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Sends an [isScreenClosed()](https://developer.roku.com/docs/references/brightscript/events/romessagedialogevent.mdisscreenclosed-as-boolean) event when the remote control's back button is pressed.", + "name": "EnableBackButton", + "params": [ + { + "default": null, + "isRequired": true, + "name": "enableBackButton", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "With overlay enabled, the background screen is no longer dimmed.", + "name": "EnableOverlay", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable overlays.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the button to be highlighted. The default is the first button from the top.", + "name": "SetFocusedMenuItem", + "params": [ + { + "default": null, + "description": "The ID of the button to gain focus.", + "isRequired": true, + "name": "item", + "type": "Integer" + } + ], + "returnType": "Boolean" + }, + { + "description": "Sets the format of the buttons to top-left justified. Otherwise the format defaults to bottom-right justified.", + "name": "SetMenuTopLeft", + "params": [ + { + "default": null, + "description": "A flag specifying the format of the buttons. * true: Makes the buttons top-left justified. * false: Makes the buttons bottom-right justified.", + "isRequired": true, + "name": "topLeft", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Appends a paragraph with the specified text. The dialog will automatically resize to accommodate the text, up to the limit of available display space.", + "name": "SetText", + "params": [ + { + "default": null, + "description": "The text to be displayed in the message dialog.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title for the dialog to the specified string.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The title to be displayed in the message dialog.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was displayed.", + "returnType": "Boolean" + }, + { + "description": "Displays a spinning busy animation to indicate work in progress. This animation continues until the screen is closed.", + "name": "ShowBusyAnimation", + "params": [], + "returnType": "Void" + }, + { + "description": "Updates the title of the specified button.", + "name": "UpdateButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be updated.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button to be updated.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully updated.", + "returnType": "Boolean" + }, + { + "description": "Replaces the last paragraph with the specified text, or appends a paragraph if there is not a prior paragraph. If the last paragraph added was a static paragraph, this function appends a new paragraph.", + "name": "UpdateText", + "params": [ + { + "default": null, + "description": "The text to replace the last paragraph in the message dialog.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifMessageDialog", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmessagedialog.md" + }, + "ifmessageport": { + "implementers": [ + { + "description": "A Message Port is the place messages (events) are sent", + "name": "roMessagePort", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessageport.md" + } + ], + "methods": [ + { + "description": "If an event object is available, it is returned. Otherwise invalid is returned. The method returns immediately in either case and does not wait.", + "name": "GetMessage", + "params": [], + "returnDescription": "An event object.", + "returnType": "Dynamic" + }, + { + "description": "This method is similar to the [GetMessage()](#getmessage-as-dynamic) method, but the returned object (if not invalid) remains in the message queue. A later call to [WaitMessage()](#waitmessagetimeout-as-integer-as-dynamic), [GetMessage()](#getmessage-as-dynamic) or PeekMessage() will return the same message.", + "name": "PeekMessage", + "params": [], + "returnDescription": "An event object.", + "returnType": "Dynamic" + }, + { + "description": "Waits until an event object is available or timeout milliseconds have passed.", + "name": "WaitMessage", + "params": [ + { + "default": null, + "description": "The number of milliseconds to wait for a message. If this parameter is set to 0, this method waits indefinitely for a message, with no timeout. The native [wait()](https://developer.roku.com/docs/references/brightscript/language/global-utility-functions.mdwaittimeout-as-integer-port-as-object-as-object) function can also be used to get the event object which WaitMessage() would return. This means that the following two statements have the same effect: ``` msg = port.WaitMessage(timeout) msg = wait(timeout, port) ```", + "isRequired": true, + "name": "timeout", + "type": "Integer" + } + ], + "returnDescription": "If an event is available, it is returned. If the timeout expires, invalid is returned.", + "returnType": "Dynamic" + } + ], + "name": "ifMessagePort", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmessageport.md" + }, + "ifmicrophone": { + "implementers": [ + { + "description": "The roMicrophone API allows channel applications to receive audio data from the user’s microphone-supported remote control device or mobile phone", + "name": "roMicrophone", + "url": "https://developer.roku.com/docs/references/brightscript/components/romicrophone.md" + } + ], + "methods": [ + { + "description": "Indicates whether the platform and paired remote control can be requested to open the microphone.", + "name": "CanRecord", + "params": [], + "returnDescription": "A flag indicating whether the microphone can be opened.", + "returnType": "Boolean" + }, + { + "description": "Opens the microphone and records to create a WAV file at the specified output file path. Only tmp:/ paths are supported.", + "name": "RecordToFile", + "params": [ + { + "default": null, + "description": "The file path where the WAV file is to be stored.", + "isRequired": true, + "name": "wavFilePath", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the recording was performed and saved successfully.", + "returnType": "Boolean" + }, + { + "description": "Sets the text to be displayed in the system microphone UI.", + "name": "SetPrompt", + "params": [ + { + "default": null, + "description": "The text to be displayed in the system microphone UI.", + "isRequired": true, + "name": "prompt", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Opens the microphone and begins streaming microphone events to the channel. The channel must have called the [SetMessagePort()](https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.mdsetmessageportport-as-object--as-void) method previously.", + "name": "StartRecording", + "params": [], + "returnDescription": "A flag indicating whether the microphone was opened successfully.", + "returnType": "Boolean" + }, + { + "description": "Stops recording and closes the microphone. This method is useful if the microphone was previously opened via the [StartRecording()](#startrecording-as-boolean) method and the channel needs to cancel the current recording prematurely, (for example, the duration limit was reached or an application error).", + "name": "StopRecording", + "params": [], + "returnDescription": "A flag indicating whether the microphone was opened and closed successfully.", + "returnType": "Boolean" + } + ], + "name": "ifMicrophone", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifmicrophone.md" + }, + "ifonelinedialog": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The One Line Dialog is a special type of dialog optimized for single line text", + "name": "roOneLineDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/roonelinedialog.md" + } + ], + "methods": [ + { + "description": "Closes the screen and deletes the associated object. Useful for avoiding screen flicker when the display order of your screens does not resemble a stack.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Sets the title to be displayed in the upper right-hand corner of the screen in the overhang area.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The text to be displayed on the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was successfully displayed.", + "returnType": "Boolean" + }, + { + "description": "Displays a spinning busy animation to indicate work in progress is displayed. The animation will continue until the screen is closed.", + "name": "ShowBusyAnimation", + "params": [], + "returnType": "Void" + } + ], + "name": "ifOneLineDialog", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifonelinedialog.md" + }, + "ifparagraphscreen": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Paragraph Screen provides a way to display text and selection choices to the user", + "name": "roParagraphScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roparagraphscreen.md" + } + ], + "methods": [ + { + "description": "Adds a button to the screen. The buttons are displayed in a standard location on the screen and appear in the order added. When the button is pressed, the script will receive an event from the application containing the ID of the button pressed and allowing the script to perform the desired action for that case", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds a graphic image to the screen at the current cursor position and centers it. The current cursor position moves as headers, paragraphs, graphics and buttons are added to the screen. The graphic image is displayed unscaled.", + "name": "AddGraphic", + "params": [ + { + "default": null, + "isRequired": true, + "name": "url", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "displayMode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Adds a string of bold, high visibility text to the screen as a header to introduce the subsequent paragraph(s).", + "name": "AddHeaderText", + "params": [ + { + "default": null, + "description": "The text to be added to the header.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Adds a paragraph of text to the screen. Paragraphs are specified as a single string and they are ordered on the screen in the same order as they are added. Making multiple calls to AddParagraph() will continue to add additional paragraphs of text in order until the screen has been filled. The [roParagraphScreen](https://developer.roku.com/docs/references/brightscript/components/roparagraphscreen.md\"roParagraphScreen\") component handles all text formatting and justification. Spacing is automatically inserted between paragraphs for readability.", + "name": "AddParagraph", + "params": [ + { + "default": null, + "description": "The paragraph to be added to the screen.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "name": "AddParagraphAligned", + "params": [ + { + "default": null, + "isRequired": true, + "name": "text", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "alignment", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Specifies the text to be displayed for the title in breadcrumb format.", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The text to be used for the first location.", + "isRequired": true, + "name": "location1", + "type": "String" + }, + { + "default": null, + "description": "The text to be used for the second location.", + "isRequired": true, + "name": "location2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the button to be highlighted when the screen is initially displayed.", + "name": "SetDefaultMenuItem", + "params": [ + { + "default": null, + "description": "The index of the button to be highlighted by default. This is the zero-based position of the button in the list of buttons on the screen (not the button id passed to the [AddButton()](#addbuttonid-as-integer-title-as-string--as-boolean) method). The default is the first button (index 0).", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the default menu item was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the title for the screen.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The title for the screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A boolean indicating whether the screen was successfully displayed.", + "returnType": "Boolean" + } + ], + "name": "ifParagraphScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifparagraphscreen.md" + }, + "ifpath": { + "description": "| Name | Description |\n| --- | --- |\n| [roPath](https://developer.roku.com/docs/references/brightscript/components/ropath.md\"roPath\") | The roPath component provides developers an easy way to create valid file system paths |", + "implementers": [ + { + "description": "The roPath component provides developers an easy way to create valid file system paths", + "name": "roPath", + "url": "https://developer.roku.com/docs/references/brightscript/components/ropath.md" + } + ], + "methods": [ + { + "description": "Modifies or changes the current path via the specified relative or absolute path.", + "name": "Change", + "params": [ + { + "default": null, + "description": "The new relative or absolute file system path to be used.", + "isRequired": true, + "name": "path", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the path was successfully changed.", + "returnType": "Boolean" + }, + { + "description": "Checks whether the current path is valid (the path is correctly formed). This does not check whether the file actually exists.", + "name": "IsValid", + "params": [], + "returnDescription": "A flag indicating whether the current path is valid.", + "returnType": "Boolean" + }, + { + "description": "Return Value", + "name": "Split", + "params": [], + "returnDescription": "An [roAssociativeArray](/docs/references/brightscript/components/roassociativearray.md \"roAssociativeArray\") that contains the following keys:", + "returnType": "Object" + } + ], + "name": "ifPath", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifpath.md" + }, + "ifpinentrydialog": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Pin Entry Dialog is designed to allow the user to enter a numeric PIN for purchasing content", + "name": "roPinEntryDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/ropinentrydialog.md" + } + ], + "methods": [ + { + "description": "Adds a button to the pin entry dialog, with the specified title displayed on the button. The buttons are aligned at the bottom of the screen and appear in the order added. When a button is pressed, the script will receive an [roPinEntryDialogEvent](https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.md\"roPinEntryDialogEvent\") from the application indicating the ID of the button pressed.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID of the button to be added to the pin entry dialog.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The text to be displayed on the button.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Enables the channel to receive an [isScreenClosed()](https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.mdisscreenclosed-as-boolean) event when the back button is entered. By default the [PinEntryDialog](https://developer.roku.com/docs/references/brightscript/components/ropinentrydialog.md component will not send an [isScreenClosed()](https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.mdisscreenclosed-as-boolean) event so that scripts that did not expect this event will not break", + "name": "EnableBackButton", + "params": [ + { + "default": null, + "description": "Enables [isScreenClosed()](https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.mdisscreenclosed-as-boolean) events to be sent when the back button is entered.", + "isRequired": true, + "name": "enableBackButton", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Gets the user-entered PIN.", + "name": "Pin", + "params": [], + "returnDescription": "A string containing the PIN entered by the user.", + "returnType": "String" + }, + { + "description": "Sets the maximum number of digits to be entered and displayed for the PIN.", + "name": "SetNumPinEntryFields", + "params": [ + { + "default": null, + "description": "The maximum number of digits for the PIN.", + "isRequired": true, + "name": "numFields", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title for the pin entry dialog to the specified string.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The text to be displayed on the pin entry dialog.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnType": "Void" + } + ], + "name": "ifPinEntryDialog", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifpinentrydialog.md" + }, + "ifposterscreen": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\n| Name | Description |\n| --- | --- |\n| [roPosterScreen](https://developer.roku.com/docs/references/brightscript/components/roposterscreen.md\"roPosterScreen\") | The Poster Screen provides a graphical display of poster art for content selection or can be used as a submenu to provide hierarchical structure to the application |", + "implementers": [ + { + "description": "The Poster Screen provides a graphical display of poster art for content selection or can be used as a submenu to provide hierarchical structure to the application", + "name": "roPosterScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roposterscreen.md" + } + ], + "methods": [ + { + "description": "Clears the message from the previous [ShowMessage()](https://developer.roku.com/docs/references/brightscript/interfaces/ifposterscreen.mdshowmessagemessage-as-string-as-void) method call.", + "name": "ClearMessage", + "params": [], + "returnType": "Void" + }, + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Gets the content metadata of each title displayed on the screen. This is the content metadata passed via the [SetContentList()](#setcontentlistcontentlist-as-object-as-void) method.", + "name": "GetContentList", + "params": [], + "returnDescription": "An [roArray](/docs/references/brightscript/components/roarray.md \"roArray\") containing content metadata.", + "returnType": "Object" + }, + { + "description": "Sets the scale mode for displaying ad images on the poster screen. The ad display is only available on \"arced-landscape\" and \"flat-category\" list styles.", + "name": "SetAdDisplayMode", + "params": [ + { + "default": null, + "isRequired": true, + "name": "displayMode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Enables the banner ad to be selectable. When selected, the screen will receive an [roPosterScreenEvent](https://developer.roku.com/docs/references/brightscript/events/roposterscreenevent.md and the msg will return true for [isAdSelected()](https://developer.roku.com/docs/references/brightscript/events/roposterscreenevent.mdisadselected-as-boolean).", + "name": "SetAdSelectable", + "params": [ + { + "default": null, + "description": "A flag specifying if the user can navigate and move focus to the banner image (true) or if the banner ad is not selectable (false).", + "isRequired": true, + "name": "isSelectable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the URL of the banner ad image to be displayed on the poster screen. This is currently only valid for the \"arced-landscape\" and \"flat-category\" style of poster screens.", + "name": "SetAdURL", + "params": [ + { + "default": null, + "description": "The URL of the standard definition banner ad image (540x60) to be displayed.", + "isRequired": true, + "name": "sdAdURL", + "type": "String" + }, + { + "default": null, + "description": "The URL of the high definition banner ad image (728x90) to be displayed.", + "isRequired": true, + "name": "hdAdURL", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Shows or hides the breadcrumb text in the title area.", + "name": "SetBreadcrumbEnabled", + "params": [ + { + "default": null, + "description": "A flag specifying whether the breadcrumb text is to be displayed on the poster screen.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the text to be displayed for the title in breadcrumb format.", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The text to be used for the first location.", + "isRequired": true, + "name": "location1", + "type": "String" + }, + { + "default": null, + "description": "The text to be used for the second location.", + "isRequired": true, + "name": "location2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the list of content to be displayed by the screen. The screen is responsible for fetching the poster art from the URLs specified and all user navigation within the list.", + "name": "SetContentList", + "params": [ + { + "default": null, + "description": "An [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roAssociativeArrays](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArrays\") ([Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") objects) representing the information for each title to be displayed on screen.", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "The item the in filter banner to be given focus. The selected item is displayed in the center of the filter banner, and it is highlighted to denote that it has focus.", + "name": "SetFocusedList", + "params": [ + { + "default": null, + "description": "The zero-based index of the item to obtain focus.", + "isRequired": true, + "name": "itemIndex", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "name": "SetFocusedListItem", + "params": [ + { + "default": null, + "description": "The zero-based index of the item to obtain focus.", + "isRequired": true, + "name": "itemIndex", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the focus to the filter banner.", + "name": "SetFocusToFilterBanner", + "params": [ + { + "default": null, + "description": "A flag specifying whether the filter banner is to obtain focus.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the mode for displaying images in the poster screen. This allows images to be either scaled to completely fill the poster frame (scale-to-fill) or scaled to fit inside the poster frame (scale-to-fit) while maintaining aspect ratio", + "name": "SetListDisplayMode", + "params": [ + { + "default": null, + "description": "Specifies the mode used to display images on the poster screen, which may be one of the following values: * scale-to-fill – scale image to completely fill the rectangle of the bounding frame (Default) * scale-to-fit – scale image to fit horizontally or vertically as appropriate while still maintaining aspect ratio. Note that scale-to-fit may result in pillar-box or letter-box display of images. * zoom-to-fill – scales and crops image to maintain aspect ratio and completely fill the rectangle of the bounding frame. * photo-fit – Uses several methods to fit the image with a different aspect ratio to the screen. First, it will asymmetrically scale up to a maximum of 5%. Second, for landscape images, if vertical cropping is necessary, it will remove two lines off the bottom for every one line off the top up to a maximum of 30% of the image. For all images, if horizontal cropping is necessary it will crop an equal amount from both sides.", + "isRequired": true, + "name": "displayMode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the list of categories to be displayed in the filter banner at the top of the poster screen.", + "name": "SetListNames", + "params": [ + { + "default": null, + "description": "An array of Strings, where each String represents a new category to be displayed at the top. The display order is the same as the order of the categories in the array passed by the caller", + "isRequired": true, + "name": "names", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the display style for the poster screen. Styles allow the poster screen to look differently for different types of content or different usage.", + "name": "SetListStyle", + "params": [ + { + "default": null, + "isRequired": true, + "name": "style", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the default images to be shown in the poster screen while the poster screen content is loading", + "name": "SetLoadingPoster", + "params": [ + { + "default": null, + "isRequired": true, + "name": "sdPosterUrl", + "type": "String" + }, + { + "default": null, + "isRequired": true, + "name": "hdPosterUrl", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title for the poster screen.", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The text to be used as the title for the poster screen.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after initial creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A boolean indicating whether the screen was successfully displayed.", + "returnType": "Boolean" + }, + { + "description": "Displays a semi-transparent popup message box to the user in the center of the screen over the poster screen. Generally, this is used for error messages.", + "name": "ShowMessage", + "params": [ + { + "default": null, + "description": "The text to be displayed in the popup message box.", + "isRequired": true, + "name": "message", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifPosterScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifposterscreen.md" + }, + "ifprogramguide": { + "implementers": [ + { + "description": "Represents Electronic Program Guide (EPG) information from the tuner.", + "name": "roProgramGuide", + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/roprogramguide.md" + } + ], + "methods": [ + { + "description": "Returns the list of logical channel numbers on which the given program ID can be found.", + "name": "GetChannels", + "params": [ + { + "default": null, + "description": "The program ID containing the channels to be returned.", + "isRequired": true, + "name": "id", + "type": "Integer" + } + ], + "returnType": "Object" + }, + { + "description": "Returns details about the current and next program on a channel.", + "name": "GetNowNextPrograms", + "params": [ + { + "default": null, + "description": "The channel number for which programs are to be retrieved.", + "isRequired": true, + "name": "channel", + "type": "String" + } + ], + "returnDescription": "An roAssociativeArray containing two roArray components: one for the current program and another for the next program on the channel. Each roArray contains the following key/value pairs detailing the program:", + "returnType": "Dynamic" + }, + { + "description": "Returns an integer which is incremented each time the underlying data in the guide changes.", + "name": "GetVersion", + "params": [], + "returnDescription": "The version number of the program guide.", + "returnType": "Integer" + } + ], + "name": "ifProgramGuide", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifprogramguide.md" + }, + "ifregex": { + "description": "> See the PCRE documentation ([http://www.pcre.org/](http://www.pcre.org/)) for documentation on the PCRE library used for regular expression matching. See the [Perlre main page](http://perldoc.perl.org/perlre.html \"Perlre main page\") for complete documentation of the possible regular expressions this library can parse and match. In general, most Perl compatible regular expressions are supported.", + "implementers": [ + { + "description": "The roRegex component provides the powerful regular expression processing of the PCRE library to Brightscript strings", + "name": "roRegex", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregex.md" + } + ], + "methods": [ + { + "description": "Checks if a string matches the matching pattern.", + "name": "IsMatch", + "params": [ + { + "default": null, + "description": "The string to be checked.", + "isRequired": true, + "name": "str", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the string matches the matching pattern.", + "returnType": "Boolean" + }, + { + "description": "If the matching pattern contains N parenthetical substrings, the relevant substrings are returned as an array of length N+1, where array\\[0\\] is again the entire match and each additional entry in the array is the match for the corresponding parenthetical expression.", + "name": "Match", + "params": [ + { + "default": null, + "description": "The string to be searched for matching substrings.", + "isRequired": true, + "name": "str", + "type": "String" + } + ], + "returnDescription": "An roArray of matched substrings from str. If no match was made, an empty array is returned. If a match was made, the entire match is returned in array\\[0\\]. If there are no parenthetical substrings this is the only entry in the array", + "returnType": "Object" + }, + { + "description": "Returns all matches of the specific regular expression pattern in the target string.", + "name": "MatchAll", + "params": [ + { + "default": null, + "description": "The string to be searched for matching substrings.", + "isRequired": true, + "name": "str", + "type": "String" + } + ], + "returnDescription": "An roArray where the first element is the full matched string and if there are any capture groups those are returned in subsequent array elements", + "returnType": "Object" + }, + { + "description": "Replaces the first occurrence of a matching pattern in str with replacement and returns the result. The replacement may contain numbered back-references to parenthetical substrings.", + "name": "Replace", + "params": [ + { + "default": null, + "description": "The string to be searched.", + "isRequired": true, + "name": "str", + "type": "String" + }, + { + "default": null, + "description": "The string to be used to replace matches in source string.", + "isRequired": true, + "name": "replacement", + "type": "String" + } + ], + "returnDescription": "A string with the result of the replace operation.", + "returnType": "String" + }, + { + "description": "Replaces all occurrences of a matching pattern in str with replacement and returns the result. The replacement may contain numbered back-references to parenthetical substrings.", + "name": "ReplaceAll", + "params": [ + { + "default": null, + "description": "The string to be searched.", + "isRequired": true, + "name": "str", + "type": "String" + }, + { + "default": null, + "description": "The string to be used to replace matches in source string.", + "isRequired": true, + "name": "replacement", + "type": "String" + } + ], + "returnDescription": "A string with the result of the replace all operation.", + "returnType": "String" + }, + { + "description": "Uses the matching pattern as a separator and splits the string on the separator boundaries.", + "name": "Split", + "params": [ + { + "default": null, + "description": "The string to be split.", + "isRequired": true, + "name": "str", + "type": "String" + } + ], + "returnDescription": "An roList of substrings of str that were separated by strings which match the pattern in the CreateObject call. The separator strings are not returned. If no matches were found, the returned list contains a single item with the string unchanged.", + "returnType": "Object" + } + ], + "name": "ifRegex", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregex.md" + }, + "ifregion": { + "implementers": [ + { + "description": "The roRegion component is used to represent a subsection of a bitmap", + "name": "roRegion", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregion.md" + } + ], + "methods": [ + { + "description": "Returns a newly created copy of the region as a new [roRegion](https://developer.roku.com/docs/references/brightscript/components/roregion.md\"roRegion\") object.", + "name": "Copy", + "params": [], + "returnDescription": "An roRegion Object.", + "returnType": "Object" + }, + { + "description": "Returns the roBitmap object of the bitmap to which this region refers. A region is always a section of a bitmap.", + "name": "GetBitmap", + "params": [], + "returnDescription": "An roBitmap object of the bitmap.", + "returnType": "Object" + }, + { + "description": "Returns the collision type.", + "name": "GetCollisionType", + "params": [], + "returnDescription": "The collision type, which may be one of the following values:", + "returnType": "Integer" + }, + { + "description": "Returns the height of the region.", + "name": "GetHeight", + "params": [], + "returnDescription": "The height of the region.", + "returnType": "Integer" + }, + { + "description": "Returns the pre-translation x value.", + "name": "GetPretranslationX", + "params": [], + "returnDescription": "The pre-translation x value.", + "returnType": "Integer" + }, + { + "description": "Returns the pre-translation y value.", + "name": "GetPretranslationY", + "params": [], + "returnDescription": "The pre-translation y value.", + "returnType": "Integer" + }, + { + "description": "Returns the scaling mode.", + "name": "GetScaleMode", + "params": [], + "returnDescription": "The scaling mode, which may be one of the following values:", + "returnType": "Integer" + }, + { + "name": "GetTime", + "params": [], + "returnType": "Integer" + }, + { + "description": "Returns the width of the region.", + "name": "GetWidth", + "params": [], + "returnDescription": "The width of the region.", + "returnType": "Integer" + }, + { + "description": "Returns if the region can be wrapped.", + "name": "GetWrap", + "params": [], + "returnType": "Boolean" + }, + { + "description": "Returns the x coordinate of the region in its bitmap.", + "name": "GetX", + "params": [], + "returnDescription": "The x coordinate value", + "returnType": "Integer" + }, + { + "description": "Returns the y coordinate of the region in its bitmap.", + "name": "GetY", + "params": [], + "returnDescription": "The y coordinate value", + "returnType": "Integer" + }, + { + "description": "Adds the passed parameters x,y, w, and h to the values of those roRegion fields. Respects the wrap setting when adjusting the fields by the offsets.", + "name": "Offset", + "params": [ + { + "default": null, + "description": "The x-coordinate of the region.", + "isRequired": true, + "name": "x", + "type": "Dynamic" + }, + { + "default": null, + "description": "The y-coordinate of the region.", + "isRequired": true, + "name": "y", + "type": "Dynamic" + }, + { + "default": null, + "description": "The width of the region.", + "isRequired": true, + "name": "w", + "type": "Dynamic" + }, + { + "default": null, + "description": "The height of the region.", + "isRequired": true, + "name": "h", + "type": "Dynamic" + } + ], + "returnType": "Void" + }, + { + "description": "Initializes the fields of this region to be the same as the values of the fields in the srcRegion.", + "name": "Set", + "params": [ + { + "default": null, + "description": "An roRegion object.", + "isRequired": true, + "name": "srcRegion", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the collision circle used for type-2 collision tests. The center of the circle is the (x,y) position of the sprite plus the specified offsets. The radius specifies the size of the circle.", + "name": "SetCollisionCircle", + "params": [ + { + "default": null, + "description": "The offset for the x position of the sprite.", + "isRequired": true, + "name": "xOffset", + "type": "Integer" + }, + { + "default": null, + "description": "The offset for the y position of the sprite.", + "isRequired": true, + "name": "yOffset", + "type": "Integer" + }, + { + "default": null, + "description": "The size of the circle.", + "isRequired": true, + "name": "Radius", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the collision rectangle used for type-1 collision tests. The upper left corner of the rectangle is the (x,y) position of the sprite plus the specified offsets. The width and height specify the size of the rectangle.", + "name": "SetCollisionRectangle", + "params": [ + { + "default": null, + "description": "The offset for the x position of the sprite.", + "isRequired": true, + "name": "xOffset", + "type": "Integer" + }, + { + "default": null, + "description": "The offset for the y position of the sprite.", + "isRequired": true, + "name": "yOffset", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the rectangle.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the rectangle.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the type of region to be used for collision tests with this sprite.", + "name": "SetCollisionType", + "params": [ + { + "default": null, + "description": "The collision type, which may be one of the following values: * Type 0– Use the entire defined region of the sprite. Type 0 is the default * Type 1 – Use the defined rectangular region specified by the SetCollisionRectangle() method * Type 2 – Use a circular region specified by the SetCollisionCircle() method", + "isRequired": true, + "name": "collisiontype", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the pre-translation for DrawObject, DrawRotatedObject, and DrawScaledObject.", + "name": "SetPretranslation", + "params": [ + { + "default": null, + "description": "The pre-translation x-value.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The pre-translation y-value.", + "isRequired": true, + "name": "y", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the scaling mode used for DrawScaledObject.", + "name": "SetScaleMode", + "params": [ + { + "default": null, + "description": "The scaling mode, which may be one of the following values: * 0 = fast scaling operation (may have jaggies) * 1 = smooth scaling operation (may be slow)", + "isRequired": true, + "name": "mode", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the duration of each frame of any animated sprite that uses this region.", + "name": "SetTime", + "params": [ + { + "default": null, + "description": "The \"frame hold time\" in milliseconds.", + "isRequired": true, + "name": "time", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Wraps any part of a region that extends beyond the bounds of its bitmap to the other side of the bitmap and renders it there.", + "name": "SetWrap", + "params": [ + { + "default": null, + "description": "A flag specifying whether wrapping of the region is enabled. If this flag is set to false, the part of the region beyond the bounds of its bitmap is not rendered.", + "isRequired": true, + "name": "wrap", + "type": "Boolean" + } + ], + "returnType": "Void" + } + ], + "name": "ifRegion", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregion.md" + }, + "ifregistry": { + "implementers": [ + { + "description": "The Registry is an area of non-volatile storage where a small number of persistent settings can be stored", + "name": "roRegistry", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregistry.md" + } + ], + "methods": [ + { + "description": "Deletes the specified registry section.", + "name": "Delete", + "params": [ + { + "default": null, + "description": "The registry section to be deleted.", + "isRequired": true, + "name": "section", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the registry section was successfully deleted.", + "returnType": "Boolean" + }, + { + "description": "Flushes the contents of the registry out to persistent storage in order to permanently store a token or other setting on the device.", + "name": "Flush", + "params": [], + "returnDescription": "A flag indicating whether the registry was successfully flushed.", + "returnType": "Boolean" + }, + { + "description": "Returns the registry sections on the device.", + "name": "GetSectionList", + "params": [], + "returnDescription": "An roList with one entry for each registry section. Each registry section is an roString containing the name of the section. The section itself can be accessed by creating an [roRegistrySection](/docs/references/brightscript/components/roregistrysection.md \"roRegistrySection\") object using that name.", + "returnType": "Object" + }, + { + "description": "Returns the number of bytes available in the channel application's device registry (16K minus current file size). This function can be used, for example, to check the remaining space and remove older entries before writing newer ones. The following code demonstrates how to do this:", + "name": "GetSpaceAvailable", + "params": [], + "returnDescription": "An integer representing the the number of bytes available in the device registry.", + "returnType": "Integer" + } + ], + "name": "ifRegistry", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregistry.md" + }, + "ifregistrysection": { + "implementers": [ + { + "description": "A Registry Section enables the organization of settings within the registry", + "name": "roRegistrySection", + "url": "https://developer.roku.com/docs/references/brightscript/components/roregistrysection.md" + } + ], + "methods": [ + { + "name": "Delete", + "params": [ + { + "default": null, + "description": "The key to be deleted.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the key was successfully deleted.", + "returnType": "Boolean" + }, + { + "description": "Checks if the specified key resides in the registry.", + "name": "Exists", + "params": [ + { + "default": null, + "description": "The key to be checked.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the key is in the registry.", + "returnType": "Boolean" + }, + { + "description": "Flushes the contents of the registry out to persistent storage in order to permanently store a token or other setting on the device. Developers should explicitly this method after performing a write or series of writes. This method is transactional and all writes between calls to it are atomic.", + "name": "Flush", + "params": [], + "returnDescription": "A flag indicating whether the registry was successfully flushed.", + "returnType": "Boolean" + }, + { + "description": "Gets a list of the keys in the registry.", + "name": "GetKeyList", + "params": [], + "returnDescription": "An roList containing one entry per registry key in this section.", + "returnType": "Object" + }, + { + "description": "Reads and returns the value of the specified key.", + "name": "Read", + "params": [ + { + "default": null, + "description": "The key name to be read.", + "isRequired": true, + "name": "key", + "type": "String" + } + ], + "returnDescription": "The value of the key.", + "returnType": "String" + }, + { + "description": "Reads multiple values from the registry.", + "name": "ReadMulti", + "params": [ + { + "default": null, + "description": "An array of strings containing the key names to be read.", + "isRequired": true, + "name": "keysArray", + "type": "Object" + } + ], + "returnDescription": "An associative array containing the keys and corresponding values read from the registry.", + "returnType": "Object" + }, + { + "description": "Replaces the value of the specified key. Does not guarantee a commit to non-volatile storage until an explicit [Flush()](#flush-as-boolean) is done.", + "name": "Write", + "params": [ + { + "default": null, + "description": "The name of the key to be updated.", + "isRequired": true, + "name": "key", + "type": "String" + }, + { + "default": null, + "description": "The updated value to be written to the specified key.", + "isRequired": true, + "name": "value", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the value of the key was successfully updated.", + "returnType": "Boolean" + }, + { + "description": "Writes multiple values to the registry.", + "name": "WriteMulti", + "params": [ + { + "default": null, + "description": "An associative array with key-value pairs to be updated.", + "isRequired": true, + "name": "roAssociativeArray", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the values were successfully updated.", + "returnType": "Boolean" + } + ], + "name": "ifRegistrySection", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifregistrysection.md" + }, + "ifrsa": { + "implementers": [ + { + "description": "The RSA component provides an interface to the OpenSSL RSA library of signing algorithms", + "name": "roRSA", + "url": "https://developer.roku.com/docs/references/brightscript/components/rorsa.md" + } + ], + "methods": [ + { + "name": "SetDigestAlgorithm", + "params": [ + { + "default": null, + "description": "An openssl string with the digest to be used. Common digest algorithms are \"sha1\", \"ripemd160\", and \"md5\".", + "isRequired": true, + "name": "digestAlgorithm", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the algorithm was successfully set (true) or the string was not recognized (false).", + "returnType": "Boolean" + }, + { + "description": "Specifies the private key to use for signing.", + "name": "SetPrivateKey", + "params": [ + { + "default": null, + "description": "Specifies the private key to be used for signing. The file name should specify a path, either in the package or a temp path.", + "isRequired": true, + "name": "keyFileName", + "type": "String" + } + ], + "returnType": "Integer" + }, + { + "name": "SetPublicKey", + "params": [ + { + "default": null, + "description": "Specifies the public key to be used for signing. The file name should specify a path, either in the package or a temp path.", + "isRequired": true, + "name": "keyFileName", + "type": "String" + } + ], + "returnType": "Integer" + }, + { + "description": "Generates a signature based on the specified digest.", + "name": "Sign", + "params": [ + { + "default": null, + "description": "The roByteArray to be signed. Errors will be printed in the BrightScript console. If the digest algorithm is not set (using SetDigestAlgorithm) before calling Sign(), the digest is not encapsulated. This would be equivalent to simply calling the openssl function RSA\\_private\\_encrypt()", + "isRequired": true, + "name": "digest", + "type": "roByteArray Object" + } + ], + "returnDescription": "An roByteArray containing the signature, or invalid if an error occurred. Typical values include the following:", + "returnType": "Object" + }, + { + "description": "Verifies the given digest and signature. Both digest and signature should be roByteArrays. If the digest algorithm is not set (using the [SetDigestAlgorithm](#setdigestalgorithmdigestalgorithm-as-string-as-boolean) method) before calling Verify(), the digest associated with the signature is not expected to be encapsulated. This would be equivalent to simply calling the openssl function RSA\\_public\\_decrypt(signature) and then comparing the result with the digest", + "name": "Verify", + "params": [ + { + "default": null, + "description": "The digest to be verified.", + "isRequired": true, + "name": "digest", + "type": "roByteArray Object" + }, + { + "default": null, + "description": "The signature to be verified.", + "isRequired": true, + "name": "signature", + "type": "roByteArray Object" + } + ], + "returnDescription": "Indicates the result of the validation. This may be one of the following values:", + "returnType": "Integer" + } + ], + "name": "ifRSA", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifrsa.md" + }, + "ifscreen": { + "description": "| Name | Description |\n| --- | --- |\n| [roScreen](https://developer.roku.com/docs/references/brightscript/components/roscreen.md\"roScreen\") | The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from |", + "implementers": [ + { + "description": "The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from", + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + } + ], + "methods": [ + { + "description": "This function first operates the same as a call to [ifDraw2D](https://developer.roku.com/docs/references/brightscript/interfaces/ifdraw2d.md\"ifDraw2D\"), completing all queued drawing operations on the back buffer (draw surface).", + "name": "SwapBuffers", + "params": [], + "returnType": "Void" + } + ], + "name": "ifScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifscreen.md" + }, + "ifsearchhistory": { + "description": "> _This interface is known as ifRoSearchHistory in some Roku OS versions._", + "implementers": [ + { + "description": "The Search History object implements the system-wide storage of search terms for use in implementing the roSearchScreen", + "name": "roSearchHistory", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosearchhistory.md" + } + ], + "methods": [ + { + "description": "Clears all elements from the search history.", + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the current search history stack.", + "name": "GetAsArray", + "params": [], + "returnDescription": "An [roArray](/docs/references/brightscript/components/roarray.md \"roArray\") of Strings with all available search history elements.", + "returnType": "Object" + }, + { + "description": "Pushes a new search term onto the search history stack.", + "name": "Push", + "params": [ + { + "default": null, + "description": "The search term to be pushed onto the search history stack.", + "isRequired": true, + "name": "searchTerm", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifSearchhistory", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsearchhistory.md" + }, + "ifsearchscreen": { + "implementers": [ + { + "description": "The Search Screen provides a standard way to allow users to enter text for searching", + "name": "roSearchScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosearchscreen.md" + } + ], + "methods": [ + { + "description": "Adds an individual value to the search term list.", + "name": "AddSearchTerm", + "params": [ + { + "default": null, + "description": "The search term to be added.", + "isRequired": true, + "name": "searchTerm", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Clears all values from the search terms list.", + "name": "ClearSearchTerms", + "params": [], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Shows or hides the breadcrumb text in the title area", + "name": "SetBreadcrumbEnabled", + "params": [ + { + "default": null, + "description": "A flag specifying whether to show (true) or hide (false) the breadcrumb text.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets a two-part navigational title that shows the current and the previous locations in the application hierarchy (for example, TV – Friends).", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The location 1 name.", + "isRequired": true, + "name": "location1", + "type": "String" + }, + { + "default": null, + "description": "The location 2 name.", + "isRequired": true, + "name": "location2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Shows or hides the clear button on the keypad.", + "name": "SetClearButtonEnabled", + "params": [ + { + "default": null, + "description": "A flag specifying whether to show (true) or hide (false) the clear button.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Set the text label for the button at the bottom of the list area. Example text might be \"clear history\", \"clear results\" or similar", + "name": "SetClearButtonText", + "params": [ + { + "default": null, + "description": "The text label to be displayed on the button at the bottom of the list area", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays the default text in the search terms box when no search terms have been entered.", + "name": "SetEmptySearchTermsText", + "params": [ + { + "default": null, + "description": "The text to be displayed in the search terms box by default.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the text label to be displayed on the search button. For example \"search\", \"find\", etc.", + "name": "SetSearchButtonText", + "params": [ + { + "default": null, + "description": "The text label to be displayed on the search button.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the text to be displayed for the header in the list area. This area could contain a list of search terms previously used as a search history or partial results in the case of a progressive disclosure search", + "name": "SetSearchTermHeaderText", + "params": [ + { + "default": null, + "description": "The text to be displayed for the header in the list are.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the search terms list to the values contained in the array provided.", + "name": "SetSearchTerms", + "params": [ + { + "default": null, + "description": "An array of string values to be displayed.", + "isRequired": true, + "name": "searchTerms", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the keyboard search string box to the specified text.", + "name": "SetSearchText", + "params": [ + { + "default": null, + "description": "The text to be displayed in the keyboard search string box.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the search screen successfully was displayed.", + "returnType": "Boolean" + } + ], + "name": "ifSearchScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsearchscreen.md" + }, + "ifsetmessageport": { + "implementers": [ + { + "description": "The HDMI status component provides an interface to the current HDMI operational status", + "name": "roHdmiStatus", + "url": "https://developer.roku.com/docs/references/brightscript/components/rohdmistatus.md" + }, + { + "description": "The roScreen component provides a full screen drawing surface that can be stacked and that you can receive input events from", + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + }, + { + "description": "The roTextToSpeech component provides text to speech capabilities to applications", + "name": "roTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md" + }, + { + "description": "A roUrlTransfer object transfers data to or from remote servers specified by URLs. It can perform mutual authentication with a web server", + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + } + ], + "methods": [ + { + "description": "Sets the [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md\"roMessagePort\") to be used for all events from the screen.", + "name": "SetMessagePort", + "params": [ + { + "default": null, + "description": "The roMessagePort to be used for screen events.", + "isRequired": true, + "name": "port", + "type": "Object" + } + ], + "returnType": "Void" + } + ], + "name": "ifSetMessagePort", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsetmessageport.md" + }, + "ifsgnodeboundingrect": { + "description": "The ifSGNodeBoundingRect interface can be used to query the bounding rectangle of subject node. The ifSGNodeBoundingRect interface methods return a node bounding rectangle as an associative array with four elements:\n\n| Name | Value |\n| --- | --- |\n| x | x-coordinate of the origin of the bounding rectangle |\n| y | y-coordinate of the origin of the bounding rectangle |\n| width | width of the bounding rectangle |\n| height | height of the bounding rectangle |\n\n> These methods return the bounding rectangle dimensions and location of component objects at the time they are called. If they are called before an object is fully constructed, such as before all graphical images have been loaded, they will return the dimensions and location at the time of the call, which may not be the correct values for placing the component object properly. To ensure that your screen has the component objects located as you intended, make sure you call these methods after the component object is fully constructed. For example, if the component object relies on loading graphical images to construct its appearance, it is best to use these methods as part of an observer callback function triggered by the image loading field events, such as the loadStatus field of the Poster node.", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Returns the node bounding rectangle. The bounding rectangle of a node is the axis-aligned rectangle computed by transforming the local bounding rectangle of the node by the node transformation matrix. The resulting rectangle corresponds to the node local bounding rectangle transformed into its parent node local coordinate system.", + "name": "boundingRect", + "params": [], + "returnDescription": "An associative array with the node bounding rectangle.", + "returnType": "Dynamic" + }, + { + "description": "Returns the node local bounding rectangle. The local bounding rectangle of a node is the axis-aligned rectangle, that includes the union of the bounding rectangle of the geometry of the node, and the bounding rectangles of all of the node children, transformed into the local coordinate system of the node.", + "name": "localBoundingRect", + "params": [], + "returnDescription": "An associative array with the node local bounding rectangle.", + "returnType": "Dynamic" + }, + { + "description": "Returns the local bounding rectangle of this node's identified sub part in the node's local coordinate system. If the subpart does not exist, the node's local bounding rectangle is returned.", + "name": "localSubBoundingRect", + "params": [ + { + "default": null, + "description": "The index of the grid item for the local bounding rectangle to be returned in the following format: _rowindex_\\__itemindex_.", + "isRequired": true, + "name": "itemnumber", + "type": "String" + } + ], + "returnDescription": "An associative array with the local bounding rectangle of the node's identified sub part.", + "returnType": "Dynamic" + }, + { + "description": "Returns the bounding rectangle for scene components (component nodes extended from a [Scene](https://developer.roku.com/docs/references/scenegraph/scene.md\"Scene\") or [OverhangPanelSetScene](https://developer.roku.com/docs/references/scenegraph/sliding-panels-nodes/overhangpanelsetscene.md\"OverhangPanelSetScene\") node class).", + "name": "sceneBoundingRect", + "params": [], + "returnDescription": "An associative array with the bounding rectangle.", + "returnType": "Dynamic" + }, + { + "description": "Returns the bounding rectangle of this node's subpart in its Scene's coordinate system If the subpart does not exist or if the node is not an ancestor of a Scene node, this will return the node's bounding rectangle.", + "name": "sceneSubBoundingRect", + "params": [ + { + "default": null, + "description": "The index of the grid item for the bounding rectangle to be returned in the following format: _rowindex_\\__itemindex_.", + "isRequired": true, + "name": "itemnumber", + "type": "String" + } + ], + "returnDescription": "An associative array with the bounding rectangle.", + "returnType": "Dynamic" + }, + { + "description": "Returns the bounding rectangle of this node's identified sub part, as transformed by this node's transformation matrix, in its parent node's coordinate system. If the subpart does not exist, the node's bounding rectangle is returned.", + "name": "subBoundingRect", + "params": [ + { + "default": null, + "description": "The index of the grid item for the local bounding rectangle to be returned in the following format: _rowindex_\\__itemindex_.", + "isRequired": true, + "name": "itemnumber", + "type": "String" + } + ], + "returnDescription": "An associative array with the bounding rectangle.", + "returnType": "Dynamic" + } + ], + "name": "ifSGNodeBoundingRect", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodeboundingrect.md" + }, + "ifsgnodechildren": { + "description": "The ifSGNodeChildren interface allows querying and manipulation of nodes in a SceneGraph node tree, such as creating new nodes, placing them at certain positions in the tree, and removing them.", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Appends an array of children nodes to the subject node.", + "name": "appendChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "child" + } + ], + "returnDescription": "A flag indicating whether the children nodes were successfully appended." + }, + { + "description": "Creates a child node of type nodeType, and adds the new node to the end of the subject node list of children.", + "name": "createChild", + "params": [ + { + "default": null, + "description": "The node class to be created.", + "isRequired": true, + "name": "nodeType", + "type": "String" + } + ], + "returnDescription": "The child node that was created.", + "returnType": "Object" + }, + { + "description": "Creates a specific number of new child nodes of a specific type or extended type.", + "name": "createChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "num" + } + ], + "returnDescription": "An roArray containing the new child nodes created." + }, + { + "description": "Returns an array with every existing node created by the currently running channel.", + "name": "getAll", + "params": [], + "returnDescription": "An roArray with the all the existing nodes created by the channel.", + "returnType": "Object" + }, + { + "description": "Returns an array with every existing node created by the currently running channel (similar to the [getAll()](#getall-as-object) method) organized as an XML forest of trees according to the usual parent-child node relationship. Cycles are handled with a reference entry in the tree rather than indefinite recursion.", + "name": "getAllMeta", + "params": [], + "returnDescription": "An roArray of strings with the all the existing nodes created by the channel.", + "returnType": "Object" + }, + { + "description": "Returns the child node specified by the index.", + "name": "getChild", + "params": [ + { + "default": null, + "description": "The index of the child node to be retrieved.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "The child node at the index position; otherwise, \"invalid\".", + "returnType": "Dynamic" + }, + { + "description": "Returns the current number of children in the subject node list of children. This is always a non-negative number.", + "name": "getChildCount", + "params": [], + "returnDescription": "The number of child nodes in the tree.", + "returnType": "Integer" + }, + { + "description": "Retrieves a specific number of child nodes from the subject node, starting at a specific position.", + "name": "getChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "num" + } + ], + "returnDescription": "An roArray containing the child nodes retrieved. If num\\_children is -1, all the child nodes are returned." + }, + { + "description": "Returns the parent node of a node has been added to a list of children.", + "name": "getParent", + "params": [], + "returnDescription": "The parent node; otherwise, \"invalid\".", + "returnType": "roSGNode" + }, + { + "description": "Returns an array with every existing node without a parent created by the currently running channel.", + "name": "getRoots", + "params": [], + "returnType": "Object" + }, + { + "description": "Returns an array with every existing node without a parent created by the currently running channel.", + "name": "getRootsMeta", + "params": [], + "returnDescription": "An roArray with every existing node without a parent created by the currently running channel.", + "returnType": "Object" + }, + { + "description": "Returns the node's root Scene. This returns a valid Scene even if the node is not parented.", + "name": "getScene", + "params": [], + "returnDescription": "The node's root Scene.", + "returnType": "roSGNode" + }, + { + "description": "Inserts an array of child nodes to the subject node, starting at a specific position.", + "name": "insertChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "child" + } + ], + "returnDescription": "A flag indicating whether the children nodes were successfully inserted." + }, + { + "description": "If the subject node has a child node in the index position, removes that child node from the subject node list of children.", + "name": "removeChildIndex", + "params": [ + { + "default": null, + "description": "The position in the tree of the child node to be removed.", + "isRequired": true, + "name": "index", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the child node that was successfully removed.", + "returnType": "Boolean" + }, + { + "description": "Removes an array of child nodes from the subject node.", + "name": "removeChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "child" + } + ], + "returnDescription": "A flag indicating whether the children nodes were successfully removed." + }, + { + "description": "Removes a specific number of child nodes from the subject node starting at a specific position.", + "name": "removeChildrenIndex", + "params": [ + { + "default": null, + "isRequired": true, + "name": "num" + } + ], + "returnDescription": "A flag indicating whether the children nodes were successfully removed." + }, + { + "description": "Replaces the child nodes in the subject node, starting at the position specified by index, with new child nodes specified by child\\_nodes", + "name": "replaceChildren", + "params": [ + { + "default": null, + "isRequired": true, + "name": "child" + } + ], + "returnDescription": "A flag indicating whether the children nodes were successfully replaced." + } + ], + "name": "ifSGNodeChildren", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodechildren.md" + }, + "ifsgnodedict": { + "description": "The ifSGNodeDict interface allows you access information about the nodes in a SceneGraph node tree, and find and return a node with a specific ID.", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "name": "clone", + "params": [ + { + "default": null, + "isRequired": true, + "name": "isDeepCopy", + "type": "Boolean" + } + ], + "returnDescription": "A node tree.", + "returnType": "Object" + }, + { + "description": "Returns the node that is a descendant of the nearest component ancestor of the subject node (possibly the subject node itself) and whose id field is set to name. The search for the descendant node is a breadth-first search that includes child nodes in nodes that are declared as custom components defined in other XML component files. These together allow finding siblings and cousins of a node within the context of a component. If a node with the specified name is not found, an invalid object is returned", + "name": "findNode", + "params": [ + { + "default": null, + "description": "The name of the node to be retrieved.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "The node that is a descendant of the nearest component ancestor of the subject node.", + "returnType": "Object" + }, + { + "description": "Checks whether a specific roSGNode refers to the same SceneGraph node object as the subject node.", + "name": "isSameNode", + "params": [ + { + "default": null, + "isRequired": true, + "name": "RoSGNode", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the nodes refer to the same SceneGraph node object.", + "returnType": "Boolean" + }, + { + "description": "Checks whether the subtype of the subject node is a descendant of the subtype nodeType in the SceneGraph node class hierarchy.", + "name": "isSubtype", + "params": [ + { + "default": null, + "description": "The node type of the subject node.", + "isRequired": true, + "name": "nodeType", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the subtype of the subject node is a descendant of the subtype nodeType.", + "returnType": "Boolean" + }, + { + "description": "Returns the subtype of the parent of the nodeType in the SceneGraph node class hierarchy.", + "name": "parentSubtype", + "params": [ + { + "default": null, + "description": "The node type of the parent node.", + "isRequired": true, + "name": "nodeType", + "type": "String" + } + ], + "returnDescription": "The subtype of the parent node.", + "returnType": "String" + }, + { + "description": "Returns the subtype of the subject node as specified when it was created.", + "name": "subtype", + "params": [], + "returnDescription": "The subtype of the subject node.", + "returnType": "String" + } + ], + "name": "ifSGNodeDict", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodedict.md" + }, + "ifsgnodefield": { + "description": "The ifSGNodeField interface allows querying, getting, setting, and performing other similar manipulation operations on Scene Graph node fields. This interface also allows you to set and unset event observers on a subject node field.", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Adds a field with the specified name and type to the subject node. The added field is initialized to the default value for the type.", + "name": "addField", + "params": [ + { + "default": null, + "description": "The name of the field to be added.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The type of the field to be added. Type declarations must be lowercase or the field will not be added to the node. For example, declaring \"Boolean\" as the type will prevent the field from being added.", + "isRequired": true, + "name": "type", + "type": "String" + }, + { + "default": null, + "description": "Specifies whether observers of the field are triggered when the field value is updated to the same or new value (true), or only when the field changes to a new value (false).", + "isRequired": true, + "name": "alwayNotify", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the field have been successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds the field(s) and corresponding field value(s) defined as key-value pair(s) in the associative array fields to the subject node. The types of the added fields are determined by the values which correspond to the allowable types for an `` field.", + "name": "addFields", + "params": [ + { + "default": null, + "description": "An roAssociativeArray containing key-value pairs for the fields to be added.", + "isRequired": true, + "name": "fields", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the fields have been successfully added.", + "returnType": "Boolean" + }, + { + "description": "Returns the appropriately-typed value from the specified field of the subject node.", + "name": "getField", + "params": [ + { + "default": null, + "description": "The name of the field to be retrieved.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "A typed value.", + "returnType": "Object" + }, + { + "description": "Returns the names and values of all the fields in the node.", + "name": "getFields", + "params": [], + "returnDescription": "An roAssociativeArray containing key-value pairs with the element names and values.", + "returnType": "Object" + }, + { + "description": "Returns the type of a specific field of the subject node.", + "name": "getFieldType", + "params": [ + { + "default": null, + "description": "The name of the field to have its type retrieved.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "The field type.", + "returnType": "String" + }, + { + "description": "Returns the names and types of all the fields in the node.", + "name": "getFieldTypes", + "params": [], + "returnDescription": "An roAssociativeArray containing key-value pairs with the element names and types.", + "returnType": "Object" + }, + { + "description": "Checks whether a field exists in the node.", + "name": "hasField", + "params": [ + { + "default": null, + "description": "The name of the field to be checked for whether it exists in the node.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the subject node has a field whose name exactly matches fieldName, or whose fully lowercase analog is identical to that of fieldName.", + "returnType": "Boolean" + }, + { + "description": "Calls a function when a field of the subject node changes. The function called must be in the scope of the current component.", + "name": "observeField", + "params": [ + { + "default": null, + "description": "The name of the field to be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The name of the method to be executed when the value of the field changes.", + "isRequired": true, + "name": "functionName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the value of the field being monitored changes." + }, + { + "description": "This overloaded form sends an [roSGNodeEvent](https://developer.roku.com/docs/references/brightscript/components/rosgnode.md\"roSGNodeEvent\") message to the [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md\"roMessagePort\") identified by port when the subject node field identified by fieldName changes value.", + "name": "observeField", + "params": [ + { + "default": null, + "description": "The name of the field to be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md\"roMessagePort\") to receive a [roSGNodeEvent](https://developer.roku.com/docs/references/brightscript/components/rosgnode.md\"roSGNodeEvent\") message when the value of the field changes.", + "isRequired": true, + "name": "port", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the value of the field being monitored changes." + }, + { + "description": "Sets up a connection between the observed node's field and the current component from which this call is made. This method is similar to the [observeField()](https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.mdobservefieldfieldname-as-string-functionname-as-string-as-boolean \"observeField(fieldName as String, functionName as String)\") method.", + "name": "observeFieldScoped", + "params": [ + { + "default": null, + "description": "The name of the field to be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The name of the method to be executed when the value of the field changes.", + "isRequired": true, + "name": "functionName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the value of the field has changed." + }, + { + "description": "Sets up a connection between the observed node's field and the current component from which this call is made. This method is similar to the [observeField()](https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.mdobservefieldfieldname-as-string-functionname-as-string-as-boolean \"observeField(fieldName as String, functionName as String)\") method.", + "name": "observeFieldScoped", + "params": [ + { + "default": null, + "description": "The name of the field to be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md\"roMessagePort\") to receive a [roSGNodeEvent](https://developer.roku.com/docs/references/brightscript/components/rosgnode.md\"roSGNodeEvent\") message when the value of the field changes.", + "isRequired": true, + "name": "port", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the value of the field has changed." + }, + { + "description": "Makes subsequent operations on the node fields to queue on the node itself rather than on the [Scene](https://developer.roku.com/docs/references/scenegraph/scene.md\"Scene\") node render thread. This prevents the operations from being executed immediately.", + "name": "queueFields", + "params": [ + { + "default": null, + "description": "A flag enabling queuing on the node.", + "isRequired": true, + "name": "queueNode", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating the current state of **queueNode**.", + "returnType": "Boolean" + }, + { + "description": "Removes a field from the subject node. Fields defined in [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") and the related SceneGraph node class metadata bindings can be removed, but will be dynamically re-added at any time they are explicitly accessed.", + "name": "removeField", + "params": [ + { + "default": null, + "description": "The name of the field to be removed.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the field has been successfully removed.", + "returnType": "Boolean" + }, + { + "description": "Removes one or more fields from the subject node.", + "name": "removeFields", + "params": [ + { + "default": null, + "description": "An roArray containing the names of the fields to be removed.", + "isRequired": true, + "name": "fieldNames", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the fields have been successfully removed.", + "returnType": "Boolean" + }, + { + "description": "Sets the value of a subject node field. This will fail and stop script execution if the value is not of the appropriate type.", + "name": "setField", + "params": [ + { + "default": null, + "description": "The name of the field to be updated.", + "isRequired": true, + "name": "fieldName", + "type": "String" + }, + { + "default": null, + "description": "The updated value for the field.", + "isRequired": true, + "name": "value", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the field was successfully updated.", + "returnType": "Boolean" + }, + { + "description": "Sets the values for one or more fields.", + "name": "setFields", + "params": [ + { + "default": null, + "description": "An roAssociativeArray containing key-value pairs for the fields to be updated.", + "isRequired": true, + "name": "fields", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the fields have been successfully updated.", + "returnType": "Boolean" + }, + { + "description": "Signals start and/or stop points for measuring channel launch and Electronic Program Grid (EPG) launch times.", + "name": "signalBeacon", + "params": [ + { + "default": null, + "isRequired": true, + "name": "beacon", + "type": "String" + } + ], + "returnDescription": "When you fire a launch event, the system will return an integer indicating the result of its signaling:", + "returnType": "Integer" + }, + { + "description": "A runtime debugging method for helping minimize Rendezvous spread. This method can be called on any node from any thread.", + "name": "threadinfo", + "params": [], + "returnDescription": "An roAssociatveArray with the following information:", + "returnType": "Object" + }, + { + "description": "Removes the previously established connections between the subject node field identified by fieldName and any callback functions or message ports.", + "name": "unobserveField", + "params": [ + { + "default": null, + "description": "The name of the field to no longer be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Removes the connection between the observing component and the observed node's field.", + "name": "unobserveFieldScoped", + "params": [ + { + "default": null, + "description": "The name of the field to no longer be monitored.", + "isRequired": true, + "name": "fieldName", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + } + ], + "name": "ifSGNodeField", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefield.md" + }, + "ifsgnodefocus": { + "description": "The ifSGNodeFocus interface is used to query and manipulate the remote control focus of the nodes in a SceneGraph node tree.", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Checks whether the subject node has the remote control focus.", + "name": "hasFocus", + "params": [], + "returnDescription": "A flag indicating whether the subject node has the remote control focus.", + "returnType": "Boolean" + }, + { + "description": "Checks whether the subject node or any of its descendants in the SceneGraph node tree have remote control focus.", + "name": "isInFocusChain", + "params": [], + "returnDescription": "A flag indicating whether the subject node or any of its descendants in the SceneGraph node tree have the remote control focus.", + "returnType": "Boolean" + }, + { + "description": "Sets the current remote control focus to the subject node.", + "name": "setFocus", + "params": [ + { + "default": null, + "description": "True = Sets the current remote control focus to the subject node. This also automatically removes focus from the node on which it was previously set. False = Removes focus from the subject node if it had it. Setting the remote control focus to false is rarely necessary, and can lead to unexpected behavior.", + "isRequired": true, + "name": "on", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether focus on the subject node has successfully been updated.", + "returnType": "Boolean" + } + ], + "name": "ifSGNodeFocus", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodefocus.md" + }, + "ifsgnodehttpagentaccess": { + "description": "The ifSGNodeHttpAgentAccess interface allows you to get an [roHttpAgent](https://developer.roku.com/docs/references/brightscript/components/rohttpagent.md\"roHttpAgent\") object from a SceneGraph node, and set an roHttpAgent object for a nod", + "implementers": [ + { + "description": "The roSGNode object is the BrightScript equivalent of SceneGraph XML file node creation", + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Returns the roHttpAgent object for the node.", + "name": "getHttpAgent", + "params": [], + "returnDescription": "The roHttpAgent object for the node, which may be one of the following:", + "returnType": "Object" + }, + { + "name": "setHttpAgent", + "params": [ + { + "default": null, + "isRequired": true, + "name": "HTTP" + } + ], + "returnDescription": "A flag indicating whether the roHttpAgent object was successfully set." + } + ], + "name": "ifSGNodeHttpAgentAccess", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgnodehttpagentaccess.md" + }, + "ifsgscreen": { + "implementers": [ + { + "description": "The roSGScreen object is a SceneGraph canvas that displays the contents of a SceneGraph Scene node tree", + "name": "roSGScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "description": "Removes the SceneGraph scene from the display screen.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Creates the SceneGraph scene object based on the specified sceneType object.", + "name": "CreateScene", + "params": [ + { + "default": null, + "description": "The sceneType object to be used to create the scene object.", + "isRequired": true, + "name": "sceneType", + "type": "String" + } + ], + "returnDescription": "The roSGScene object associated with the screen.", + "returnType": "Object" + }, + { + "description": "Returns a global reference object for the SceneGraph application.", + "name": "getGlobalNode", + "params": [], + "returnDescription": "A global reference object.", + "returnType": "roSGNode" + }, + { + "description": "Returns the roMessagePort object for the SceneGraph scene.", + "name": "GetMessagePort", + "params": [], + "returnDescription": "The roMessagePort object.", + "returnType": "roMessagePort" + }, + { + "description": "The roSGScene object associated with the screen.", + "name": "GetScene", + "params": [], + "returnDescription": "Typically, the scene created in main.brs by a roSGScreen.CreateScene() call.", + "returnType": "roSGNode" + }, + { + "description": "Renders the SceneGraph scene defined by the roSGScreen object on the display screen.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen is displayed.", + "returnType": "Boolean" + } + ], + "name": "ifSGScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsgscreen.md" + }, + "ifslideshow": { + "deprecatedDescription": "This interface is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Slide Show screen provides the ability to setup a photo slide show to playback a series of images", + "name": "roSlideShow", + "url": "https://developer.roku.com/docs/references/brightscript/components/roslideshow.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Adds a button to the screen. The buttons are displayed in a standard location on the screen and appear in the order added. When the button is pressed, the script will receive an event from the application containing the ID of the button pressed", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The title used for the button.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Add a new [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\") item to the end of the content list for the slide show.", + "name": "AddContent", + "params": [ + { + "default": null, + "description": "The content metadata item to be added to the slideshow list.", + "isRequired": true, + "name": "contentItem", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "name": "AddRatingButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the rating button instance.", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "A value between 1-100 that represents the number of stars (1 to 5) to be displayed. Essentially this a percentage value: <20% = 1 star.", + "isRequired": true, + "name": "rating", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the rating button was successfully added.", + "returnType": "Boolean" + }, + { + "name": "ClearButtons", + "params": [], + "returnType": "Void" + }, + { + "description": "Clears all content from the content list.", + "name": "ClearContent", + "params": [], + "returnType": "Void" + }, + { + "description": "Closes the screens and delete the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the count of all the buttons added to the slide show screen.", + "name": "CountButtons", + "params": [], + "returnDescription": "The button count value.", + "returnType": "Integer" + }, + { + "description": "Retrieves the maximum scale factor.", + "name": "GetMaxUpscale", + "params": [], + "returnDescription": "The scale factor.", + "returnType": "Float" + }, + { + "description": "Puts the slide show into pause mode. Setting the player to pause mode if it is not in play mode generates in error.", + "name": "Pause", + "params": [], + "returnDescription": "A flag that indicates whether the slide show was successfully set to pause mode.", + "returnType": "Boolean" + }, + { + "description": "Puts the slide show into play mode starting from the pause point. Setting the player to play mode when it is not in pause mode generates in error.", + "name": "Resume", + "params": [], + "returnDescription": "A flag that indicates whether the slide show was successfully set to pause mode.", + "returnType": "Boolean" + }, + { + "description": "Sets the border color used as the background around the slide.", + "name": "SetBorderColor", + "params": [ + { + "default": null, + "description": "The HTML hex color value to be used as the border.", + "isRequired": true, + "name": "color", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Set the content to be played by the slide show.", + "name": "SetContentList", + "params": [ + { + "default": null, + "description": "An [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roAssociativeArrays](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArrays\") ([Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \"Content Meta-Data\") objects) representing the information for each title to be displayed on screen", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the mode for displaying slideshow images. This allows images to be either scaled to completely fill the screen (scale-to-fill) or scaled to fit inside the screen (scale-to-fit) while maintaining aspect ratio.", + "name": "SetDisplayMode", + "params": [ + { + "default": null, + "description": "The display mode, which may be one of the following values: * scale-to-fill – scale image to completely fill the rectangle of the bounding frame (Default) * scale-to-fit – scale image to fit horizontally or vertically as appropriate while still maintaining aspect ratio. Note that scale-to-fit may result in pillar-box or letter-box display of images. * zoom-to-fill – scales and crops image to maintain aspect ratio and completely fill the rectangle of the bounding frame. * photo-fit – Uses several methods to fit the image with a different aspect ratio to the screen. First, it will asymmetrically scale up to a maximum of 5%. Second, for landscape images, if vertical cropping is necessary, it will remove two lines off the bottom for every one line off the top up to a maximum of 30% of the image. For all images, if horizontal cropping is necessary it will crop an equal amount from both sides.", + "isRequired": true, + "name": "displayMode", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Makes the slideshow loop through all the slides.", + "name": "SetLoop", + "params": [ + { + "default": null, + "description": "A flag specifying whether to loop through all the slides.", + "isRequired": true, + "name": "loop", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Set the maximum scale factor for scale-to-fill, zoom-to-fill, and photo-fit modes.", + "name": "SetMaxUpscale", + "params": [ + { + "default": null, + "description": "The maximum scale factor to be used.", + "isRequired": true, + "name": "maxUpscale", + "type": "Float" + } + ], + "returnType": "Void" + }, + { + "description": "Makes the SlideShow object queue a specific slide up as the next slide.", + "name": "SetNext", + "params": [ + { + "default": null, + "description": "The zero-based index of the item in the content list.", + "isRequired": true, + "name": "item", + "type": "Integer" + }, + { + "default": null, + "description": "Forces an immediate update to the slideshow.", + "isRequired": true, + "name": "isImmediate", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the number of seconds that each slide is displayed.", + "name": "SetPeriod", + "params": [ + { + "default": null, + "description": "The number of seconds that each slide is displayed.", + "isRequired": true, + "name": "seconds", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Defines the number of milliseconds to display the text overlay for each slide.", + "name": "SetTextOverlayHoldTime", + "params": [ + { + "default": null, + "description": "The number of milliseconds to display the text overlay. If this is set to 0, the overlay is off.", + "isRequired": true, + "name": "milliseconds", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Displays the overlay. This method is OR'd with the overlay hold time; therefore, even if **IsVisible** is false, the overlay is displayed during the slide's overlay hold time.", + "name": "SetTextOverlayIsVisible", + "params": [ + { + "default": null, + "description": "A flag specifying whether the overlay is displayed.", + "isRequired": true, + "name": "IsVisible", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Set the percentage to reduce the image size by to compensate for monitor overscan (for example, 2.5 for 2.5%).", + "name": "SetUnderscan", + "params": [ + { + "default": null, + "description": "The percentage to be used to reduce the image size.", + "isRequired": true, + "name": "percentage", + "type": "Float" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was successfully displayed.", + "returnType": "Boolean" + } + ], + "name": "ifSlideshow", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifslideshow.md" + }, + "ifsocket": { + "implementers": [ + { + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets", + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the roSocketAddress object bound to this socket.", + "name": "GetAddress", + "params": [], + "returnDescription": "roSocketAddress Object.", + "returnType": "Object" + }, + { + "description": "Returns the number of bytes in the receive buffer.", + "name": "GetCountRcvBuf", + "params": [], + "returnDescription": "Number of bytes.", + "returnType": "Integer" + }, + { + "description": "Returns the number of bytes in the send buffer.", + "name": "GetCountSendBuf", + "params": [], + "returnDescription": "Number of bytes.", + "returnType": "Integer" + }, + { + "description": "Returns the roSocketAddress for the remote address of the last message received via the [receive()](#receivedata-as-object-startindex-as-integer-length-as-integer-as-integer) method. This method can also be used to return the remote address on newly accepted sockets.", + "name": "GetReceivedFromAddress", + "params": [], + "returnDescription": "The roSocketAddress for the remote address of the last message received.", + "returnType": "Object" + }, + { + "description": "Returns the roSocketAddress for the remote address of the next message to be sent. This method can also be used to return the remote address on newly accepted sockets.", + "name": "GetSendToAddress", + "params": [], + "returnDescription": "The roSocketAddress for the remote address of the next message to be sent.", + "returnType": "Object" + }, + { + "description": "Reads data from the socket.", + "name": "Receive", + "params": [ + { + "default": null, + "description": "A [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md\"roByteArray\") containing the data to be stored.", + "isRequired": true, + "name": "data", + "type": "Object" + }, + { + "default": null, + "description": "The index of the byte array from which to start reading data.", + "isRequired": true, + "name": "startIndex", + "type": "Integer" + }, + { + "default": null, + "description": "The amount of data to be read from the socket.", + "isRequired": true, + "name": "length", + "type": "Integer" + } + ], + "returnDescription": "The number of bytes read.", + "returnType": "Integer" + }, + { + "name": "ReceiveStr", + "params": [ + { + "default": null, + "description": "The amount of data to be read from the socket.", + "isRequired": true, + "name": "length", + "type": "Integer" + } + ], + "returnDescription": "The received byte length string. If no bytes are received, the string is empty.", + "returnType": "String" + }, + { + "description": "Sends up to length bytes of data to the socket.", + "name": "Send", + "params": [ + { + "default": null, + "description": "A [roByteArray](https://developer.roku.com/docs/references/brightscript/components/robytearray.md\"roByteArray\") containing the data to be sent.", + "isRequired": true, + "name": "data", + "type": "Object" + }, + { + "default": null, + "description": "The index of the byte array from which to start sending data.", + "isRequired": true, + "name": "startIndex", + "type": "Integer" + }, + { + "default": null, + "description": "The amount of data to be sent to the socket.", + "isRequired": true, + "name": "length", + "type": "Integer" + } + ], + "returnDescription": "The number of bytes sent.", + "returnType": "Integer" + }, + { + "description": "Sends the whole string to the socket, if possible.", + "name": "SendStr", + "params": [ + { + "default": null, + "description": "A string containing the data to be sent.", + "isRequired": true, + "name": "data", + "type": "String" + } + ], + "returnDescription": "The number of bytes sent.", + "returnType": "Integer" + }, + { + "description": "Sets the address using a BSD bind() call", + "name": "SetAddress", + "params": [ + { + "default": null, + "description": "An roSocketAddress.", + "isRequired": true, + "name": "sockAddr", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the address was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the remote address for next message to be sent.", + "name": "SetSendToAddress", + "params": [ + { + "default": null, + "description": "An roSocketAddress.", + "isRequired": true, + "name": "sockAddr", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the address was successfully stored as the first half of underlying BSD sendto() call.", + "returnType": "Boolean" + }, + { + "description": "Indicates whether the last operation was successful.", + "name": "Status", + "params": [], + "returnDescription": "This method returns 0 if the last operation was successful or an error number if it failed.", + "returnType": "Integer" + } + ], + "name": "ifSocket", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocket.md" + }, + "ifsocketaddress": { + "implementers": [ + { + "description": "The roSocketAddress is used by the roStreamSocket and roDataGramSocket components for TCP and UDP traffic respectively", + "name": "roSocketAddress", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosocketaddress.md" + } + ], + "methods": [ + { + "name": "GetAddress", + "params": [], + "returnDescription": "The IPV4 address.", + "returnType": "String" + }, + { + "description": "Returns the hostname.", + "name": "GetHostName", + "params": [], + "returnDescription": "The hostname.", + "returnType": "String" + }, + { + "description": "Returns the port number.", + "name": "GetPort", + "params": [], + "returnDescription": "The port number.", + "returnType": "Integer" + }, + { + "description": "Checks whether the component contains a valid IP address.", + "name": "IsAddressValid", + "params": [], + "returnDescription": "A flag indicating whether the component contains a valid IP address.", + "returnType": "Boolean" + }, + { + "description": "Sets the IPV4 address.", + "name": "SetAddress", + "params": [ + { + "default": null, + "description": "The string consists of a hostname, optionally followed by a colon and a decimal port number. The hostname may be either dotted quad (such as \"192.168.1.120\") or a DNS name (such as \"roku.com\"). If a name is given, a DNS lookup is performed to convert it to dotted quad. Use IsAddressValid() to determine the result of the DNS lookup. Example: \"192.168.1.120:8888\" or \"roku.com\".", + "isRequired": true, + "name": "address", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the IPV4 address was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the hostname. The port number is unchanged.", + "name": "SetHostName", + "params": [ + { + "default": null, + "description": "The hostname to be used.", + "isRequired": true, + "name": "hostname", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the hostname was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the port number. The hostname is unchanged.", + "name": "SetPort", + "params": [ + { + "default": null, + "description": "The port number to be used.", + "isRequired": true, + "name": "port", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the port number was successfully set.", + "returnType": "Boolean" + } + ], + "name": "ifSocketAddress", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketaddress.md" + }, + "ifsocketasync": { + "description": "The ifSocketAsync interface provides asynchronous socket features that utilize a full-featured select loop in the Roku OS that communicates to the application using a BrightScript [roMessagePort](https://developer.roku.com/docs/references/brightscript/components/romessageport.md\"roMessagePort\"). This interface is valid on roStreamSocket and roDataGramSocket objects that were assigned a BrightScript port via SetMessagePort().", + "implementers": [ + { + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets", + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Returns a unique identifier that can be compared to the value returned by the [roSocketEvent.getSocketID()](https://developer.roku.com/docs/references/brightscript/events/rosocketevent.mdgetsocketid-as-integer) method to match the underlying socket to receive the event.", + "name": "GetID", + "params": [], + "returnType": "Integer" + }, + { + "description": "Checks whether underlying select determines non-blocking read of OOB data is possible.", + "name": "IsException", + "params": [], + "returnDescription": "A flag indicating whether underlying select determines non-blocking read of OOB data is possible.", + "returnType": "Boolean" + }, + { + "description": "Checks whether underlying select determines non-blocking read is possible.", + "name": "IsReadable", + "params": [], + "returnDescription": "A flag indicating whether underlying select determines non-blocking read is possible.", + "returnType": "Boolean" + }, + { + "description": "Checks whether underlying select determines non-blocking write is possible.", + "name": "IsWritable", + "params": [], + "returnDescription": "A flag indicating whether underlying select determines non-blocking write is possible.", + "returnType": "Boolean" + }, + { + "description": "Enables roSocketEvent events to be sent via the message port when the underlying socket gets an exception or OOB data.", + "name": "NotifyException", + "params": [ + { + "default": null, + "description": "A flag specifying whether roSocketEvent events are to be sent when the underlying socket gets an exception or OOB data.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables roSocketEvent events to be sent via the message port when the underlying socket becomes readable.", + "name": "NotifyReadable", + "params": [ + { + "default": null, + "description": "A flag specifying whether roSocketEvent events are to be sent when the underlying socket becomes readable.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables roSocketEvent events to be sent via the message port when the underlying socket becomes writable.", + "name": "NotifyWritable", + "params": [ + { + "default": null, + "description": "A flag specifying whether roSocketEvent events are to be sent when the underlying socket becomes writable.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + } + ], + "name": "ifSocketAsync", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketasync.md" + }, + "ifsocketcastoption": { + "implementers": [ + { + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets", + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + } + ], + "methods": [ + { + "description": "Drops out of a specific multicast group.", + "name": "DropGroup", + "params": [ + { + "default": null, + "description": "An [roSocketAddress](https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md\"roSocketAddress\") representing the group to leave. IPV4 multicast addresses are in the range of 224.0.0.0 through 239.255.255.255.", + "isRequired": true, + "name": "ipAddress", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Checks whether broadcast messages may be sent or received.", + "name": "GetBroadcast", + "params": [], + "returnDescription": "A flag indicating whether broadcast messages may be sent or received.", + "returnType": "Boolean" + }, + { + "description": "Checks whether multicast messages are enabled for local loopback.", + "name": "GetMulticastLoop", + "params": [], + "returnDescription": "A flag indicating whether multicast messages are enabled for local loopback. If this flag is true, multicast message sent locally are to be received locally.", + "returnType": "Boolean" + }, + { + "description": "Returns the TTL integer value for multicast messages. This is the number of hops a packet is allowed before a router drops the packet.", + "name": "GetMulticastTTL", + "params": [], + "returnDescription": "The multicast messages value.", + "returnType": "Integer" + }, + { + "description": "Joins a specific multicast group.", + "name": "JoinGroup", + "params": [ + { + "default": null, + "description": "An [roSocketAddress](https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md\"roSocketAddress\") representing the group to be joined. IPV4 multicast addresses are in the range of 224.0.0.0 through 239.255.255.255.", + "isRequired": true, + "name": "ipAddress", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Enables broadcast messages to be sent or received.", + "name": "SetBroadcast", + "params": [ + { + "default": null, + "description": "A flag specifying whether broadcast messages may be sent or received.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether this operation succeeded.", + "returnType": "Boolean" + }, + { + "description": "Enables local loopback of multicast messages.", + "name": "SetMulticastLoop", + "params": [ + { + "default": null, + "description": "A flag specifying whether local loopback of multicast messages; otherwise do not send or receive broadcast messages.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Sets the TTL integer value for multicast messages.", + "name": "SetMulticastTTL", + "params": [ + { + "default": null, + "description": "The number of hops a packet is allowed before a router drops the packet", + "isRequired": true, + "name": "ttl", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the TTL value was successfully set.", + "returnType": "Boolean" + } + ], + "name": "ifSocketCastOption", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketcastoption.md" + }, + "ifsocketconnection": { + "implementers": [ + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Accepts incoming requests.", + "name": "Accept", + "params": [], + "returnDescription": "An roStreamSocket if the connection is pending; invalid otherwise. Use status to distinguish among success (eSuccess() or isConnected()), not ready (eOK()), and error.", + "returnType": "Object" + }, + { + "description": "Establishes a connection.", + "name": "Connect", + "params": [], + "returnDescription": "A flag indicating whether a socket connection has successfully been created. The connection might still not be complete if the socket is non-blocking", + "returnType": "Boolean" + }, + { + "description": "Checks whether a [connect](#connect-as-boolean) or [accept](#accept-as-object) function has been completed on this socket.", + "name": "IsConnected", + "params": [], + "returnDescription": "A flag indicating whether a connection has been established or accepted on this socket.", + "returnType": "Boolean" + }, + { + "name": "IsListening", + "params": [], + "returnDescription": "A flag indicating whether the [listen()](#listenbacklog-as-integer-as-boolean) method has been successfully called on this socket.", + "returnType": "Boolean" + }, + { + "description": "Puts the socket into the listen state.", + "name": "Listen", + "params": [ + { + "default": null, + "description": "The limit for the queue of incoming connections", + "isRequired": true, + "name": "backlog", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether listening can be done (generally, if bound address is valid).", + "returnType": "Boolean" + } + ], + "name": "ifSocketConnection", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnection.md" + }, + "ifsocketconnectionoption": { + "implementers": [ + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Checks whether keep alive is set. If keep alive is set, occasional no-data packets are sent to keep the connection alive.", + "name": "GetKeepAlive", + "params": [], + "returnDescription": "A flag indicating whether keep alive is set.", + "returnType": "Boolean" + }, + { + "description": "Returns the max time in seconds that the socket close() blocks to allow send data to be flushed in synchronous mode.", + "name": "GetLinger", + "params": [], + "returnDescription": "The max time in seconds.", + "returnType": "Integer" + }, + { + "description": "Returns the max TCP segment size.", + "name": "GetMaxSeg", + "params": [], + "returnDescription": "The segment size.", + "returnType": "Integer" + }, + { + "description": "Checks whether the no delay property is enabled on the socket. This means that data is sent as soon as it is available rather than once there is enough data to fill a segment.", + "name": "GetNoDelay", + "params": [], + "returnDescription": "A flag indicating whether the no delay property is enabled.", + "returnType": "Boolean" + }, + { + "description": "Sends no-data packets to keep the connection alive.", + "name": "SetKeepAlive", + "params": [ + { + "default": null, + "description": "A flag specifying whether keep alive is enabled.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether keep alive was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the max time in seconds that the socket close() blocks to allow send data to be flushed in synchronous mode.", + "name": "SetLinger", + "params": [ + { + "default": null, + "description": "The max time.", + "isRequired": true, + "name": "time", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the linger was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the max TCP segment size.", + "name": "SetMaxSeg", + "params": [ + { + "default": null, + "description": "The max TCP segment size.", + "isRequired": true, + "name": "time", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the max TCP segment size was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Enables the no delay property on the socket. This means that data is sent as soon as it is available rather than once there is enough data to fill a segment.", + "name": "SetNoDelay", + "params": [ + { + "default": null, + "description": "A flag specifying whether the no delay property is enabled.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the no delay property was successfully set.", + "returnType": "Boolean" + } + ], + "name": "ifSocketConnectionoption", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnectionoption.md" + }, + "ifsocketconnectionstatus": { + "implementers": [ + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Checks whether a connection aborted error (ECONNABORTED) has occurred.", + "name": "eConnAborted", + "params": [], + "returnDescription": "A flag indicating whether an ECONNABORTED error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether a connection refused (ECONNREFUSED) has occurred.", + "name": "eConnRefused", + "params": [], + "returnDescription": "A flag indicating whether an ECONNREFUSED error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether a connection reset error (ECONNRESET) has occurred.", + "name": "eConnReset", + "params": [], + "returnDescription": "A flag indicating whether an ECONNRESET error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an is connected error (EISCONN) has occurred.", + "name": "eIsConn", + "params": [], + "returnDescription": "A flag indicating whether an EISCONN error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether a not connected error (ENOTCONN) has occurred.", + "name": "eNotConn", + "params": [], + "returnDescription": "A flag indicating whether an ENOTCONN error has occurred.", + "returnType": "Boolean" + } + ], + "name": "ifSocketConnectionstatus", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketconnectionstatus.md" + }, + "ifsocketoption": { + "implementers": [ + { + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets", + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Checks whether Out Of Bounds (OOB) data is read inline with regular data.", + "name": "GetOOBInline", + "params": [], + "returnDescription": "A flag indicating whether OOB data is read inline with regular data.", + "returnType": "Boolean" + }, + { + "description": "Returns the current receive buffer size.", + "name": "GetRcvBuf", + "params": [], + "returnDescription": "The buffer size.", + "returnType": "Integer" + }, + { + "description": "Returns the current receive timeout.", + "name": "GetReceiveTimeout", + "params": [], + "returnDescription": "The number of seconds for the receive timeout.", + "returnType": "Integer" + }, + { + "description": "Checks whether an address that has been previously assigned can be immediately reassigned.", + "name": "GetReuseAddr", + "params": [], + "returnDescription": "A flag indicating whether the previously assigned address can be reassigned.", + "returnType": "Boolean" + }, + { + "description": "Returns the current send buffer size.", + "name": "GetSendBuf", + "params": [], + "returnDescription": "The buffer size.", + "returnType": "Integer" + }, + { + "description": "Returns the current send timeout.", + "name": "GetSendTimeout", + "params": [], + "returnDescription": "The number of seconds for the send timeout.", + "returnType": "Integer" + }, + { + "description": "Returns the TTL (Time To Live) value for all IP packets on the socket.", + "name": "GetTTL", + "params": [], + "returnDescription": "The TTL value.", + "returnType": "Integer" + }, + { + "description": "Enables Out Of Bounds (OOB) data to be read inline with regular data.", + "name": "SetOOBInline", + "params": [ + { + "default": null, + "description": "A flag specifying whether OOB data is read inline.", + "isRequired": true, + "name": "inline", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the OOB inline data feature was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the current receive buffer size.", + "name": "SetRcvBuf", + "params": [ + { + "default": null, + "description": "The receive buffer size to be used.", + "isRequired": true, + "name": "size", + "type": "Integer" + } + ], + "returnType": "Boolean" + }, + { + "description": "Sets the current receive timeout (in seconds).", + "name": "SetReceiveTimeout", + "params": [ + { + "default": null, + "description": "The number of seconds for the receive timeout.", + "isRequired": true, + "name": "timeout", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the receive timeout was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Enables a previously assigned address to be immediately reassigned.", + "name": "SetReuseAddr", + "params": [ + { + "default": null, + "description": "A flag specifying whether the address can be reused.", + "isRequired": true, + "name": "reuse", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the reuse address feature was successfully set.", + "returnType": "Dynamic" + }, + { + "description": "Sets the current send buffer size.", + "name": "SetSendBuf", + "params": [ + { + "default": null, + "description": "The send buffer size to be used.", + "isRequired": true, + "name": "size", + "type": "Integer" + } + ], + "returnType": "Boolean" + }, + { + "description": "Sets the current send timeout (in seconds).", + "name": "SetSendTimeout", + "params": [ + { + "default": null, + "description": "The number of seconds for the send timeout.", + "isRequired": true, + "name": "timeout", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the send timeout was successfully set.", + "returnType": "Boolean" + }, + { + "description": "Sets the TTL value for all IP packets on the socket.", + "name": "SetTTL", + "params": [ + { + "default": null, + "description": "The TTL value to be used for IP packets on the socket.", + "isRequired": true, + "name": "ttl", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the TTL was successfully set.", + "returnType": "Boolean" + } + ], + "name": "ifSocketOption", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketoption.md" + }, + "ifsocketstatus": { + "implementers": [ + { + "description": "The roDataGramSocket component enables Brightscript apps to send and receive UDP packets", + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + { + "description": "The roStreamSocket component enables BrightScript apps to accept and connect to TCP streams as well as send and receive data with them", + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "description": "Checks whether an EAGAIN error has occurred.", + "name": "eAgain", + "params": [], + "returnDescription": "A flag indicating whether an EAGAIN error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EALREADY error has occurred.", + "name": "eAlready", + "params": [], + "returnDescription": "A flag indicating whether an EALREADY error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EBADADDR error has occurred.", + "name": "eBadAddr", + "params": [], + "returnDescription": "A flag indicating whether an EBADADDR error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EDESTADDRREQ error has occurred.", + "name": "eDestAddrReq", + "params": [], + "returnDescription": "A flag indicating whether an EDESTADDRREQ error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EHOSTUNREACH error has occurred.", + "name": "eHostUnreach", + "params": [], + "returnDescription": "A flag indicating whether an EHOSTUNREACH error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EINPROGRESS error has occurred.", + "name": "eInProgress", + "params": [], + "returnDescription": "A flag indicating whether an EINPROGRESS error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether an EINVALID error has occurred.", + "name": "eInvalid", + "params": [], + "returnDescription": "A flag indicating whether an EINVALID error has occurred.", + "returnType": "Boolean" + }, + { + "description": "Checks whether there is no hard error, but possibly one of the following async conditions: EAGAIN, EALREADY, EINPROGRESS, EWOULDBLOCK.", + "name": "eOK", + "params": [], + "returnDescription": "A flag indicating whether an EOK error has occurred.", + "returnType": "Boolean" + }, + { + "name": "eSuccess", + "params": [], + "returnType": "Boolean" + }, + { + "description": "Checks whether an EWOULDBLOCK error has occurred.", + "name": "eWouldBlock", + "params": [], + "returnDescription": "A flag indicating whether an EWOULDBLOCK error has occurred.", + "returnType": "Boolean" + } + ], + "name": "ifSocketStatus", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsocketstatus.md" + }, + "ifsourceidentity": { + "implementers": [ + { + "description": "The roChannelStore sends an roChannelStoreEvent in response to a call to any of several Get\\* methods in ifChannelStore", + "name": "roChannelStoreEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rochannelstoreevent.md" + }, + { + "description": "The roUrlTransfer component sends the roUrlEvent", + "name": "roUrlEvent", + "url": "https://developer.roku.com/docs/references/brightscript/events/rourlevent.md" + } + ], + "methods": [ + { + "description": "Returns the ID currently associated with this source (event generating) or event object", + "name": "GetSourceIdentity", + "params": [], + "returnDescription": "The ID value of the source or event object.", + "returnType": "Integer" + } + ], + "name": "ifSourceIdentity", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsourceidentity.md" + }, + "ifspringboardscreen": { + "deprecatedDescription": "This interface is deprecated.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Springboard Screen shows detailed information about an individual piece of content and provides options for actions that may be taken on that content", + "name": "roSpringboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rospringboardscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "description": "Adds a button to the screen. The buttons are displayed in a standard location on the screen and appear in the order added. When the button is pressed, the script will receive an event from the application containing the ID of the button pressed and allowing the script to perform the desired action for that case.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "buttonID", + "type": "Integer" + }, + { + "default": null, + "description": "The title of the button.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "name": "AddRatingButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "buttonID", + "type": "Integer" + }, + { + "default": null, + "description": "A value between 1-100 that represents the number of stars that the user rated the content. A value of 1-20 corresponds to 1 star, 21-40 corresponds to 2 stars, etc. The userRating takes precedence and determines the color of the buttons if set.", + "isRequired": true, + "name": "userRating", + "type": "Integer" + }, + { + "default": null, + "description": "The total average rating from all users. Half-stars maybe displayed for this rating.", + "isRequired": true, + "name": "aggregateRating", + "type": "Integer" + } + ], + "returnType": "Boolean" + }, + { + "description": "Adds a thumbs up/down button to the screen. This is a special type of button that is displayed in the standard location on the screen and appears in the order added sequenced with other buttons. When the button is pressed, the script will receive an event from the application indicating containing the ID of the button pressed and allowing the script to perform the desired action for that case.", + "name": "AddThumbsUpDownButton", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "buttonID", + "type": "Integer" + }, + { + "default": null, + "description": "The rating, which determines how the button appears on the screen. This may be one of the following values: * \\-1 = thumbs down * 0 = no rating * 1 = thumbs up", + "isRequired": true, + "name": "thumbRating", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds a thumbs up/down button with tip text to the screen. This is a special type of button that is displayed in the standard location on the screen and appears in the order added sequenced with other buttons. When the button is pressed, the script will receive an event from the application indicating containing the ID of the button pressed and allowing the script to perform the desired action for that case.", + "name": "AddThumbsUpDownButtonWithTips", + "params": [ + { + "default": null, + "description": "The ID used to uniquely identify the button instance.", + "isRequired": true, + "name": "buttonID", + "type": "Integer" + }, + { + "default": null, + "description": "The rating, which determines how the button appears on the screen. This may be one of the following values: * \\-1 = thumbs down * 0 = no rating * 1 = thumbs up", + "isRequired": true, + "name": "thumbRating", + "type": "Integer" + }, + { + "default": null, + "description": "An array of strings (size 2) which lets the script override the default tip text \\[\"didn't like it\", \"liked it\"\\]", + "isRequired": true, + "name": "tipText", + "type": "Object" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Enables the fast forward remote event to be sent to the script (it is enabled by default).", + "name": "AllowNavFastForward", + "params": [ + { + "default": null, + "description": "True = Pressing FAST FORWARD sends an roSpringboardScreenEvent /isRemoteKeyPressed() event to the screen's message port. False = Pressing FAST FORWARD plays a deadend sound.", + "isRequired": true, + "name": "allow", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables navigating LEFT on the springboard screen (it is enabled by default).", + "name": "AllowNavLeft", + "params": [ + { + "default": null, + "description": "True = Pressing LEFT sends an roSpringboardScreenEvent /isRemoteKeyPressed() event to the screen's message port. False = Pressing LEFT plays a deadend sound.", + "isRequired": true, + "name": "allow", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables the rewind remote event to be sent to the script (it is enabled by default).", + "name": "AllowNavRewind", + "params": [ + { + "default": null, + "description": "True = Pressing REWIND sends an roSpringboardScreenEvent /isRemoteKeyPressed() event to the screen's message port. False = Pressing REWIND plays a deadend sound.", + "isRequired": true, + "name": "allow", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables navigating RIGHT on the springboard screen (it is enabled by default).", + "name": "AllowNavRight", + "params": [ + { + "default": null, + "description": "True = Pressing RIGHT sends an roSpringboardScreenEvent /isRemoteKeyPressed() event to the screen's message port. False = Pressing RIGHT plays a deadend sound.", + "isRequired": true, + "name": "allow", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Refreshes the display. When adding multiple buttons to the springboard dynamically, you may want to defer screen updates temporarily to avoid flashing. For example, call AllowUpdates(false), add several buttons, and then call AllowUpdates(true) to refresh the display.", + "name": "AllowUpdates", + "params": [ + { + "default": null, + "description": "A flag specifying whether to refresh the display.", + "isRequired": true, + "name": "allow", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Clears all of the buttons from the screen and resets the array of buttons back to default with no buttons set.", + "name": "ClearButtons", + "params": [], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "name": "CountButtons", + "params": [], + "returnType": "Integer" + }, + { + "description": "Allows the screen to pre-fetch the poster images before the screen is displayed as a display optimization technique. This is useful when doing left-right navigation between springboard screens. The pre-fetch is done, loading the image cache and then the screen is displayed.", + "name": "PrefetchPoster", + "params": [ + { + "default": null, + "description": "The URL of the standard definition poster image.", + "isRequired": true, + "name": "sdPosterURL", + "type": "String" + }, + { + "default": null, + "description": "The URL of the high definition poster image.", + "isRequired": true, + "name": "hdPosterURL", + "type": "String" + } + ], + "returnType": "Voidmet" + }, + { + "name": "SetAdDisplayMode", + "params": [ + { + "default": null, + "description": "The scale mode to be used, which may be one of the following values: * scale-to-fill: Scales image to completely fill the rectangle of the bounding frame (default)
Valid logTypesDescription\"http.connect\"Sent whenever a successful HTTP connection is made. This means that the server responded to the HTTP request with a success (2xx) status code. However, this does not necessarily mean that all of the body of the request has been received successfully\"http.error\"Sent whenever an error occurs while executing an HTTP request. This may be sent during the time of the initial connection for two possible reasons:
  • because the server responded with an error code, or
  • data is being read from the body after the initial connection takes place
\"bandwidth.minute\"Sent every minute to report the current measured bandwidth“http.complete”
Property/KeyTypeDescription
LogTypeStringWhen enabled, the “http.complete” events will be sent to Roku after an http transfer is completed for adaptive streams. This event consolidates information related to:
  • a cURL transfer such as DNS look up time,
  • connection latency,
  • transfer speed,
  • and number of bytes.
DateTimeroDateTimeThe GMT time of the event, with a resolution of one millisecond
UrlStringThe URL that was requested
OrigUrlStringThe original URL. If the original URL was redirected, then Url represents the new redirected URL and OrigURL the original. OrigURL is included so that it's easy to correlate between events and URLs passed to components
MethodStringThe HTTP method. \"GET\", \"POST\", or \"HEAD\"
StatusStringFor LogType “http.complete”, this will be “ok”
TargetIpStringThe IP address of the target server
HttpCodeIntegerThe HTTP response code if available
ContentTypeStringContent type or MIME type
DNSLookupTimeDoubleDNS name resolution time in seconds with double precision
ConnectTimeDoubleTime taken to connect to the server (seconds)
FirstByteTimeDoubleTime taken to receive the first byte from the server (seconds)
TransferTimeDoubleTotal data transfer time (seconds)
DownloadSpeedDoubleTransfer download speed in bytes per second
BytesDownloadedIntegerNumber of bytes downloaded from the server
UploadSpeedDoubleTransfer upload speed in bytes per second
BytesUploadedIntegerNumber of bytes uploaded to the server
", + "isRequired": true, + "name": "logType", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifSystemLog", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifsystemlog.md" + }, + "iftextscreen": { + "description": "> This component is no longer updated and will be deprecated on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "roTextScreen provides a way of displaying large amounts of scrollable text", + "name": "roTextScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotextscreen.md" + } + ], + "methods": [ + { + "description": "Adds a button to the screen, which is displayed in a standard location on the screen and appears in the order added. When the button is pressed, the script will receive an event from the application containing the ID of the button pressed and allowing the script to perform the desired action for that case.", + "name": "AddButton", + "params": [ + { + "default": null, + "description": "The unique ID to be assigned to the button instance", + "isRequired": true, + "name": "id", + "type": "Integer" + }, + { + "default": null, + "description": "The text to be used to identify the button instance", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the button was successfully added.", + "returnType": "Boolean" + }, + { + "description": "Adds text to the main text of the screen. The text is appended to the end of any existing text", + "name": "AddText", + "params": [ + { + "default": null, + "description": "The text to be appended to the main text on the screen", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Closes the screen and deletes the associated object. This is useful for avoiding screen flicker when the display order of your screens does not resemble a stack.", + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Breadcrumbs allow the application to display a two-part navigational title that shows the previous and current locations in the application hierarchy (e.g. TV – Friends).", + "name": "SetBreadcrumbText", + "params": [ + { + "default": null, + "description": "The previous location in the application hierarchy.", + "isRequired": true, + "name": "location1", + "type": "String" + }, + { + "default": null, + "description": "The current location in the application hierarchy.", + "isRequired": true, + "name": "location2", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Adds a string of bold, high visibility text to the screen as a header to appear above the main body of text. Multiple calls to this method are ignored", + "name": "SetHeaderText", + "params": [ + { + "default": null, + "description": "The text for the header that will be displayed above the main text body", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the main text of the screen. The roTextScreen handles all text formatting and justification. Multiple calls to this method are ignored", + "name": "SetText", + "params": [ + { + "default": null, + "description": "The text for the main screen.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the title for the screen, which appears in the overhang", + "name": "SetTitle", + "params": [ + { + "default": null, + "description": "The text to be displayed in the overhang.", + "isRequired": true, + "name": "title", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Displays or refreshes the screen after initial creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was successfully refreshed.", + "returnType": "Boolean" + } + ], + "name": "ifTextScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftextscreen.md" + }, + "iftexttospeech": { + "description": "> Please note this component is only available on the following devices: Roku Streaming Stick (3600X), Roku Express (3700X) and Express+ (3710X), Roku Premiere (4620X) and Premiere+ (4630X), Roku Ultra (4640X), and any Roku TV running Roku OS version 7.2 and later.", + "implementers": [ + { + "description": "The roTextToSpeech component provides text to speech capabilities to applications", + "name": "roTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md" + } + ], + "methods": [ + { + "description": "Interrupts and stops any current text to speech spoken string, to be used when the channel does not want to the text to speech to continue.", + "name": "Flush", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns an array containing the current list of languages available for text-to-speech.", + "name": "GetAvailableLanguages", + "params": [], + "returnDescription": "A list of languages.", + "returnType": "Object" + }, + { + "description": "Returns an array containing the current list of voices available for text-to-speech.", + "name": "GetAvailableVoices", + "params": [], + "returnDescription": "A list of voices.", + "returnType": "Object" + }, + { + "description": "Returns the name of the currently-selected text-to-speech language.", + "name": "GetLanguage", + "params": [], + "returnDescription": "The language name.", + "returnType": "String" + }, + { + "description": "Returns the pitch at which text is spoken. The possible values range from -60 to +60.", + "name": "GetPitch", + "params": [], + "returnDescription": "The pitch.", + "returnType": "Integer" + }, + { + "description": "Returns the rate at which text is spoken. The value ranges from -40 to 200 with a default value of 0.", + "name": "GetRate", + "params": [], + "returnDescription": "The rate.", + "returnType": "Integer" + }, + { + "description": "Returns the currently-selected voice.", + "name": "GetVoice", + "params": [], + "returnDescription": "The selected voice.", + "returnType": "String" + }, + { + "description": "Returns the volume at which text is spoken. The value ranges from 0 for muted to 1000 for the highest volume. The default value is 1000.", + "name": "GetVolume", + "params": [], + "returnDescription": "The volume.", + "returnType": "Integer" + }, + { + "description": "Checks whether text-to-speech is enabled. Text-to-speech may be enabled or disabled for various technical reasons (for example, on some platforms, text-to-speech may only be enabled once in connected mode). This is not affected by the state of any of its clients. In particular, it does not depend on whether a CVAA compliant accessibility feature is enabled or not.", + "name": "IsEnabled", + "params": [], + "returnDescription": "A flag indicating whether text-to-speech is enabled.", + "returnType": "Boolean" + }, + { + "description": "Returns an ID for the spoken string to notify observer callbacks about a specific spoken string.", + "name": "Say", + "params": [ + { + "default": null, + "description": "The UTF8 text to be spoken.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnDescription": "The ID for the spoken string.", + "returnType": "Integer" + }, + { + "description": "Sets the language specified by `name` for text to speech, from one of the available languages returned by the [GetAvailableLanguages()](#getavailablelanguages-as-object) method.", + "name": "SetLanguage", + "params": [ + { + "default": null, + "description": "The text-to-speech language to be used.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the pitch at which text is spoken.", + "name": "SetPitch", + "params": [ + { + "default": null, + "description": "The pitch at which text is to be spoken. The possible values range from -60 to +60.", + "isRequired": true, + "name": "pitch", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the rate at which text is spoken.", + "name": "SetRate", + "params": [ + { + "default": null, + "description": "The rate at which text is to be spoken. The possible values range from -40 to 200.", + "isRequired": true, + "name": "rate", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the voice specified by name for text to speech, from one of the available voices returned by the [GetAvailableVoices()](#getavailablevoices-as-object) method.", + "name": "SetVoice", + "params": [ + { + "default": null, + "description": "The available text-to-speech voice to be used", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the volume at which text is spoken.", + "name": "SetVolume", + "params": [ + { + "default": null, + "description": "The volume at which text is spoken. The value ranges from 0 for muted to 1000 for the highest volume. The default value is 1000.", + "isRequired": true, + "name": "volume", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Causes text to speech to continue to suppress any application background sound for the amount of time specified by `duration`.", + "name": "Silence", + "params": [ + { + "default": null, + "description": "The amount of time to suppress application background sound", + "isRequired": true, + "name": "duration", + "type": "Integer" + } + ], + "returnDescription": "The duration for the speech suppression.", + "returnType": "Integer" + } + ], + "name": "ifTextToSpeech", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexttospeech.md" + }, + "iftexturemanager": { + "implementers": [ + { + "description": "The Texture Manager provides a set of API's for managing an roBitmap cache", + "name": "roTextureManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturemanager.md" + } + ], + "methods": [ + { + "description": "Cancels the request specified by req, which should be an roTextureRequest previously passed to the [RequestTexture()](#requesttexturereq-as-object-as-void) method.", + "name": "CancelRequest", + "params": [ + { + "default": null, + "description": "The previoulsy passed roTextureRequest to be cancelled.", + "isRequired": true, + "name": "req", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Removes all bitmaps from the roTextureManager.", + "name": "Cleanup", + "params": [], + "returnType": "Void" + }, + { + "description": "Makes a request for an roBitmap with the attributes specified by the roTextureRequest. The roTextureManager will pass an roTextureRequestEvent to the message port when completed.", + "name": "RequestTexture", + "params": [ + { + "default": null, + "description": "The roTextureRequest", + "isRequired": true, + "name": "req", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Removes a bitmap from the roTextureManager with the specified URL.", + "name": "UnloadBitmap", + "params": [ + { + "default": null, + "description": "The URL of the bitmap to be removed from the roTextureManager", + "isRequired": true, + "name": "url", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifTextureManager", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexturemanager.md" + }, + "iftexturerequest": { + "implementers": [ + { + "description": "An roTextureRequest is used to make requests to the roTextureManager", + "name": "roTextureRequest", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturerequest.md" + } + ], + "methods": [ + { + "description": "Returns a unique id for the request.", + "name": "GetId", + "params": [], + "returnType": "Integer" + }, + { + "description": "Returns the state of the request.", + "name": "GetState", + "params": [], + "returnDescription": "The state value, which may be one of the following:", + "returnType": "Integer" + }, + { + "description": "Sets the request to be either asynchronous (true) or synchronous (false). The default is asynchronous", + "name": "SetAsync", + "params": [ + { + "default": null, + "description": "The method used to send the request: asynchronous (true) or synchronous (false).", + "isRequired": true, + "name": "async", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the scaling mode to be used.", + "name": "SetScaleMode", + "params": [ + { + "default": null, + "description": "The scaling mode to be used, which may be one of the following values:
ValueScaling mode
0Nearest neighbor (fast). This is the default.
1Bilinear (smooth)
", + "isRequired": true, + "name": "mode", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the desired size of the roBitmap. The default is to return a bitmap in its native size.", + "name": "SetSize", + "params": [ + { + "default": null, + "description": "The width of the roBitmap.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the roBitmap.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnType": "Void" + } + ], + "name": "ifTextureRequest", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftexturerequest.md" + }, + "iftimespan": { + "implementers": [ + { + "description": "The Timespan object provides an interface to a simple timer for tracking the duration of activities", + "name": "roTimespan", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotimespan.md" + } + ], + "methods": [ + { + "description": "Parses the ISO8601 date (e.g. 2008-11-29T14:54:02.171) and returns the number of seconds from now (not the \"Mark\" point) until the specified date/time.", + "name": "GetSecondsToISO8601Date", + "params": [ + { + "default": null, + "isRequired": true, + "name": "date", + "type": "String" + } + ], + "returnDescription": "The number of seconds.", + "returnType": "Integer" + }, + { + "description": "Sets the \"Mark\" point to the current time. The Mark point is also automatically set to the current time when an roTimespan object is created.", + "name": "Mark", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the total number of milliseconds from the \"Mark\" point to the current time.", + "name": "TotalMilliseconds", + "params": [], + "returnDescription": "The number of milliseconds.", + "returnType": "Integer" + }, + { + "description": "Returns the total number of seconds from the \"Mark\" point to the current time.", + "name": "TotalSeconds", + "params": [], + "returnDescription": "The number of seconds.", + "returnType": "Integer" + } + ], + "name": "ifTimeSpan", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftimespan.md" + }, + "iftostr": { + "implementers": [ + { + "description": "Object equivalent for intrinsic type Boolean", + "name": "roBoolean", + "url": "https://developer.roku.com/docs/references/brightscript/components/roboolean.md" + }, + { + "description": "Object equivalent for intrinsic type 'Double'", + "name": "roDouble", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodouble.md" + }, + { + "description": "Object equivalent for intrinsic type 'Float'", + "name": "roFloat", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofloat.md" + }, + { + "description": "Object equivalent for intrinsic type Function", + "name": "roFunction", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofunction.md" + }, + { + "description": "Object equivalent for intrinsic type Integer", + "name": "roInt", + "url": "https://developer.roku.com/docs/references/brightscript/components/roint.md" + }, + { + "description": "Object equivalent for intrinsic type 'Invalid'", + "name": "roInvalid", + "url": "https://developer.roku.com/docs/references/brightscript/components/roinvalid.md" + }, + { + "description": "Object equivalent for intrinsic type LongInteger", + "name": "roLongInteger", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolonginteger.md" + }, + { + "description": "Object equivalent for intrinsic type 'String'", + "name": "roString", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostring.md" + } + ], + "methods": [ + { + "description": "Returns the value as a string.", + "name": "ToStr", + "params": [], + "returnDescription": "The string.", + "returnType": "String" + } + ], + "name": "ifToStr", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/iftostr.md" + }, + "ifurltransfer": { + "implementers": [ + { + "description": "A roUrlTransfer object transfers data to or from remote servers specified by URLs. It can perform mutual authentication with a web server", + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + } + ], + "methods": [ + { + "name": "AsyncCancel", + "params": [], + "returnType": "Boolean" + }, + { + "description": "Starts a transfer without waiting for it to complete, similar to the [AsyncGetToString()](#asyncgettostring-as-boolean) method. However, the response body will be written to a file on the device's filesystem instead of being returned in a String object.", + "name": "AsyncGetToFile", + "params": [ + { + "default": null, + "description": "The file on the Roku device's filesystem to which the response body is to be written", + "isRequired": true, + "name": "filename", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Starts a GET request to a server, but does not wait for the transfer to complete.", + "name": "AsyncGetToString", + "params": [], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Begins an HTTP HEAD request without waiting for it to complete. When the HEAD completes, an [roUrlEvent](https://developer.roku.com/docs/references/brightscript/events/rourlevent.md\"roUrlEvent\") will be sent to the message port associated with the object. If false is returned then the request could not be issued and no events will be delivered.", + "name": "AsyncHead", + "params": [], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Uses the HTTP POST method to send the contents of the specified file to the current URL. When the POST request completes, an [roUrlTransfer](https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md\"roUrlTransfer\") will be sent to the message port associated with the object. If false is returned then the request could not be issued and no events will be delivered.", + "name": "AsyncPostFromFile", + "params": [ + { + "default": null, + "description": "The file containing the POST request to be sent asynchronously", + "isRequired": true, + "name": "filename", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Uses the HTTP POST method to send the contents of the specified file (fromFile) to the current URL. When the POST request completes successfully, an [roUrlTransfer](https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md\"roUrlTransfer\") will be sent to the message port associated with the object. If false is returned then the request could not be issued and no events will be delivered. This function is the same as AsyncPostFromFile, except that the HTTP response is written to the file specified by the toFile parameter.", + "name": "AsyncPostFromFileToFile", + "params": [ + { + "default": null, + "description": "The file containing the POST request to be sent asynchronously", + "isRequired": true, + "name": "fromFile", + "type": "String" + }, + { + "default": null, + "description": "The file on the Roku device's filesystem to which the response body is to be written", + "isRequired": true, + "name": "toFile", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Uses the HTTP POST method to send the supplied string to the current URL. When the POST request completes, an [roUrlTransfer](https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md\"roUrlTransfer\") will be sent to the message port associated with the object. If false is returned then the request could not be issued and no events will be delivered.", + "name": "AsyncPostFromString", + "params": [ + { + "default": null, + "description": "The POST request to be sent asynchronously", + "isRequired": true, + "name": "request", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the request was issued.", + "returnType": "Boolean" + }, + { + "description": "Enables gzip encoding of transfers", + "name": "EnableEncodings", + "params": [ + { + "default": null, + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether this operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Enables a fresh connection using CURLOPT\\_FRESH\\_CONNECT.", + "name": "EnableFreshConnection", + "params": [ + { + "default": null, + "description": "A flag specifying whether to enable fresh connections.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Verifies that the certificate belongs to the host using CURLOPT\\_SSL\\_VERIFYHOST.", + "name": "EnableHostVerification", + "params": [ + { + "default": null, + "description": "A flag specifying whether to verify a certificate belonging to the host.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Verifies that the certificate has a chain of trust up to a valid root certificate using CURLOPT\\_SSL\\_VERIFYPEER.", + "name": "EnablePeerVerification", + "params": [ + { + "default": null, + "description": "A flag specifying whether to verify a certificate has a chain-of-trust up to a valid root certificate", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Enables automatic resumption of `AsyncGetToFile` and `GetToFile` requests", + "name": "EnableResume", + "params": [ + { + "default": null, + "description": "A flag specifying whether to automatically resume `AsyncGetToFile` and `GetToFile` requests", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "URL encodes the specified string per [RFC 3986](https://www.ietf.org/rfc/rfc3986.txt \"RFC 3986\") and return the encoded string", + "name": "Escape", + "params": [ + { + "default": null, + "description": "The string to be URL-encoded", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnDescription": "The URL-encoded string.", + "returnType": "String" + }, + { + "description": "If any of the `roUrlEvent` functions indicate failure then this function may provide more information regarding the failure.", + "name": "GetFailureReason", + "params": [], + "returnDescription": "Failure reason.", + "returnType": "String" + }, + { + "description": "Returns a unique number for this object that can be used to identify whether events originated from this object. The value can be any arbitrary value as assigned by the Roku OS, and should only be used for comparison purposes. For example, the value should not be used as an array index. For use as a look-up key, one option would be to use `GetIdentity().ToStr()` as an associative array key.", + "name": "GetIdentity", + "params": [], + "returnDescription": "A unique number for the object.", + "returnType": "Integer" + }, + { + "description": "Returns the current request method.", + "name": "GetRequest", + "params": [], + "returnDescription": "The request method.", + "returnType": "String" + }, + { + "description": "Connect to the remote service as specified in the URL and write the response body to a file on the Roku device's filesystem. This function does not return until the exchange is complete and may block for a long time. The HTTP response code from the server is returned. It is not possible to access any of the response headers. If this information is required use the [AsyncGetToFile()](#asyncgettofilefilename-as-string-as-boolean) method instead.", + "name": "GetToFile", + "params": [ + { + "default": null, + "description": "The file on the Roku device's filesystem to which the response body is to be written", + "isRequired": true, + "name": "filename", + "type": "String" + } + ], + "returnDescription": "The HTTP response code.", + "returnType": "Integer" + }, + { + "description": "Connects to the remote service as specified in the URL and returns the response body as a string. This function waits for the transfer to complete and it may block for a long time. This calls discards the headers and response codes. If that information is required, use the [AsyncGetToString()](#asyncgettostring-as-boolean) method.", + "name": "GetToString", + "params": [], + "returnDescription": "The response body.", + "returnType": "String" + }, + { + "description": "Returns the current URL.", + "name": "GetUrl", + "params": [], + "returnDescription": "The URL.", + "returnType": "String" + }, + { + "description": "Synchronously performs an HTTP HEAD request and returns an [roUrlTransfer](https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md\"roUrlTransfer\") object.", + "name": "Head", + "params": [], + "returnDescription": "An [roUrlTransfer](/docs/references/brightscript/interfaces/ifurltransfer.md \"roUrlTransfer\") object. If a catastrophic failure occurs (for example, an asynchronous operation is already active), invalid is returned", + "returnType": "Dynamic" + }, + { + "description": "Uses the HTTP POST method to send the contents of the specified file to the current URL. The HTTP response code is returned. Any response body is discarded", + "name": "PostFromFile", + "params": [ + { + "default": null, + "description": "The file containing the POST request to be sent", + "isRequired": true, + "name": "filename", + "type": "String" + } + ], + "returnDescription": "The HTTP response code.", + "returnType": "Integer" + }, + { + "description": "Uses the HTTP POST method to send the supplied string to the current URL. The HTTP response code is returned. Any response body is discarded", + "name": "PostFromString", + "params": [ + { + "default": null, + "description": "The POST request to be sent", + "isRequired": true, + "name": "request", + "type": "String" + } + ], + "returnDescription": "The HTTP response code.", + "returnType": "Integer" + }, + { + "description": "Returns the body of the response even if the HTTP status code indicates that an error occurred.", + "name": "RetainBodyOnError", + "params": [ + { + "default": null, + "description": "A flag specifying whether to return the response body when there is an HTTP error response code.", + "isRequired": true, + "name": "retain", + "type": "Boolean" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "An optional function that enables HTTP/2 support. If version is set to `\"http2\"`, HTTP/2 will be used for all underlying transfers.", + "name": "SetHttpVersion", + "params": [ + { + "default": null, + "description": "The http version to be used (for example, \"http2\" for HTTP/2). `\"AUTO\"` is the default value, which causes the roUrlTransfer connection to auto-negotiate HTTP/1.x or HTTP/2, depending on the agreement reached by client and server.", + "isRequired": true, + "name": "version", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Terminates the transfer automatically if the transfer rate drops below the specified rate (bytes\\_per\\_second) over a specific interval (period\\_in\\_seconds).", + "name": "SetMinimumTransferRate", + "params": [ + { + "default": null, + "isRequired": true, + "name": "bytes" + } + ], + "returnDescription": "A flag indicating whether the operation was successful." + }, + { + "description": "Changes the request method from the normal GET, HEAD or POST to the value passed as a string.", + "name": "SetRequest", + "params": [ + { + "default": null, + "description": "The request method to be used", + "isRequired": true, + "name": "req", + "type": "String" + } + ] + }, + { + "description": "Sets the URL to use for the transfer request.", + "name": "SetUrl", + "params": [ + { + "default": null, + "description": "The URL to be used for the transfer request", + "isRequired": true, + "name": "url", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Enables HTTP authentication using the specified user name and password.", + "name": "SetUserAndPassword", + "params": [ + { + "default": null, + "description": "The user name to be authenticated", + "isRequired": true, + "name": "user", + "type": "String" + }, + { + "default": null, + "description": "The password to be authenticated", + "isRequired": true, + "name": "password", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Decodes the specified string per [RFC 3986](https://www.ietf.org/rfc/rfc3986.txt \"RFC 3986\") and returns the unencoded string.", + "name": "Unescape", + "params": [ + { + "default": null, + "description": "The string to be URL-decoded", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnDescription": "The decoded string.", + "returnType": "String" + }, + { + "deprecatedDescription": "This method is deprecated. Use the [Escape()](/docs/references/brightscript/interfaces/ifurltransfer.md#escapetext-as-string-as-string) method.\n", + "description": "URL encodes the specified string per RFC 3986 and return the encoded string", + "isDeprecated": true, + "name": "UrlEncode", + "params": [ + { + "default": null, + "isRequired": true, + "name": "url", + "type": "String" + } + ], + "returnDescription": "The encoded string.", + "returnType": "String" + } + ], + "name": "ifUrlTransfer", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifurltransfer.md" + }, + "ifvideoplayer": { + "implementers": [ + { + "description": "The roVideoPlayer component implements a video player with more programmatic control, but less user control than the roVideoScreen component", + "name": "roVideoPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoplayer.md" + } + ], + "methods": [ + { + "description": "Adds a new [Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") item to the end of the content list for the roVideoPlayer. roVideoPlayer playback buffers on each Content item transition.", + "name": "AddContent", + "params": [ + { + "default": null, + "description": "The [content metadata](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") item to be added to the content list.", + "isRequired": true, + "name": "contentItem", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Changes the currently playing audio track. For content with multiple audio tracks, the current track can be selected programmatically using this function.", + "name": "ChangeAudioTrack", + "params": [ + { + "default": null, + "description": "The audio track identifier returned by GetAudioTracks().", + "isRequired": true, + "name": "trackID", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Clears all content from the roVideoPlayer. If the player is currently playing, it stops. Next, the current player position is reset so the next time the [Play()](#play-as-boolean) method is called, playback starts at the first item of the content list (unless the [Seek()](#seekoffsetms-as-integer-as-boolean) method is called prior to Play()).", + "name": "ClearContent", + "params": [], + "returnType": "Void" + }, + { + "description": "Returns the audio tracks contained in the current stream.", + "name": "GetAudioTracks", + "params": [], + "returnDescription": "An roArray, where each element in the array represents a single audio track that contains the following attributes: ${getaudiotracksvalues}", + "returnType": "Object" + }, + { + "description": "This method returns the [roCaptionRenderer](https://developer.roku.com/docs/references/brightscript/components/rocaptionrenderer.md\"roCaptionRenderer\") instance associated with this roVideoPlayer.", + "name": "GetCaptionRenderer", + "params": [], + "returnDescription": "The [roCaptionRenderer](/docs/references/brightscript/components/rocaptionrenderer.md \"roCaptionRenderer\") instance associated with this roVideoPlayer.", + "returnType": "Object" + }, + { + "description": "Returns the duration of the video, in seconds. This information may not be available until after the video starts playing.", + "name": "GetPlaybackDuration", + "params": [], + "returnDescription": "The duration of the video. A value of 0 is returned if the duration is unknown.", + "returnType": "Integer" + }, + { + "description": "Puts the roVideoPlayer object into pause mode. If the player is already in pause mode, this will generate an error.", + "name": "Pause", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Puts the roVideoPlayer object into play mode starting at the beginning of the content list. This will stop any currently playing Content List.", + "name": "Play", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Begins downloading and buffering of a video that may be selected by a user. This method can be used to reduce buffering delays after a user has selected a video for playback. It is typically called when the user is in the roSpringboardScreen (or equivalent), anticipating that the user will select a video on the springboard screen for download.", + "name": "PreBuffer", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Puts the roVideoPlayer object into play mode starting from the pause point. This method must be called when the roVideoPlayer object is in pause mode; otherwise, it will generate an error.", + "name": "Resume", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Sets the start point of playback for the current video to a specific offset.", + "name": "Seek", + "params": [ + { + "default": null, + "description": "The number of milliseconds to offset the playback of the current content item.", + "isRequired": true, + "name": "offsetMs", + "type": "Integer" + } + ], + "returnType": "Boolean" + }, + { + "description": "Sets CGMS (Copy Guard Management System) on analog outputs to the desired level.", + "name": "SetCGMS", + "params": [ + { + "default": null, + "description": "The level to which CGMS is set. This may be one of the following values: * 0 - No Copy Restriction * 1 - Copy No More * 2 - Copy Once Allowed * 3 – No Copying Permitted", + "isRequired": true, + "name": "level", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the content to be played by the roVideoPlayer.", + "name": "SetContentList", + "params": [ + { + "default": null, + "description": "An [roArray](https://developer.roku.com/docs/references/brightscript/components/roarray.md\"roArray\") of [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") ([Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\") objects) representing the information for each stream to be played. If the player is currently playing the player will be stopped. Next, the current player position is reset so the next time Play() is called, playback will start at the first item of the content list (unless Seek() is called prior to Play()). roVideoPlayer prefetches the next item in the content list while the current item is playing. Given sufficient network throughput, there is no rebuffering when the player switches to the next item in the list. To signal the content transition, the player sends an isRequestSucceeded notification with the old content index and isListItemSelected notification with the new content index.", + "isRequired": true, + "name": "contentList", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the target display window for the video.", + "name": "SetDestinationRect", + "params": [ + { + "default": null, + "description": "The parameters of the target display window, which include the x and y coordinates, width, and height {x:Integer, y:Integer, w:Integer, h:Integer} The default value is: {x:0, y:0, w:0, h:0}, which is full screen.", + "isRequired": true, + "name": "rect", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the target display window for the video. This is similar to the [SetDestinationRect()](#setdestinationrectrect-as-object-as-void) function except that the values are specified as separate parameters.", + "name": "SetDestinationRect", + "params": [ + { + "default": null, + "description": "The x coordinate of the target display window.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y coordinate of the target display window.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the target display window.", + "isRequired": true, + "name": "w", + "type": "Integer" + }, + { + "default": null, + "description": "The height coordinate of the target display window.", + "isRequired": true, + "name": "h", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Automatically replays the content list. This method buffers on every loop to the beginning of the content list.", + "name": "SetLoop", + "params": [ + { + "default": null, + "description": "Enables the automatic replaying of the content list.", + "isRequired": true, + "name": "loop", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "deprecatedDescription": "This function is deprecated. Roku no longer supports Macrovision and this function exists as a no-op so that legacy scripts do not break.", + "isDeprecated": true, + "name": "SetMacrovisionLevel", + "params": [ + { + "default": null, + "isRequired": true, + "name": "level", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the max resolution required by your video.", + "name": "SetMaxVideoDecodeResolution", + "params": [ + { + "default": null, + "description": "The maximum height required by your video.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The maximum width required by your video.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the next item in the Content List to be played.", + "name": "SetNext", + "params": [ + { + "default": null, + "description": "The unique ID of the content item to be played next.", + "isRequired": true, + "name": "item", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the interval to receive playback position events from the roVideoPlayer.", + "name": "SetPositionNotificationPeriod", + "params": [ + { + "default": null, + "description": "The notification period for receiving playback position events in seconds. Notification events sent to the script specify the position in seconds relative to the beginning of the stream. If the value is 0, position notifications are never sent. The default value is 0.", + "isRequired": true, + "name": "period", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the timedMetaData keys that the BrightScript channel is interested in receiving from the timedMetaData event.", + "name": "SetTimedMetaDataForKeys", + "params": [ + { + "default": null, + "isRequired": true, + "name": "keys" + } + ] + }, + { + "description": "Stops playback and resets the seek position; keeps the player’s current position unchanged.", + "name": "Stop", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + } + ], + "name": "ifVideoPlayer", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoplayer.md" + }, + "ifvideoscreen": { + "deprecatedDescription": "This interface is deprecated.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This interface is deprecated.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.", + "implementers": [ + { + "description": "The Video Screen object implements the video playback portion of the user interface", + "name": "roVideoScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "Close", + "params": [], + "returnType": "Void" + }, + { + "description": "Enables [trickplay mode](/docs/developer-program/media-playback/trick-mode.md) during playback. Trickplay mode provides visual feedback during playback operations such as forward and rewind by displaying the timestamp of the content being seeked.", + "name": "EnableTrickPlay", + "params": [ + { + "default": null, + "description": "Specifies the player to use trick play mode during playback.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Pauses the video.", + "name": "Pause", + "params": [], + "returnType": "Void" + }, + { + "description": "Begins downloading and buffering of a video that may be selected by a user. This can be used to reduce buffering delays after a user has selected a video for playback. A typical use would be to call PreBuffer() when the user is in the roSpringboardScreen (or equivalent), anticipating that the user will select a video on the springboard screen for download.", + "name": "PreBuffer", + "params": [], + "returnDescription": "A flag indicating whether the operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Resumes the video.", + "name": "Resume", + "params": [], + "returnType": "Void" + }, + { + "description": "Acknowledges the previously generated resume request event with a trickplay ID. When roVideoScreen receives this message, it will resume playback unless a more recent trickplay session has started. In the latter case, the ResumeAck message is ignored and playback is not resumed.", + "name": "ResumeAck", + "params": [ + { + "default": null, + "isRequired": true, + "name": "Integer" + } + ] + }, + { + "description": "Sets the start point of playback for the current video to a specific offset.", + "name": "Seek", + "params": [ + { + "default": null, + "description": "The number of milliseconds to offset the playback of the current content item.", + "isRequired": true, + "name": "milliSeconds", + "type": "Integer" + } + ], + "returnDescription": "A flag indicating whether the seek operation was successful.", + "returnType": "Boolean" + }, + { + "description": "Sets CGMS (Copy Guard Management System) on analog outputs to the desired level.", + "name": "SetCGMS", + "params": [ + { + "default": null, + "description": "The level to which CGMS is set. This may be one of the following values: * 0 - No Copy Restriction * 1 - Copy No More * 2 - Copy Once Allowed * 3 – No Copying Permitted", + "isRequired": true, + "name": "level", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the content to be played in the [roVideoScreen](https://developer.roku.com/docs/references/brightscript/components/rovideoscreen.md\"roVideoScreen\"). You can call this method while playing video, and it will use the new data (release date, length, and title) when showing the program info in the Heads Up Display (HUD).", + "name": "SetContent", + "params": [ + { + "default": null, + "description": "An [roAssociativeArray](https://developer.roku.com/docs/references/brightscript/components/roassociativearray.md\"roAssociativeArray\") describing the attributes ([Content Meta-Data](/docs/developer-program/getting-started/architecture/content-metadata.md \" Content Meta-Data\")) for the title.", + "isRequired": true, + "name": "content", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the target display window for the video.", + "name": "SetDestinationRect", + "params": [ + { + "default": null, + "description": "The parameters of the target display window, which include the x and y coordinates, width, and height {x:Integer, y:Integer, w:Integer, h:Integer} The default value is: {x:0, y:0, w:0, h:0}, which is full screen.", + "isRequired": true, + "name": "rect", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the target display window for the video. This is an alternate way to call the **SetDestinationRectangle()** method with each parameter passed separately instead of in an Associative Array.", + "name": "SetDestinationRect", + "params": [ + { + "default": null, + "description": "The x-coordinate of the rectangle.", + "isRequired": true, + "name": "x", + "type": "Integer" + }, + { + "default": null, + "description": "The y-coordinate of the rectangle.", + "isRequired": true, + "name": "y", + "type": "Integer" + }, + { + "default": null, + "description": "The width of the rectangle.", + "isRequired": true, + "name": "w", + "type": "Integer" + }, + { + "default": null, + "description": "The height of the rectangle.", + "isRequired": true, + "name": "h", + "type": "Integer" + } + ] + }, + { + "description": "Enables guided trickplay mode (the channel takes control of resuming playback after trickplay). This means that the roVideoScreen object no longer resumes playback automatically, but instead waits for an acknowledgment from the channel.", + "name": "SetGuidedTrickPlay", + "params": [ + { + "default": null, + "description": "Specifies the player to use guided trickplay mode during playback. By default guided trickplay mode is disabled.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Enables the player to start again from the beginning immediately after finishing a content item.", + "name": "SetLoop", + "params": [ + { + "default": null, + "description": "Specifies the player to start playing a content item from the beginning as soon as the item has been completed. The default is false.", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "deprecatedDescription": "This function is deprecated. Roku no longer supports Macrovision and this function exists as a no-op so that legacy scripts do not break.\n", + "isDeprecated": true, + "name": "SetMacrovisionLevel", + "params": [ + { + "default": null, + "isRequired": true, + "name": "level", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the max resolution required by your video.", + "name": "SetMaxVideoDecodeResolution", + "params": [ + { + "default": null, + "description": "The maximum height required by your video.", + "isRequired": true, + "name": "width", + "type": "Integer" + }, + { + "default": null, + "description": "The maximum width required by your video.", + "isRequired": true, + "name": "height", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the interval for receiving playback position events from the roVideoScreen object.", + "name": "SetPositionNotificationPeriod", + "params": [ + { + "default": null, + "description": "The notification period for receiving playback position events in seconds. Notification events sent to the script specify the position in seconds relative to the beginning of the stream. If the value is 0, position notifications are never sent. The default value is 0.", + "isRequired": true, + "name": "period", + "type": "Integer" + } + ], + "returnType": "Void" + }, + { + "description": "Sets preview mode on/off. In preview mode, trick play operations (fast forward and rewind) are disabled.", + "name": "SetPreviewMode", + "params": [ + { + "default": null, + "description": "A flag specifying to turn preview mode on (true) or off (false).", + "isRequired": true, + "name": "enable", + "type": "Boolean" + } + ], + "returnType": "Void" + }, + { + "description": "Specifies the timedMetaData keys that the BrightScript channel is interested in receiving from the timedMetaData event.", + "name": "SetTimedMetaDataForKeys", + "params": [ + { + "default": null, + "isRequired": true, + "name": "keys" + } + ] + }, + { + "description": "Displays/refreshes the screen after creation or state changes.", + "name": "Show", + "params": [], + "returnDescription": "A flag indicating whether the screen was successfully refreshed.", + "returnType": "Boolean" + } + ], + "name": "ifVideoScreen", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifvideoscreen.md" + }, + "ifxmlelement": { + "implementers": [ + { + "description": "roXMLElement is used to contain an XML tree", + "name": "roXMLElement", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmlelement.md" + } + ], + "methods": [ + { + "description": "Adds an attribute value to the element. If an attribute of the same name already exists it is replaced. XML attribute order is not preserved.", + "name": "AddAttribute", + "params": [ + { + "default": null, + "description": "The name of the attribute to be added to the element.", + "isRequired": true, + "name": "attr", + "type": "String" + }, + { + "default": null, + "description": "The value of the attribute being added to the element.", + "isRequired": true, + "name": "value", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Adds a new unnamed/empty child element and returns it. This should generally be followed by a call to the [SetName()](#setnamename-as-string-as-void) method of the child element. Alternatively, the [AddElement()](#addelementname-as-string-as-object) or [AddElementWidthBody()](#addelementwithbodyname-as-string-body-as-object-as-object) method can be used to combine this step with additional construction into one call.", + "name": "AddBodyElement", + "params": [], + "returnDescription": "Object value.", + "returnType": "Object" + }, + { + "description": "Adds a new child element with the specified name and returns the new element.", + "name": "AddElement", + "params": [ + { + "default": null, + "description": "The name of the child element to be added.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "The new element added.", + "returnType": "Object" + }, + { + "description": "Adds a new child element with the specified name and text from the specified body string, and returns the new element.", + "name": "AddElementWithBody", + "params": [ + { + "default": null, + "description": "The name of the child element to be added.", + "isRequired": true, + "name": "name", + "type": "String" + }, + { + "default": null, + "description": "The text of the child element to be added (via the body string).", + "isRequired": true, + "name": "body", + "type": "Object" + } + ], + "returnDescription": "The new element added.", + "returnType": "Object" + }, + { + "description": "Adds text to the element.", + "name": "AddText", + "params": [ + { + "default": null, + "description": "The text to be added to the element.", + "isRequired": true, + "name": "text", + "type": "String" + } + ], + "returnType": "Void" + }, + { + "description": "Removes all attributes and children from the element, as well as setting the name to empty.", + "name": "Clear", + "params": [], + "returnType": "Void" + }, + { + "description": "Serializes the element to XML document text.", + "name": "GenXML", + "params": [ + { + "default": null, + "isRequired": true, + "name": "gen" + } + ], + "returnDescription": "A serialized string." + }, + { + "description": "Serializes the element to XML document text. The specified header is used to begin the output (for example, as a custom XML declaration).", + "name": "GenXMLHdr", + "params": [ + { + "default": null, + "description": "Specify the header with which the output begins.", + "isRequired": true, + "name": "hdr", + "type": "String" + } + ], + "returnDescription": "A serialized string.", + "returnType": "String" + }, + { + "description": "Returns the XML attributes of the element.", + "name": "GetAttributes", + "params": [], + "returnDescription": "An associative array representing the XML attributes of the element.", + "returnType": "Object" + }, + { + "description": "Returns the body of the element. If the element contains child elements, `GetBody()` returns an [roXMLList](https://developer.roku.com/docs/references/brightscript/components/roxmllist.md\"roXMLList\") representing those elements, like GetChildElements(). If there are no children but the element contains text, `GetBody()` returns an [roString](https://developer.roku.com/docs/references/brightscript/components/rostring.md\"roString\") like `GetText()`. If the element is empty, `GetBody()` returns invalid.", + "name": "GetBody", + "params": [], + "returnDescription": "Object body.", + "returnType": "Object" + }, + { + "description": "If this element contains any child elements, this method returns an [roXMLList](https://developer.roku.com/docs/references/brightscript/components/roxmllist.md\"roXMLList\") representing those elements. If there are no child elements, returns invalid.", + "name": "GetChildElements", + "params": [], + "returnDescription": "An element list.", + "returnType": "Object" + }, + { + "description": "If this element contains any child elements, this method returns an [roList](https://developer.roku.com/docs/references/brightscript/components/rolist.md\"roList\") representing those elements. If there are no child elements, returns invalid. The difference between this function and `GetChildElements()` is that `GetChildNodes()` handles the case of mixed XML content, i.e., content with both child elements and text such as: Child TextMore Text. In this case `GetChildNodes()` called with the top level as an argument would return an roList with two elements. The first element would be an `roXMLElement` containing the information about. The second would be an `roString` containing \"More Text\".", + "name": "GetChildNodes", + "params": [], + "returnDescription": "An element list.", + "returnType": "Object" + }, + { + "description": "Returns the name of the element.", + "name": "GetName", + "params": [], + "returnDescription": "Element name.", + "returnType": "String" + }, + { + "description": "Returns an [roXMLList](https://developer.roku.com/docs/references/brightscript/components/roxmllist.md\"roXMLList\") representing all child elements of this element whose case-sensitive name is specified. If only one element matches the name, an roXMLList containing one element is returned. If no elements match, an empty roXMLList is returned.", + "name": "GetNamedElements", + "params": [ + { + "default": null, + "description": "The parent element containing the child elements to be listed. Matching of the parent element name is case-sensitive.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "An element list.", + "returnType": "Object" + }, + { + "description": "Returns an [roXMLList](https://developer.roku.com/docs/references/brightscript/components/roxmllist.md\"roXMLList\") representing all child elements of this element whose case-insensitive name is specified. If only one element matches the name, an roXMLList containing one element is returned. If no elements match, an empty roXMLList is returned.", + "name": "GetNamedElementsCi", + "params": [ + { + "default": null, + "description": "The parent element containing the child elements to be listed. Matching of the parent element name is case-sensitive.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "An element list.", + "returnType": "Object" + }, + { + "description": "Returns any text contained in the element. This returns immediate body text only (for example, it does not include text from child elements).", + "name": "GetText", + "params": [], + "returnDescription": "The text in the element.", + "returnType": "String" + }, + { + "name": "HasAttribute", + "params": [ + { + "default": null, + "description": "The element attribute to be verified.", + "isRequired": true, + "name": "attr", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the element has the specified attribute.", + "returnType": "Boolean" + }, + { + "description": "Checks whether the element has the specified name.", + "name": "IsName", + "params": [ + { + "default": null, + "description": "The element name to be verified.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the element has the specified name.", + "returnType": "Boolean" + }, + { + "description": "Parses a string of XML.", + "name": "Parse", + "params": [ + { + "default": null, + "description": "The XML string to be parsed", + "isRequired": true, + "name": "xml", + "type": "String" + } + ], + "returnDescription": "A flag indicating whether the operation was successful. In that case, other methods below can then be used to extract information about the parsed element.", + "returnType": "Boolean" + }, + { + "description": "Sets the element text from the specified string", + "name": "SetBody", + "params": [ + { + "default": null, + "description": "The string to be used to set the element text.", + "isRequired": true, + "name": "body", + "type": "Object" + } + ], + "returnType": "Void" + }, + { + "description": "Sets the name of the element.", + "name": "SetName", + "params": [ + { + "default": null, + "description": "The name of the element.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnType": "Void" + } + ], + "name": "ifXMLElement", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifxmlelement.md" + }, + "ifxmllist": { + "implementers": [ + { + "description": "Contains a list of roXML objects", + "name": "roXMLList", + "url": "https://developer.roku.com/docs/references/brightscript/components/roxmllist.md" + } + ], + "methods": [ + { + "description": "If the list contains exactly one item, this function returns the attributes of that item. Otherwise it returns invalid.", + "name": "GetAttributes", + "params": [], + "returnDescription": "The object item.", + "returnType": "Object" + }, + { + "description": "If the list contains exactly one item, this function returns the child elements of that item. Otherwise it returns invalid.", + "name": "GetChildElements", + "params": [], + "returnDescription": "The object item.", + "returnType": "Object" + }, + { + "description": "Returns a new XMLList that contains all roXMLElements that matched the passed in name (case-sensitive matching is used). This is the same as using the dot operator on an roXMLList.", + "name": "GetNamedElements", + "params": [ + { + "default": null, + "description": "The XML element to be used to find matches.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "An XMLList that contains the matches.", + "returnType": "Object" + }, + { + "description": "Returns a new XMLList that contains all roXMLElements that matched the passed in name (case-insensitive matching is used). This is the same as using the dot operator on an roXMLList.", + "name": "GetNamedElementsCi", + "params": [ + { + "default": null, + "description": "The XML element to be used to find matches.", + "isRequired": true, + "name": "name", + "type": "String" + } + ], + "returnDescription": "An XMLList that contains the matches.", + "returnType": "Object" + }, + { + "description": "If the list contains exactly one item, this function returns the text of that item. Otherwise, it returns an empty string.", + "name": "GetText", + "params": [], + "returnDescription": "The object string.", + "returnType": "String" + }, + { + "description": "If the list contains exactly one item, this function returns that item. Otherwise, it returns itself.", + "name": "Simplify", + "params": [], + "returnDescription": "The object item.", + "returnType": "Object" + } + ], + "name": "ifXMLList", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/interfaces/ifxmllist.md" + } + }, + "events": { + "roaudioplayerevent": { + "description": "The roAudioPlayer sends the roAudioPlayerEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roAudioPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/roaudioplayer.md" + } + ], + "methods": [ + { + "name": "isFullResult", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPartialResult", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPaused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestFailed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestSucceeded", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isResumed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStatusMessage", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isTimedMetaData", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roAudioPlayerEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roaudioplayerevent.md" + }, + "rocaptionrendererevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nroCaptionRenderer events are sent by the roCaptionRenderer to notify a BrightScript channel when caption data needs to be rendered. Specifically, one of two events, isCaptionText and isUpdateCaptionRequest, will be fired depending on the mode that the caption renderer is in. The mode is set by calling the [ifCaptionRenderer.SetMode()](https://developer.roku.com/docs/references/brightscript/interfaces/ifcaptionrenderer.mdsetmodemode-as-integer-as-void \"ifCaptionRenderer.SetMode()\") function.\n\n| Mode | Event |\n| --- | --- |\n| 1 | isCaptionUpdateRequest |\n| 2 | isCaptionText |", + "implementers": [ + { + "name": "roCaptionRenderer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocaptionrenderer.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isCaptionText", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isCaptionUpdateRequest", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roCaptionRendererEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rocaptionrendererevent.md" + }, + "rocecstatusevent": { + "availableSince": "8", + "description": "_Available since Roku OS 8_\n\nThis event determines the active source status for set boxes. Channels subscribing to the `roCECStatusEvent` will be notified when the active-source status of the device changes per the CEC message traffic.\n\nTo use the roCECStatusEvent, follow the steps below:\n\n1. Connect a Roku STB to a TV which transmits and receives CEC messages\n2. Select the HDMI input to which the STB is connected\n3. Switch away and then back to the STB's HDMI input", + "implementers": [], + "methods": [ + { + "name": "isActiveSource", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roCECStatusEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rocecstatusevent.md" + }, + "rochannelstoreevent": { + "description": "The roChannelStore sends an roChannelStoreEvent in response to a call to any of several Get\\* methods in [ifChannelStore](https://developer.roku.com/docs/references/brightscript/interfaces/ifchannelstore.md\"ifChannelStore\"). The following predicates indicate its valid event types:", + "implementers": [ + { + "name": "roChannelStore", + "url": "https://developer.roku.com/docs/references/brightscript/components/rochannelstore.md" + } + ], + "methods": [ + { + "name": "isRequestFailed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestInterrupted", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestSucceeded", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roChannelStoreEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rochannelstoreevent.md" + }, + "rocoderegistrationscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roCodeRegistrationScreen sends the roCodeRegistrationScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roCodeRegistrationScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rocoderegistrationscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roCodeRegistrationScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rocoderegistrationscreenevent.md" + }, + "rodeviceinfoevent": { + "availableSince": "8", + "description": "_Available since Roku OS 8_\n\nThe roDeviceInfo component sends the roDeviceInfoEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roDeviceInfo", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodeviceinfo.md" + } + ], + "methods": [ + { + "name": "isCaptionModeChanged", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStatusMessage", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roDeviceInfoEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rodeviceinfoevent.md" + }, + "rofilesystemevent": { + "description": "The roFileSystem component sends the roFileSystemEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roFileSystem", + "url": "https://developer.roku.com/docs/references/brightscript/components/rofilesystem.md" + } + ], + "methods": [ + { + "name": "isStorageDeviceAdded", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStorageDeviceRemoved", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roFileSystemEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rofilesystemevent.md" + }, + "rogridscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roGridScreen sends the roGridScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roGridScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rogridscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isListItemFocused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roGridScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rogridscreenevent.md" + }, + "rohdmihotplugevent": { + "description": "The roHdmiStatus sends the roHdmiHotPlugEvent with the following predicates that indicate its valid event types:", + "implementers": [], + "methods": [ + { + "name": "isHdmiHotPlug", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roHdmiHotPlugEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rohdmihotplugevent.md" + }, + "rohdmistatusevent": { + "description": "The roHdmiStatus sends the roHdmiStatusEvent with the following predicates that indicate its valid event types:", + "implementers": [], + "methods": [ + { + "name": "isHdmiStatus", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roHdmiStatusEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rohdmistatusevent.md" + }, + "roimagecanvasevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roImageCanvas sends the roImageCanvasEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roImageCanvas", + "url": "https://developer.roku.com/docs/references/brightscript/components/roimagecanvas.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roImageCanvasEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roimagecanvasevent.md" + }, + "roinputevent": { + "description": "The roInput component sends the roInputEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roInput", + "url": "https://developer.roku.com/docs/references/brightscript/components/roinput.md" + } + ], + "methods": [ + { + "name": "GetInfo", + "params": [], + "returnType": "Object" + }, + { + "name": "isInput", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roInputEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roinputevent.md" + }, + "rokeyboardscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roKeyboardScreen sends the roKeyboardScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roKeyboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rokeyboardscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roKeyboardScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rokeyboardscreenevent.md" + }, + "rolistscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roListScreen sends the roListScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roListScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rolistscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isListItemFocused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roListScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rolistscreenevent.md" + }, + "romessagedialogevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roMessageDialog sends the roMessageDialogEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roMessageDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/romessagedialog.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roMessageDialogEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/romessagedialogevent.md" + }, + "romicrophoneevent": { + "description": "The [roMicrophone](https://developer.roku.com/docs/references/brightscript/components/romicrophone.md\"roMicrophone\") component sends the `roMicrophoneEvent` with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roMicrophone", + "url": "https://developer.roku.com/docs/references/brightscript/components/romicrophone.md" + } + ], + "methods": [ + { + "name": "IsRecordingDone", + "params": [], + "returnType": "Boolean" + }, + { + "name": "IsRecordingInfo", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roMicrophoneEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/romicrophoneevent.md" + }, + "roonelinedialogevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nSends the roOneLineDialogEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roOneLineDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/roonelinedialog.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roOneLineDialogEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roonelinedialogevent.md" + }, + "roparagraphscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roParagraphScreen sends the roParagraphScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roParagraphScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roparagraphscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roParagraphScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roparagraphscreenevent.md" + }, + "ropinentrydialogevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roPinEntryDialog sends the roPinEntryDialogEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roPinEntryDialog", + "url": "https://developer.roku.com/docs/references/brightscript/components/ropinentrydialog.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roPinEntryDialogEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/ropinentrydialogevent.md" + }, + "roposterscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roPosterScreen sends the roPosterScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roPosterScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roposterscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isAdSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListFocused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemFocused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roPosterScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roposterscreenevent.md" + }, + "rosearchscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roSearchScreen sends the roSearchScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roSearchScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosearchscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "IsButtonInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "IsCleared", + "params": [], + "returnType": "Boolean" + }, + { + "name": "IsFullResult", + "params": [], + "returnType": "Dynamic" + }, + { + "name": "IsPartialResult", + "params": [], + "returnType": "Boolean" + }, + { + "name": "IsScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roSearchScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rosearchscreenevent.md" + }, + "rosgnodeevent": { + "description": "An roSGNode object sends roSGNodeEvent messages to a specified port when changes occur in nodes. An roSGNodeEvent is also sent as the argument of field observer callback functions. Both of these cases allow a SceneGraph application to respond to events. roSGNodeEvent supports the following methods.", + "implementers": [ + { + "name": "roSGNode", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgnode.md" + } + ], + "methods": [ + { + "name": "getData", + "params": [], + "returnType": "Dynamic" + }, + { + "name": "getField", + "params": [], + "returnType": "Dynamic" + }, + { + "name": "getInfo", + "params": [], + "returnType": "Object" + }, + { + "name": "getNode", + "params": [], + "returnType": "Dynamic" + }, + { + "name": "getRoSGNode", + "params": [], + "returnType": "Dynamic" + } + ], + "name": "roSGNodeEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rosgnodeevent.md" + }, + "rosgscreenevent": { + "description": "**roSGScreenEvents** are events sent to a scene graph **roSGScreen** by the framework. Other than when notifying the channel's main BrightScript thread that the screen is being closed, and thus that the channel should be terminated, channels do not generally handle these events.", + "implementers": [ + { + "name": "roSGScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rosgscreen.md" + } + ], + "methods": [ + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roSGScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rosgscreenevent.md" + }, + "roslideshowevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roSlideShow sends the roSlideShowEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roSlideShow", + "url": "https://developer.roku.com/docs/references/brightscript/components/roslideshow.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPaused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPlaybackPosition", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestFailed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestInterrupted", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestSucceeded", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isResumed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roSlideShowEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/roslideshowevent.md" + }, + "rosocketevent": { + "description": "An roStreamSocket or roDataGramSocket object sends the roSocketEvent to indicate a change in the status of the socket. The socket must enable specific event notifications via the notify methods of ifSocketAsync.", + "implementers": [ + { + "name": "roDataGramSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rodatagramsocket.md" + }, + { + "name": "roStreamSocket", + "url": "https://developer.roku.com/docs/references/brightscript/components/rostreamsocket.md" + } + ], + "methods": [ + { + "name": "GetSocketID", + "params": [], + "returnType": "Integer" + } + ], + "name": "roSocketEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rosocketevent.md" + }, + "rospringboardscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roSpringboardScreen sends the roSpringboardScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roSpringboardScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rospringboardscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRemoteKeyPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roSpringboardScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rospringboardscreenevent.md" + }, + "rosystemlogevent": { + "description": "roSystemLogEvents are sent when enabled via [roSystemLog](https://developer.roku.com/docs/references/brightscript/components/rosystemlog.md\"roSystemLog\"). roSystemLogEvent has the following method:", + "implementers": [], + "methods": [ + { + "name": "GetInfo", + "params": [], + "returnType": "Object" + } + ], + "name": "roSystemLogEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rosystemlogevent.md" + }, + "rotextscreenevent": { + "deprecatedDescription": "This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n\nBeginning July 1st, 2017, any new channels using this component will be rejected during certification.\n\nBeginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated and will be removed from Roku OS on January 1st, 2019.\n> \n> Beginning July 1st, 2017, any new channels using this component will be rejected during certification.\n> \n> Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roTextScreen sends the roTextScreenEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roTextScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotextscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "isButtonPressed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roTextScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rotextscreenevent.md" + }, + "rotexttospeechevent": { + "description": "> Please note this component is only available on the following devices: Roku Streaming Stick (3600X), Roku Express (3700X) and Express+ (3710X), Roku Premiere (4620X) and Premiere+ (4630X), Roku Ultra (4640X), and any Roku TV running Roku OS version 7.2 and later.\n\nThe [roTextToSpeech](https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md\"roTextToSpeech\") component sends the roTextToSpeechEvent with the following predicates that indicate its valid event types.", + "implementers": [ + { + "name": "roTextToSpeech", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexttospeech.md" + } + ], + "methods": [ + { + "name": "GetData", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetIndex", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetInfo", + "params": [], + "returnType": "Object" + } + ], + "name": "roTextToSpeechEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rotexttospeechevent.md" + }, + "rotexturerequestevent": { + "description": "The [roTextureManager](https://developer.roku.com/docs/references/brightscript/components/rotexturemanager.md\"roTextureManager\") sends the roTextureRequestEvent after completing a request.", + "implementers": [ + { + "name": "roTextureManager", + "url": "https://developer.roku.com/docs/references/brightscript/components/rotexturemanager.md" + } + ], + "methods": [ + { + "name": "GetBitmap", + "params": [], + "returnType": "Object" + }, + { + "name": "GetId", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetState", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetURI", + "params": [], + "returnType": "String" + } + ], + "name": "roTextureRequestEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rotexturerequestevent.md" + }, + "rouniversalcontrolevent": { + "description": "The roScreen object sends the roUniversalControlEvent with the following related methods. If an app constrains the events processed to just the roUniversalControlEvent, the app will work with any controller. The GetID(), GetChar(), GetKey(), and IsPress() methods can be used instead of parsing the GetInt() return value to more effectively distinguish between remote control and keyboard key presses, and the key press and release events.", + "implementers": [ + { + "name": "roScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/roscreen.md" + } + ], + "methods": [ + { + "name": "GetChar", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetID", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetInt", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetKey", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetRemoteID", + "params": [], + "returnType": "String" + }, + { + "name": "IsPress", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roUniversalControlEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rouniversalcontrolevent.md" + }, + "rourlevent": { + "description": "The roUrlTransfer component sends the roUrlEvent with the following methods:", + "implementers": [ + { + "name": "roUrlTransfer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rourltransfer.md" + } + ], + "methods": [ + { + "name": "GetFailureReason", + "params": [], + "returnType": "String" + }, + { + "name": "GetInt", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetResponseCode", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetResponseHeaders", + "params": [], + "returnType": "Object" + }, + { + "name": "GetResponseHeadersArray", + "params": [], + "returnType": "Object" + }, + { + "name": "GetSourceIdentity", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetString", + "params": [], + "returnType": "String" + }, + { + "name": "GetTargetIpAddress", + "params": [], + "returnType": "String" + } + ], + "name": "roUrlEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rourlevent.md" + }, + "rovideoplayerevent": { + "description": "The roVideoPlayer sends the roVideoPlayerEvent with the following predicates that indicate its valid event types:", + "implementers": [ + { + "name": "roVideoPlayer", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoplayer.md" + } + ], + "methods": [ + { + "name": "GetIndex", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetInfo", + "params": [], + "returnType": "Object" + }, + { + "name": "isCaptionModeChanged", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isDownloadSegmentInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isFormatDetected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isFullResult", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isListItemSelected", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPaused", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isPlaybackPosition", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestFailed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isRequestSucceeded", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isResumed", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isSegmentDownloadStarted", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStatusMessage", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStreamSegmentInfo", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isStreamStarted", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isTimedMetaData", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roVideoPlayerEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rovideoplayerevent.md" + }, + "rovideoscreenevent": { + "deprecatedDescription": "This component is deprecated.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n", + "description": "> This component is deprecated.Beginning July 1st, 2017, any new channels using this component will be rejected during certification.Beginning January 1st, 2018, any updates to existing channels using this component will be rejected during certification.\n\nThe roVideoScreen sends the roVideoScreenEvent with the same predicates as in [roVideoPlayerEvent](https://developer.roku.com/docs/references/brightscript/events/rovideoplayerevent.md\"roVideoPlayerEvent\"). In addition to the events listed in roVideoPlayerEvent, the following events are also supported:", + "implementers": [ + { + "name": "roVideoScreen", + "url": "https://developer.roku.com/docs/references/brightscript/components/rovideoscreen.md" + } + ], + "isDeprecated": true, + "methods": [ + { + "name": "GetData", + "params": [], + "returnType": "Integer" + }, + { + "name": "GetIndex", + "params": [], + "returnType": "Integer" + }, + { + "name": "isPartialResult", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isResumeRequest", + "params": [], + "returnType": "Boolean" + }, + { + "name": "isScreenClosed", + "params": [], + "returnType": "Boolean" + } + ], + "name": "roVideoScreenEvent", + "properties": [], + "url": "https://developer.roku.com/docs/references/brightscript/events/rovideoscreenevent.md" + } + } +} \ No newline at end of file diff --git a/src/roku-types/index.ts b/src/roku-types/index.ts new file mode 100644 index 000000000..271afcf77 --- /dev/null +++ b/src/roku-types/index.ts @@ -0,0 +1,79 @@ +// eslint-disable-next-line @typescript-eslint/no-require-imports +import data = require('./data.json'); + +//apply any transforms/overrides before exporting + +export const nodes = data.nodes; +export const components = data.components; +export const interfaces = data.interfaces; +export const events = data.events; + +interface BrightScriptDocLookup { + name: string; + url?: string; + description?: string; +} + +interface PossiblyDeprecated { + isDeprecated?: boolean; + deprecatedDescription?: string; +} + +export interface BRSBaseMethodData extends PossiblyDeprecated { + params: { + name: string; + isRequired: boolean; + type: string; + }[]; + returnType: string; +} +export interface BRSEventMethodData extends BRSBaseMethodData { + name: string; +} + +export interface BRSInterfaceMethodData extends BRSEventMethodData { + description: string; + returnDescription: string; +} + +export interface BRSPropertyData { + name: string; + description: string; + default: string; + type: string; +} + +export interface BRSFieldData { + name: string; + type: string; + default: string; + accessPermission: string; + description: string; +} + + +export interface SGNodeData extends BrightScriptDocLookup { + description: string; + fields: BRSFieldData[]; + events: BrightScriptDocLookup[]; + interfaces: BrightScriptDocLookup[]; +} + +export interface BRSComponentData extends BrightScriptDocLookup, PossiblyDeprecated { + interfaces: BrightScriptDocLookup[]; + events: BrightScriptDocLookup[]; + constructors: BRSBaseMethodData[]; + description: string; +} + +export interface BRSInterfaceData extends BrightScriptDocLookup, PossiblyDeprecated { + properties: BRSPropertyData[]; + implementers: BrightScriptDocLookup[]; + methods: BRSInterfaceMethodData[]; +} + +export interface BRSEventData extends BrightScriptDocLookup, PossiblyDeprecated { + properties: BRSPropertyData[]; + implementers: BrightScriptDocLookup[]; + methods: BRSEventMethodData[]; +} diff --git a/tsconfig.json b/tsconfig.json index fce17e636..92b871a8c 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -2,7 +2,7 @@ "compilerOptions": { "noImplicitAny": false, "target": "ES2017", - "module": "commonjs", + "module": "CommonJS", "sourceMap": true, "rootDir": "src", "outDir": "dist", @@ -15,9 +15,11 @@ "downlevelIteration": true, "noUnusedLocals": true, "allowSyntheticDefaultImports": true, + "resolveJsonModule": true, "lib": [ "es2017", - "es2019.array" + "es2019.array", + "dom" ] }, "include": [ From d96110b5ac4b0778f708ce69b6c3ad58c907457d Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 13 Apr 2022 08:02:47 -0400 Subject: [PATCH 191/278] rename benchmarks debug target --- .vscode/launch.json | 36 ++++++++++++++++++------------------ 1 file changed, 18 insertions(+), 18 deletions(-) diff --git a/.vscode/launch.json b/.vscode/launch.json index ba2f23675..d14e99f93 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -1,24 +1,6 @@ { "version": "0.2.0", "configurations": [ - { - "name": "Launch Program", - "program": "${workspaceFolder}/benchmarks/index.js", - "request": "launch", - "args": [ - "--versions", - "current", - "--targets", - "lexer", - "--noprepare", - "--project", - "C:/projects/roku/brighterscript-template" - ], - "skipFiles": [ - "/**" - ], - "type": "pwa-node" - }, { "name": "Debug Tests (Current file)", "type": "node", @@ -65,6 +47,24 @@ "!**/node_modules/vscode-languageserver/**" ] }, + { + "name": "Debug Benchmarks", + "program": "${workspaceFolder}/benchmarks/index.js", + "request": "launch", + "args": [ + "--versions", + "current", + "--targets", + "lexer", + "--noprepare", + "--project", + "C:/projects/roku/brighterscript-template" + ], + "skipFiles": [ + "/**" + ], + "type": "pwa-node" + }, { "name": "Debug Roku Docs Scraper", "type": "node", From 27ed7ce729554979ab6d2e6080203d21569045ab Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 13 Apr 2022 08:06:35 -0400 Subject: [PATCH 192/278] update changelog for v0.48.0 --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7815a97be..eaa70040d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,15 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 +## [0.48.0](https://github.com/rokucommunity/brighterscript/compare/v0.47.3...v0.48.0) - 2022-04-13 +### Added + - language support for native BrightScript optional chaining ([#546](https://github.com/rokucommunity/brighterscript/pull/546)) + - validation for all known `createObject` values and parameters. ([#435](https://github.com/rokucommunity/brighterscript/pull/435)) +### Fixed + - add missing statements and expressions to `createVisitor` ([#567](https://github.com/rokucommunity/brighterscript/pull/567)) + + + ## [0.47.3](https://github.com/rokucommunity/brighterscript/compare/v0.47.2...v0.47.3) - 2022-04-08 ### Fixed - accuracy issues when parsing the manifest ([#565](https://github.com/rokucommunity/brighterscript/pull/565)) From 39e19e241c44c31ae59ba68039ddb0e9939f351a Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Wed, 13 Apr 2022 08:07:14 -0400 Subject: [PATCH 193/278] 0.48.0 --- package-lock.json | 4 ++-- package.json | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index e0511f408..a01a16023 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "brighterscript", - "version": "0.47.3", + "version": "0.48.0", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "brighterscript", - "version": "0.47.3", + "version": "0.48.0", "license": "MIT", "dependencies": { "@rokucommunity/bslib": "^0.1.1", diff --git a/package.json b/package.json index 4e580dd0e..6bc20a01e 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "brighterscript", - "version": "0.47.3", + "version": "0.48.0", "description": "A superset of Roku's BrightScript language.", "scripts": { "preversion": "npm run build && npm run lint && npm run test", From f3c1dbda7ac4681973b18a580242ef0184fec295 Mon Sep 17 00:00:00 2001 From: Bronley Plumb Date: Thu, 14 Apr 2022 09:58:03 -0400 Subject: [PATCH 194/278] Create object fixes (#568) * Move CreateObject validation into plugin * Only add single CreateObject diagnostics for all scopes * Don't flag component library CreateObject calls * Fix lint issues --- src/Scope.spec.ts | 50 +++++++ src/Scope.ts | 82 +----------- src/bscPlugin/BscPlugin.ts | 10 +- src/bscPlugin/validation/ScopeValidator.ts | 143 ++++++++++++++++++--- src/util.ts | 7 + 5 files changed, 195 insertions(+), 97 deletions(-) diff --git a/src/Scope.spec.ts b/src/Scope.spec.ts index 311c7bc9a..62553bafc 100644 --- a/src/Scope.spec.ts +++ b/src/Scope.spec.ts @@ -210,6 +210,56 @@ describe('Scope', () => { ]); }); + it('only adds a single diagnostic when the file is used in multiple scopes', () => { + program.setFile('components/Comp1.xml', trim` + + +