()
+ return {
+ SvelteScriptElement(node: AST.SvelteScriptElement) {
+ const contextAttr = findAttribute(node, "context")
+ if (contextAttr && getStaticAttributeValue(contextAttr) === "module") {
+ contextModule = node
+ return
+ }
+
+ scriptNode = node
+ },
+ SvelteReactiveStatement(node: AST.SvelteReactiveStatement) {
+ reactiveStatements.push(node)
+ },
+ ":function"(node: TSESTree.FunctionLike) {
+ functions.push(node)
+ },
+ ExportNamedDeclaration(node: TSESTree.ExportNamedDeclaration) {
+ if (
+ contextModule &&
+ contextModule.range[0] < node.range[0] &&
+ node.range[1] < contextModule.range[1]
+ ) {
+ return
+ }
+ if (
+ node.declaration?.type === "VariableDeclaration" &&
+ (node.declaration.kind === "let" || node.declaration.kind === "var")
+ ) {
+ for (const decl of node.declaration.declarations) {
+ if (decl.id.type === "Identifier") {
+ props.add(decl.id)
+ }
+ }
+ }
+ for (const spec of node.specifiers) {
+ if (spec.exportKind === "type") {
+ continue
+ }
+ if (isMutableProp(spec.local)) {
+ props.add(spec.exported)
+ }
+ }
+ },
+ "Program:exit"() {
+ for (const prop of props) {
+ const variable = findVariable(context, prop)
+ if (
+ !variable ||
+ // ignore multiple definitions
+ variable.defs.length > 1
+ ) {
+ return
+ }
+ for (const reference of variable.references) {
+ if (reference.isWrite()) continue
+ const id = reference.identifier as TSESTree.Identifier
+ if (
+ variable.defs.some(
+ (def) =>
+ def.node.range[0] <= id.range[0] &&
+ id.range[1] <= def.node.range[1],
+ )
+ ) {
+ // The reference is in the variable definition.
+ continue
+ }
+ if (isInReactivityScope(id)) continue
+
+ context.report({
+ node: id,
+ messageId: "potentiallyLoseReactivity",
+ })
+ }
+ }
+ },
+ }
+
+ /** Checks whether given prop id is mutable variable or not */
+ function isMutableProp(id: TSESTree.Identifier) {
+ const variable = findVariable(context, id)
+ if (!variable || variable.defs.length === 0) {
+ return false
+ }
+ return variable.defs.every((def) => {
+ if (def.type !== "Variable") {
+ return false
+ }
+ return def.parent.kind === "let" || def.parent.kind === "var"
+ })
+ }
+
+ /** Checks whether given id is in potentially reactive scope or not */
+ function isInReactivityScope(id: TSESTree.Identifier) {
+ if (
+ !scriptNode ||
+ id.range[1] <= scriptNode.range[0] ||
+ scriptNode.range[1] <= id.range[0]
+ ) {
+ // The reference is in the template.
+ return true
+ }
+ if (
+ reactiveStatements.some(
+ (node) =>
+ node.range[0] <= id.range[0] && id.range[1] <= node.range[1],
+ )
+ ) {
+ // The reference is in the reactive statement.
+ return true
+ }
+ if (
+ functions.some(
+ (node) =>
+ node.range[0] <= id.range[0] && id.range[1] <= node.range[1],
+ )
+ ) {
+ // The reference is in the function.
+ return true
+ }
+ return false
+ }
+ },
+})
diff --git a/src/utils/rules.ts b/src/utils/rules.ts
index 7dac11750..11cc8eb08 100644
--- a/src/utils/rules.ts
+++ b/src/utils/rules.ts
@@ -29,6 +29,7 @@ import noExportLoadInSvelteModuleInKitPages from "../rules/no-export-load-in-sve
import noExtraReactiveCurlies from "../rules/no-extra-reactive-curlies"
import noImmutableReactiveStatements from "../rules/no-immutable-reactive-statements"
import noInnerDeclarations from "../rules/no-inner-declarations"
+import noLossOfPropReactivity from "../rules/no-loss-of-prop-reactivity"
import noNotFunctionHandler from "../rules/no-not-function-handler"
import noObjectInTextMustaches from "../rules/no-object-in-text-mustaches"
import noReactiveFunctions from "../rules/no-reactive-functions"
@@ -88,6 +89,7 @@ export const rules = [
noExtraReactiveCurlies,
noImmutableReactiveStatements,
noInnerDeclarations,
+ noLossOfPropReactivity,
noNotFunctionHandler,
noObjectInTextMustaches,
noReactiveFunctions,
diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml
new file mode 100644
index 000000000..76cfa5393
--- /dev/null
+++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml
@@ -0,0 +1,4 @@
+- message: Referencing props that potentially lose reactivity should be avoided.
+ line: 4
+ column: 17
+ suggestions: null
diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte
new file mode 100644
index 000000000..d28686862
--- /dev/null
+++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte
@@ -0,0 +1,5 @@
+
diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte
new file mode 100644
index 000000000..e4db2dec8
--- /dev/null
+++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte
new file mode 100644
index 000000000..bab40db47
--- /dev/null
+++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte
@@ -0,0 +1,5 @@
+
diff --git a/tests/src/rules/no-loss-of-prop-reactivity.ts b/tests/src/rules/no-loss-of-prop-reactivity.ts
new file mode 100644
index 000000000..20fd103a6
--- /dev/null
+++ b/tests/src/rules/no-loss-of-prop-reactivity.ts
@@ -0,0 +1,16 @@
+import { RuleTester } from "eslint"
+import rule from "../../../src/rules/no-loss-of-prop-reactivity"
+import { loadTestCases } from "../../utils/utils"
+
+const tester = new RuleTester({
+ parserOptions: {
+ ecmaVersion: 2020,
+ sourceType: "module",
+ },
+})
+
+tester.run(
+ "no-loss-of-prop-reactivity",
+ rule as any,
+ loadTestCases("no-loss-of-prop-reactivity"),
+)