Add pre-commit hook to run eslint, jest and webpack build before commiting

This commit is contained in:
2019-09-06 13:35:45 +02:00
parent 41e2a50d85
commit 878716a6e2
5 changed files with 383 additions and 78 deletions

310
.eslintrc Normal file
View File

@@ -0,0 +1,310 @@
{
"extends": [
"plugin:react/recommended"
],
"env": {
browser: true,
es6: true,
node: true,
"jest": true
},
"parser": "babel-eslint",
"parserOptions": {
"parser": "babel-eslint",
"ecmaVersion": 2018,
"sourceType": "module",
"ecmaFeatures": {
"react": true,
"jsx": true
}
},
"settings": {
"react": {
"createClass": "createReactClass", // Regex for Component Factory to use,
// default to "createReactClass"
"pragma": "React", // Pragma to use, default to "React"
"version": "detect", // React version. "detect" automatically picks the version you have installed.
// You can also use `16.0`, `16.3`, etc, if you want to override the detected value.
// default to latest and warns if missing
// It will default to "detect" in the future
"flowVersion": "0.53" // Flow version
},
"propWrapperFunctions": [
// The names of any function used to wrap propTypes, e.g. `forbidExtraProps`. If this isn't set, any propTypes wrapped in a function will be skipped.
"forbidExtraProps",
{"property": "freeze", "object": "Object"},
{"property": "myFavoriteWrapper"}
],
"linkComponents": [
// Components used as alternatives to <a> for linking, eg. <Link to={ url } />
"Hyperlink",
{"name": "Link", "linkAttribute": "to"}
]
},
"plugins": [
"standard",
"react",
"import",
"node",
"promise"
],
"rules": {
////////// Possible Errors //////////
"comma-dangle": 1, // disallow trailing commas in object literals
"no-cond-assign": 0, // disallow assignment in conditional expressions
"no-console": 0, // disallow use of console (off by default in the node environment)
"no-control-regex": 1, // disallow control characters in regular expressions
"no-debugger": 1, // disallow use of debugger
"no-dupe-args": 1, // disallow duplicate arguments in functions
"no-dupe-keys": 1, // disallow duplicate keys when creating object literals
"no-duplicate-case": 1, // disallow a duplicate case label
"no-empty-character-class": 1, // disallow the use of empty character classes in regular expressions
"no-empty": 1, // disallow empty statements
"no-empty-pattern": "error",
"no-ex-assign": 1, // disallow assigning to the exception in a catch block
"no-extra-boolean-cast": 0, // disallow double-negation boolean casts in a boolean context
"no-extra-parens": ["error", "functions"], // disallow unnecessary parentheses (off by default)
"no-extra-semi": 1, // disallow unnecessary semicolons
"no-func-assign": 1, // disallow overwriting functions written as function declarations
"no-inner-declarations": ["error", "functions"], // disallow function or variable declarations in nested blocks
"no-invalid-regexp": 1, // disallow invalid regular expression strings in the RegExp constructor
"no-irregular-whitespace": 1, // disallow irregular whitespace outside of strings and comments
"no-negated-in-lhs": 1, // disallow negation of the left operand of an in expression
"no-obj-calls": 1, // disallow the use of object properties of the global object (Math and JSON) as functions
"no-regex-spaces": 1, // disallow multiple spaces in a regular expression literal
"no-sparse-arrays": 1, // disallow sparse arrays
"no-unreachable": 1, // disallow unreachable statements after a return, throw, continue, or break statement
"use-isnan": 1, // disallow comparisons with the value NaN
"valid-jsdoc": 1, // Ensure JSDoc comments are valid (off by default)
"valid-typeof": ["error", { "requireStringLiterals": true }], // Ensure that the results of typeof are compared against a valid string
"no-unexpected-multiline": 1, // Avoid code that looks like two expressions but is actually one (off by default)
////////// Best Practices //////////
"accessor-pairs": 1, // enforces getter/setter pairs in objects (off by default)
"block-scoped-var": 0, // treat var statements as if they were block scoped (off by default)
"complexity": 1, // specify the maximum cyclomatic complexity allowed in a program (off by default)
"consistent-return": 1, // require return statements to either always or never specify values
"curly": 1, // specify curly brace conventions for all control statements
"default-case": 1, // require default case in switch statements (off by default)
"dot-notation": 1, // encourages use of dot notation whenever possible
"dot-location": ["error", "property"], // enforces consistent newlines before or after dots (off by default)
"eqeqeq": ["error", "always", { "null": "ignore" }], // require the use of === and !==
"guard-for-in": 1, // make sure for-in loops have an if statement (off by default)
"no-alert": 1, // disallow the use of alert, confirm, and prompt
"no-caller": 1, // disallow use of arguments.caller or arguments.callee
"no-div-regex": 1, // disallow division operators explicitly at beginning of regular expression (off by default)
"no-else-return": 1, // disallow else after a return in an if (off by default)
"no-eq-null": 1, // disallow comparisons to null without a type-checking operator (off by default)
"no-eval": 1, // disallow use of eval()
"no-extend-native": 1, // disallow adding to native types
"no-extra-bind": 1, // disallow unnecessary function binding
"no-fallthrough": 1, // disallow fallthrough of case statements
"no-floating-decimal": 1, // disallow the use of leading or trailing decimal points in numeric literals (off by default)
"no-implied-eval": 1, // disallow use of eval()-like methods
"no-iterator": 1, // disallow usage of __iterator__ property
"no-labels": 1, // disallow use of labeled statements
"no-lone-blocks": 1, // disallow unnecessary nested blocks
"no-loop-func": 1, // disallow creation of functions within loops
"no-multi-spaces": 1, // disallow use of multiple spaces
"no-multi-str": 1, // disallow use of multiline strings
"no-native-reassign": 1, // disallow reassignments of native objects
"no-new-func": 1, // disallow use of new operator for Function object
"no-new-wrappers": 1, // disallows creating new instances of String, Number, and Boolean
"no-new": 1, // disallow use of new operator when not part of the assignment or comparison
"no-octal-escape": 1, // disallow use of octal escape sequences in string literals, such as var foo = "Copyright \251";
"no-octal": 1, // disallow use of octal literals
"no-param-reassign": 0, // disallow reassignment of function parameters (off by default)
"no-process-env": 0, // disallow use of process.env (off by default)
"no-proto": 1, // disallow usage of __proto__ property
"no-redeclare": 0, // disallow declaring the same variable more then once
"no-return-assign": ["error", "except-parens"], // disallow use of assignment in return statement
"no-script-url": 1, // disallow use of javascript: urls
"no-self-compare": 1, // disallow comparisons where both sides are exactly the same (off by default)
"no-sequences": 1, // disallow use of comma operator
"no-throw-literal": 1, // restrict what can be thrown as an exception (off by default)
"no-void": 1, // disallow use of void operator (off by default)
"no-warning-comments": 1, // disallow usage of configurable warning terms in comments, e.g. TODO or FIXME (off by default)
"no-with": 1, // disallow use of the with statement
"radix": 1, // require use of the second argument for parseInt() (off by default)
"vars-on-top": 0, // requires to declare all vars on top of their containing scope (off by default)
"wrap-iife": ["error", "any", { "functionPrototypeMethods": true }], // require immediate function invocation to be wrapped in parentheses (off by default)
"yoda": 1, // require or disallow Yoda conditions
////////// Strict Mode //////////
//"strict": 1, // controls location of Use Strict Directives
////////// Variables //////////
"no-catch-shadow": 0, // disallow the catch clause parameter name being the same as a variable in the outer scope (off by default in the node environment)
"no-delete-var": 1, // disallow deletion of variables
"no-label-var": 1, // disallow labels that share a name with a variable
"no-shadow": 0, // disallow declaration of variables already declared in the outer scope
"no-shadow-restricted-names": 1, // disallow shadowing of names such as arguments
"no-undef": ["error"], // disallow use of undeclared variables unless mentioned in a /*global */ block
"no-undef-init": 1, // disallow use of undefined when initializing variables
"no-undefined": 0, // disallow use of undefined variable (off by default)
"no-unused-vars": ["error", { "vars": "all", "args": "none", "ignoreRestSiblings": true, "no-unused-vars": ["_", "__", "___"] }], // disallow declaration of variables that are not used in the code
"no-use-before-define": ["error", { "functions": false, "classes": false, "variables": false }], // disallow use of variables before they are defined
////////// Node.js //////////
"handle-callback-err": ["error", "^(err|error)$" ], // enforces error handling in callbacks (off by default) (on by default in the node environment)
"no-mixed-requires": 1, // disallow mixing regular variable and require declarations (off by default) (on by default in the node environment)
"no-new-require": 1, // disallow use of new operator with the require function (off by default) (on by default in the node environment)
"no-path-concat": 1, // disallow string concatenation with __dirname and __filename (off by default) (on by default in the node environment)
"no-process-exit": 1, // disallow process.exit() (on by default in the node environment)
"no-restricted-modules": 1, // restrict usage of specified node modules (off by default)
"no-sync": 0, // disallow use of synchronous methods (off by default)
////////// Stylistic Issues //////////
"array-bracket-spacing": 1, // enforce spacing inside array brackets (off by default)
"brace-style": ["error", "1tbs", { "allowSingleLine": true }], // enforce one true brace style (off by default)
"camelcase": ["error", { "properties": "never" }], // require camel case names
"comma-spacing": ["error", { "before": false, "after": true }], // enforce spacing before and after comma
"comma-style": ["error", "last"], // enforce one true comma style (off by default)
"computed-property-spacing": 1, // require or disallow padding inside computed properties (off by default)
"consistent-this": ["error", "self"], // enforces consistent naming when capturing the current execution context (off by default)
"eol-last": 1, // enforce newline at the end of file, with no multiple empty lines
"func-names": 0, // require function expressions to have a name (off by default)
//"func-style": 1, // enforces use of function declarations or expressions (off by default)
"key-spacing": ["error", { "beforeColon": false, "afterColon": true }], // enforces spacing between keys and values in object literal properties
"lines-around-comment": 0, // enforces empty lines around comments (off by default)
"linebreak-style": 1, // disallow mixed 'LF' and 'CRLF' as linebreaks (off by default)
"max-nested-callbacks": 1, // specify the maximum depth callbacks can be nested (off by default)
"new-cap": ["error", { "newIsCap": true, "capIsNew": false }], // require a capital letter for constructors
"new-parens": 1, // disallow the omission of parentheses when invoking a constructor with no arguments
"newline-after-var": 0, // allow/disallow an empty newline after var statement (off by default)
"no-array-constructor": 1, // disallow use of the Array constructor
"no-continue": 0, // disallow use of the continue statement (off by default)
"no-inline-comments": 0, // disallow comments inline after code (off by default)
"no-lonely-if": 1, // disallow if as the only statement in an else block (off by default)
"no-mixed-spaces-and-tabs": 1, // disallow mixed spaces and tabs for indentation
"no-multiple-empty-lines": 1, // disallow multiple empty lines (off by default)
"no-nested-ternary": 0, // disallow nested ternary expressions (off by default)
"no-new-object": 1, // disallow use of the Object constructor
"no-spaced-func": 1, // disallow space between function identifier and application
"no-ternary": 0, // disallow the use of ternary operators (off by default)
"no-trailing-spaces": 1, // disallow trailing whitespace at the end of lines
"no-underscore-dangle": 0, // disallow dangling underscores in identifiers
"one-var": 0, // allow just one var statement per function (off by default)
"operator-assignment": 1, // require assignment operator shorthand where possible or prohibit it entirely (off by default)
"operator-linebreak": ["error", "after", { "overrides": { "?": "before", ":": "before" } }], // enforce operators to be placed before or after line breaks (off by default)
"padded-blocks": ["error", { "blocks": "never", "switches": "never", "classes": "never" }], // enforce padding within blocks (off by default)
"quote-props": 0, // require quotes around object literal property names (off by default)
"quotes": ["error", "double", { "avoidEscape": true, "allowTemplateLiterals": true }], // specify whether double or single quotes should be used
"semi-spacing": ["error", { "before": false, "after": true }], // enforce spacing before and after semicolons
"semi": ["error", "always"], // require or disallow use of semicolons instead of ASI
"sort-vars": 1, // sort variables within the same declaration block (off by default)
"space-before-blocks": 1, // require or disallow space before blocks (off by default)
"space-before-function-paren": 0, // require or disallow space before function opening parenthesis (off by default)
"space-in-parens": 1, // require or disallow spaces inside parentheses (off by default)
"space-infix-ops": 1, // require spaces around operators
"keyword-spacing": ["error", { "before": true, "after": true }],
"space-unary-ops": ["error", { "words": true, "nonwords": false }], // require or disallow spaces before/after unary operators (words on by default, nonwords off by default)
"spaced-comment": ["warn", "always", {
"line": { "markers": ["*package", "!", "/", ",", "="] },
"block": { "balanced": true, "markers": ["*package", "!", ",", ":", "::", "flow-include", "*globals"], "exceptions": ["*", "globals.*", "eslint-env"] }
}], // require or disallow a space immediately following the // or /* in a comment (off by default)
"wrap-regex": 1, // require regex literals to be wrapped in parentheses (off by default)
////////// ECMAScript 6 //////////
"constructor-super": 1, // verify super() callings in constructors (off by default)
"generator-star-spacing": ["error", { "before": true, "after": true }], // enforce the spacing around the * in generator functions (off by default)
"no-this-before-super": 1, // disallow to use this/super before super() calling in constructors (off by default)
"no-var": 1, // require let or const instead of var (off by default)
"object-shorthand": 1, // require method and property shorthand syntax for object literals (off by default)
"prefer-const": 1, // suggest using of const declaration for variables that are never modified after declared (off by default)
////////// Legacy //////////
"max-depth": 1, // specify the maximum depth that blocks can be nested (off by default)
"max-len": ["warn", { "code": 120 }], // specify the maximum length of a line in your program (off by default)
"max-params": ["error", { max: 5 }], // limits the number of parameters that can be used in the function declaration. (off by default)
"max-statements": ["error", { "max": 50 }], // specify the maximum number of statement allowed in a function (off by default)
"no-bitwise": 1, // disallow use of bitwise operators (off by default)
"no-plusplus": 0, // disallow use of unary operators, ++ and -- (off by default)
////////// Standard 2019 ///////
"arrow-spacing": ["error", { "before": true, "after": true }],
"block-spacing": ["error", "always"],
"func-call-spacing": ["error", "never"],
"indent": ["error", 2, {
"SwitchCase": 1,
"VariableDeclarator": 1,
"outerIIFEBody": 1,
"MemberExpression": 1,
"FunctionDeclaration": { "parameters": 1, "body": 1 },
"FunctionExpression": { "parameters": 1, "body": 1 },
"CallExpression": { "arguments": 1 },
"ArrayExpression": 1,
"ObjectExpression": 1,
"ImportDeclaration": 1,
"flatTernaryExpressions": false,
"ignoreComments": false
}],
"no-class-assign": "error",
"no-compare-neg-zero": "error",
"no-const-assign": "error",
"no-constant-condition": ["error", { "checkLoops": false }],
"no-dupe-class-members": "error",
"no-global-assign": "error",
"no-mixed-operators": ["error", {
"groups": [
["==", "!=", "===", "!==", ">", ">=", "<", "<="],
["&&", "||"],
["in", "instanceof"]
],
"allowSamePrecedence": true
}],
"no-new-symbol": "error",
"no-return-await": "error",
"no-self-assign": "error",
"no-tabs": "error",
"no-template-curly-in-string": "error",
"no-unmodified-loop-condition": "error",
"no-unneeded-ternary": ["error", { "defaultAssignment": false }],
"no-unsafe-finally": "error",
"no-unsafe-negation": "error",
"no-unused-expressions": ["error", { "allowShortCircuit": true, "allowTernary": true, "allowTaggedTemplates": true }],
"no-useless-call": "error",
"no-useless-computed-key": "error",
"no-useless-constructor": "error",
"no-useless-escape": "error",
"no-useless-rename": "error",
"no-useless-return": "error",
"no-whitespace-before-property": "error",
"object-curly-spacing": ["error", "always"],
"object-property-newline": ["error", { "allowMultiplePropertiesPerLine": true }],
"prefer-promise-reject-errors": "error",
"rest-spread-spacing": ["error", "never"],
"symbol-description": "error",
"template-curly-spacing": ["error", "never"],
"template-tag-spacing": ["error", "never"],
"unicode-bom": ["error", "never"],
"yield-star-spacing": ["error", "both"],
"import/export": "error",
"import/first": "error",
"import/no-duplicates": "error",
"import/no-named-default": "error",
"import/no-webpack-loader-syntax": "error",
"node/no-deprecated-api": "error",
"node/process-exit-as-throw": "error",
"promise/param-names": "error",
"standard/array-bracket-even-spacing": ["error", "either"],
"standard/computed-property-even-spacing": ["error", "even"],
"standard/no-callback-literal": "error",
"standard/object-curly-even-spacing": ["error", "either"],
"jsx-a11y/anchor-is-valid": 0,
"react/prop-types": 0
}
}

View File

@@ -7,21 +7,37 @@
"dev": "webpack-dev-server --mode development --open",
"build": "webpack --optimize-minimize --config webpack.production.config.js",
"webpack": "webpack --config webpack.config.js --watch",
"lint": "eslint src",
"test": "jest",
"test-watch": "jest --watch"
"test-watch": "jest --watch",
"test-coverage": "jest --coverage"
},
"pre-commit": [
"lint",
"test",
"build"
],
"keywords": [],
"author": "",
"license": "ISC",
"devDependencies": {
"@babel/cli": "^7.5.5",
"@babel/core": "^7.5.5",
"@babel/plugin-proposal-class-properties": "^7.5.5",
"@babel/plugin-proposal-private-methods": "^7.4.4",
"@babel/preset-env": "^7.5.5",
"@babel/preset-react": "^7.0.0",
"babel-eslint": "^11.0.0-beta.0",
"babel-loader": "^8.0.6",
"babel-plugin-styled-components": "^1.10.6",
"eslint": "^6.3.0",
"eslint-plugin-import": "^2.18.2",
"eslint-plugin-node": "^10.0.0",
"eslint-plugin-promise": "^4.2.1",
"eslint-plugin-react": "^7.14.3",
"eslint-plugin-standard": "^4.0.1",
"jest": "^24.9.0",
"pre-commit": "^1.2.2",
"react-hot-loader": "^4.12.11",
"webpack": "^4.39.2",
"webpack-cli": "^3.3.7",

View File

@@ -1,3 +1,4 @@
/* eslint-disable indent */
import styled, { keyframes, css } from "styled-components";
const fade = keyframes`
@@ -20,12 +21,11 @@ const Apple = styled.div.attrs(({ theme, zoom }) => ({
&:after {
content: "${props => (!props.died ? "🍎" : "🐛")}";
display: inline-block;
animation: ${props =>
props.died
? "none"
: css`
${fade} 1s ease-out alternate infinite
`};
animation: ${({ died }) =>
!died &&
css`
${fade} 1s ease-out alternate infinite
`};
}
`;

View File

@@ -87,7 +87,7 @@ const Snake = ({
(cell.type === "snake" && <SnakePart zoom={zoom} value={cell} died={died} />)
}
</Stage>
{!started && <Banner fade>Press 'r' to start the game</Banner>}
{!started && <Banner fade>Press &apos;r&apos; to start the game</Banner>}
{died && lastGameId !== gameId && hasHighscore(score) && (
<HighscoreInput score={score} gameId={gameId} name={lastUsername} onSubmit={onSubmitHighscore} />
)}

View File

@@ -1,6 +1,6 @@
export class Middleware {
#action = null;
#middleware = null;
_action = null;
_middleware = null;
constructor(action, middleware) {
if (!action) {
@@ -15,26 +15,26 @@ export class Middleware {
if (typeof middleware !== "function") {
throw new Error("Argument 'middleware' must be a string.");
}
this.#action = action;
this.#middleware = middleware;
this._action = action;
this._middleware = middleware;
}
get action() {
return this.#action;
return this._action;
}
get middleware() {
return this.#middleware;
return this._middleware;
}
invoke(dispatch, state, action, getState) {
return this.#middleware(dispatch, state, action, getState);
return this._middleware(dispatch, state, action, getState);
}
}
export class After {
#action = null;
#handler = null;
_action = null;
_handler = null;
constructor(action, handler) {
if (!action) {
@@ -49,26 +49,26 @@ export class After {
if (typeof handler !== "function") {
throw new Error("Argument 'handler' must be a string.");
}
this.#action = action;
this.#handler = handler;
this._action = action;
this._handler = handler;
}
get action() {
return this.#action;
return this._action;
}
get handler() {
return this.#handler;
return this._handler;
}
invoke(dispatch, state, action, getState) {
return this.#handler(dispatch, state, action, getState);
return this._handler(dispatch, state, action, getState);
}
}
export class Reducer {
#action = null;
#reducer = null;
_action = null;
_reducer = null;
constructor(action, reducer) {
if (!action) {
@@ -83,55 +83,55 @@ export class Reducer {
if (typeof reducer !== "function") {
throw new Error("Argument 'reducer' must be a string.");
}
this.#action = action;
this.#reducer = reducer;
this._action = action;
this._reducer = reducer;
}
get action() {
return this.#action;
return this._action;
}
get reducer() {
return this.#reducer;
return this._reducer;
}
invoke(state, action) {
return this.#reducer(state, action);
return this._reducer(state, action);
}
}
export class Module {
#name = null;
#initialState = null;
#actions = {};
#after = {};
_name = null;
_initialState = null;
_actions = {};
_after = {};
constructor(name, initialState) {
if (!name) {
throw new Error("Argument 'name' is required.");
}
this.#name = name;
this.#initialState = initialState;
this._name = name;
this._initialState = initialState;
this.middlewares = this.middlewares.bind(this);
}
get name() {
return this.#name;
return this._name;
}
middleware(action, middleware) {
this.#register(new Middleware(action, middleware));
this._register(new Middleware(action, middleware));
}
reducer(action, reducer) {
this.#register(new Reducer(action, reducer));
this._register(new Reducer(action, reducer));
}
after(action, middleware) {
this.#register(new After(action, middleware));
this._register(new After(action, middleware));
}
#register(reducerOrMiddleware) {
_register(reducerOrMiddleware) {
if (
!(reducerOrMiddleware instanceof Reducer) &&
!(reducerOrMiddleware instanceof Middleware) &&
@@ -140,12 +140,12 @@ export class Module {
throw new Error("Only objects of type Reducer, Middleware or After can register");
}
if (!(reducerOrMiddleware instanceof After)) {
if (this.#actions[reducerOrMiddleware.action]) {
if (this._actions[reducerOrMiddleware.action]) {
throw new Error(`Reducer or middleware with action "${reducerOrMiddleware.action}" is already registered`);
}
this.#actions[reducerOrMiddleware.action] = reducerOrMiddleware;
this._actions[reducerOrMiddleware.action] = reducerOrMiddleware;
} else {
this.#after[reducerOrMiddleware.action] = reducerOrMiddleware;
this._after[reducerOrMiddleware.action] = reducerOrMiddleware;
}
return reducerOrMiddleware;
}
@@ -157,28 +157,28 @@ export class Module {
if (typeof action !== "function") {
throw new Error("Argument 'action' must be function");
}
const type = `[${this.#name}] ${name}`;
const type = `[${this._name}] ${name}`;
const typedAction = (...args) => ({ ...action.apply(this, args), type });
return [type, typedAction];
}
#getActionByType(Type) {
return Object.keys(this.#actions)
.map(key => ({ key, action: this.#actions[key] }))
_getActionByType(Type) {
return Object.keys(this._actions)
.map(key => ({ key, action: this._actions[key] }))
.reduce((actions, { key, action }) => (action instanceof Type ? { ...actions, [key]: action } : actions), {});
}
middlewares({ dispatch, getState }) {
return next => action => {
next(action);
const middlewares = this.#getActionByType(Middleware);
const middlewares = this._getActionByType(Middleware);
const middleware = middlewares[action.type];
if (middleware) {
middleware.invoke(dispatch, getState()[this.#name], action, getState);
middleware.invoke(dispatch, getState()[this._name], action, getState);
}
const after = this.#after[action.type];
const after = this._after[action.type];
if (after) {
after.invoke(dispatch, getState()[this.#name], action, getState);
after.invoke(dispatch, getState()[this._name], action, getState);
}
return null;
};
@@ -186,8 +186,8 @@ export class Module {
get reducers() {
return {
[this.#name]: (state = this.#initialState, action) => {
const reducers = this.#getActionByType(Reducer);
[this._name]: (state = this._initialState, action) => {
const reducers = this._getActionByType(Reducer);
const reducer = reducers[action.type];
if (reducer) {
return reducer.invoke(state, action);
@@ -199,16 +199,16 @@ export class Module {
state(selector) {
return state => {
if (!this.#name) {
if (!this._name) {
throw new Error("Unable to detect module name");
}
if (!state) {
throw new Error("No state was passed to the state selector");
}
if (!state[this.#name]) {
throw new Error(`There is no state for module '${this.#name}'`);
if (!state[this._name]) {
throw new Error(`There is no state for module '${this._name}'`);
}
return selector(state[this.#name]);
return selector(state[this._name]);
};
}
@@ -225,24 +225,3 @@ export class Module {
}
export default Module;
export const attach = (modules, Component) => {
console.log("modules, Component", modules, Component);
return (...args) => {
console.log("args", args);
const component = Component({
GAME: {
actions: {
startLoop() {
console.log("=== START LOOP ===");
}
},
state: {}
}
});
return component;
component.props = { ...component.props };
console.log("component", component);
return component.render.apply(this, args);
};
};