mirror of
https://github.com/Stremio/stremio-web.git
synced 2026-04-20 23:12:13 +00:00
conflict resolved
This commit is contained in:
commit
0aa6b06df2
52 changed files with 1256 additions and 541 deletions
85
.eslintrc
Normal file
85
.eslintrc
Normal file
|
|
@ -0,0 +1,85 @@
|
||||||
|
{
|
||||||
|
"extends": [
|
||||||
|
"eslint:recommended",
|
||||||
|
"plugin:react/recommended"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"react": {
|
||||||
|
"version": "detect"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"globals": {
|
||||||
|
"YT": "readonly",
|
||||||
|
"FB": "readonly"
|
||||||
|
},
|
||||||
|
"env": {
|
||||||
|
"commonjs": true,
|
||||||
|
"browser": true,
|
||||||
|
"es6": true
|
||||||
|
},
|
||||||
|
"parserOptions": {
|
||||||
|
"ecmaVersion": 9,
|
||||||
|
"ecmaFeatures": {
|
||||||
|
"jsx": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"ignorePatterns": [
|
||||||
|
"/*",
|
||||||
|
"!/src",
|
||||||
|
"src/routes/Settings",
|
||||||
|
"src/routes/Player",
|
||||||
|
"src/video"
|
||||||
|
],
|
||||||
|
"rules": {
|
||||||
|
"arrow-parens": "error",
|
||||||
|
"arrow-spacing": "error",
|
||||||
|
"block-spacing": "error",
|
||||||
|
"comma-spacing": "error",
|
||||||
|
"eol-last": "error",
|
||||||
|
"eqeqeq": "error",
|
||||||
|
"func-call-spacing": "error",
|
||||||
|
"indent": [
|
||||||
|
"error",
|
||||||
|
4,
|
||||||
|
{
|
||||||
|
"SwitchCase": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"no-console": "error",
|
||||||
|
"no-extra-semi": "error",
|
||||||
|
"no-eq-null": "error",
|
||||||
|
"no-multi-spaces": "error",
|
||||||
|
"no-multiple-empty-lines": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"max": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"no-template-curly-in-string": "error",
|
||||||
|
"no-trailing-spaces": "error",
|
||||||
|
"no-useless-concat": "error",
|
||||||
|
"no-unreachable": "error",
|
||||||
|
"no-unused-vars": "error",
|
||||||
|
"prefer-const": "error",
|
||||||
|
"quotes": [
|
||||||
|
"error",
|
||||||
|
"single"
|
||||||
|
],
|
||||||
|
"quote-props": [
|
||||||
|
"error",
|
||||||
|
"as-needed",
|
||||||
|
{
|
||||||
|
"unnecessary": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"semi": "error",
|
||||||
|
"semi-spacing": "error",
|
||||||
|
"space-before-blocks": "error",
|
||||||
|
"valid-typeof": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"requireStringLiterals": true
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
21
package.json
21
package.json
|
|
@ -10,7 +10,8 @@
|
||||||
"start": "webpack-dev-server --mode development",
|
"start": "webpack-dev-server --mode development",
|
||||||
"build": "webpack --mode production",
|
"build": "webpack --mode production",
|
||||||
"storybook": "start-storybook --ci --config-dir ./storybook --static-dir ./ --port 6060",
|
"storybook": "start-storybook --ci --config-dir ./storybook --static-dir ./ --port 6060",
|
||||||
"test": "jest"
|
"test": "jest",
|
||||||
|
"lint": "eslint src"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"a-color-picker": "1.2.1",
|
"a-color-picker": "1.2.1",
|
||||||
|
|
@ -24,19 +25,19 @@
|
||||||
"react": "16.12.0",
|
"react": "16.12.0",
|
||||||
"react-dom": "16.12.0",
|
"react-dom": "16.12.0",
|
||||||
"react-focus-lock": "2.2.1",
|
"react-focus-lock": "2.2.1",
|
||||||
"spatial-navigation-polyfill": "git+ssh://git@github.com/NikolaBorislavovHristov/spatial-navigation.git#964d09bf2b0853e27af6c25924b595d6621a019d",
|
"spatial-navigation-polyfill": "git+ssh://git@github.com/Stremio/spatial-navigation.git#381b4f37d138e66ae8ea025e240af3704245e5dc",
|
||||||
"stremio-colors": "git+ssh://git@github.com/Stremio/stremio-colors.git#v2.0.4",
|
"stremio-colors": "git+ssh://git@github.com/Stremio/stremio-colors.git#v2.0.4",
|
||||||
"stremio-core-web": "git+ssh://git@github.com/stremio/stremio-core-web.git#da5b37865004d0ae140518c4f276d1ed1a1483d9",
|
"stremio-core-web": "git+ssh://git@github.com/stremio/stremio-core-web.git#da5b37865004d0ae140518c4f276d1ed1a1483d9",
|
||||||
"stremio-icons": "git+ssh://git@github.com/Stremio/stremio-icons.git#v1.0.11",
|
"stremio-icons": "git+ssh://git@github.com/Stremio/stremio-icons.git#v1.0.11",
|
||||||
"vtt.js": "0.13.0"
|
"vtt.js": "0.13.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "7.7.4",
|
"@babel/core": "7.7.5",
|
||||||
"@babel/plugin-proposal-class-properties": "7.7.4",
|
"@babel/plugin-proposal-class-properties": "7.7.4",
|
||||||
"@babel/plugin-proposal-object-rest-spread": "7.7.4",
|
"@babel/plugin-proposal-object-rest-spread": "7.7.4",
|
||||||
"@babel/preset-env": "7.7.4",
|
"@babel/preset-env": "7.7.6",
|
||||||
"@babel/preset-react": "7.7.4",
|
"@babel/preset-react": "7.7.4",
|
||||||
"@babel/runtime": "7.7.4",
|
"@babel/runtime": "7.7.6",
|
||||||
"@storybook/addon-actions": "5.2.8",
|
"@storybook/addon-actions": "5.2.8",
|
||||||
"@storybook/addon-console": "1.2.1",
|
"@storybook/addon-console": "1.2.1",
|
||||||
"@storybook/addons": "5.2.8",
|
"@storybook/addons": "5.2.8",
|
||||||
|
|
@ -45,10 +46,12 @@
|
||||||
"@testing-library/react-hooks": "3.2.1",
|
"@testing-library/react-hooks": "3.2.1",
|
||||||
"babel-loader": "8.0.6",
|
"babel-loader": "8.0.6",
|
||||||
"clean-webpack-plugin": "3.0.0",
|
"clean-webpack-plugin": "3.0.0",
|
||||||
"copy-webpack-plugin": "5.0.5",
|
"copy-webpack-plugin": "5.1.1",
|
||||||
"css-loader": "3.2.1",
|
"css-loader": "3.3.2",
|
||||||
"cssnano": "4.1.10",
|
"cssnano": "4.1.10",
|
||||||
"cssnano-preset-advanced": "4.0.7",
|
"cssnano-preset-advanced": "4.0.7",
|
||||||
|
"eslint": "6.7.2",
|
||||||
|
"eslint-plugin-react": "7.17.0",
|
||||||
"html-webpack-plugin": "3.2.0",
|
"html-webpack-plugin": "3.2.0",
|
||||||
"jest": "24.9.0",
|
"jest": "24.9.0",
|
||||||
"less": "3.10.3",
|
"less": "3.10.3",
|
||||||
|
|
@ -57,8 +60,8 @@
|
||||||
"postcss-loader": "3.0.0",
|
"postcss-loader": "3.0.0",
|
||||||
"react-test-renderer": "16.12.0",
|
"react-test-renderer": "16.12.0",
|
||||||
"storybook-addon-jsx": "7.1.13",
|
"storybook-addon-jsx": "7.1.13",
|
||||||
"terser-webpack-plugin": "2.2.1",
|
"terser-webpack-plugin": "2.3.0",
|
||||||
"webpack": "4.41.2",
|
"webpack": "4.41.3",
|
||||||
"webpack-cli": "3.3.10",
|
"webpack-cli": "3.3.10",
|
||||||
"webpack-dev-server": "3.9.0"
|
"webpack-dev-server": "3.9.0"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
|
|
@ -41,4 +42,13 @@ const Button = React.forwardRef(({ className, href, disabled, children, ...props
|
||||||
|
|
||||||
Button.displayName = 'Button';
|
Button.displayName = 'Button';
|
||||||
|
|
||||||
|
Button.propTypes = {
|
||||||
|
className: PropTypes.string,
|
||||||
|
href: PropTypes.string,
|
||||||
|
disabled: PropTypes.bool,
|
||||||
|
children: PropTypes.node,
|
||||||
|
onKeyDown: PropTypes.func,
|
||||||
|
onMouseDown: PropTypes.func
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Button;
|
module.exports = Button;
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const Button = require('stremio/common/Button');
|
const Button = require('stremio/common/Button');
|
||||||
|
|
@ -22,4 +23,10 @@ const Checkbox = React.forwardRef(({ className, checked, children, ...props }, r
|
||||||
|
|
||||||
Checkbox.displayName = 'Checkbox';
|
Checkbox.displayName = 'Checkbox';
|
||||||
|
|
||||||
|
Checkbox.propTypes = {
|
||||||
|
className: PropTypes.string,
|
||||||
|
checked: PropTypes.bool,
|
||||||
|
children: PropTypes.node
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Checkbox;
|
module.exports = Checkbox;
|
||||||
|
|
|
||||||
|
|
@ -16,9 +16,8 @@ const ColorInput = ({ className, value, dataset, onChange, ...props }) => {
|
||||||
return parseColor(value);
|
return parseColor(value);
|
||||||
});
|
});
|
||||||
const labelButtonStyle = React.useMemo(() => ({
|
const labelButtonStyle = React.useMemo(() => ({
|
||||||
...props.style,
|
|
||||||
backgroundColor: value
|
backgroundColor: value
|
||||||
}), [props.style, value]);
|
}), [value]);
|
||||||
const labelButtonOnClick = React.useCallback((event) => {
|
const labelButtonOnClick = React.useCallback((event) => {
|
||||||
if (typeof props.onClick === 'function') {
|
if (typeof props.onClick === 'function') {
|
||||||
props.onClick(event);
|
props.onClick(event);
|
||||||
|
|
@ -76,9 +75,11 @@ const ColorInput = ({ className, value, dataset, onChange, ...props }) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
ColorInput.propTypes = {
|
ColorInput.propTypes = {
|
||||||
|
className: PropTypes.string,
|
||||||
value: PropTypes.string,
|
value: PropTypes.string,
|
||||||
dataset: PropTypes.objectOf(String),
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
onChange: PropTypes.func
|
onChange: PropTypes.func,
|
||||||
|
onClick: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = ColorInput;
|
module.exports = ColorInput;
|
||||||
|
|
|
||||||
|
|
@ -27,7 +27,8 @@ Image.propTypes = {
|
||||||
src: PropTypes.string,
|
src: PropTypes.string,
|
||||||
alt: PropTypes.string,
|
alt: PropTypes.string,
|
||||||
fallbackSrc: PropTypes.string,
|
fallbackSrc: PropTypes.string,
|
||||||
renderFallback: PropTypes.func
|
renderFallback: PropTypes.func,
|
||||||
|
onError: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = Image;
|
module.exports = Image;
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,8 @@ const MainNavBar = React.memo(({ className }) => {
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
MainNavBar.displayName = 'MainNavBar';
|
||||||
|
|
||||||
MainNavBar.propTypes = {
|
MainNavBar.propTypes = {
|
||||||
className: PropTypes.string
|
className: PropTypes.string
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -121,8 +121,9 @@ MetaItem.propTypes = {
|
||||||
playIcon: PropTypes.bool,
|
playIcon: PropTypes.bool,
|
||||||
progress: PropTypes.number,
|
progress: PropTypes.number,
|
||||||
options: PropTypes.array,
|
options: PropTypes.array,
|
||||||
dataset: PropTypes.objectOf(String),
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
optionOnSelect: PropTypes.func
|
optionOnSelect: PropTypes.func,
|
||||||
|
onClick: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = MetaItem;
|
module.exports = MetaItem;
|
||||||
|
|
|
||||||
|
|
@ -179,7 +179,7 @@ const MetaPreview = ({ className, compact, name, logo, background, runtime, rele
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
linksGroups.hasOwnProperty(IMDB_LINK_CATEGORY) ?
|
typeof linksGroups[IMDB_LINK_CATEGORY] === 'object' ?
|
||||||
<ActionButton
|
<ActionButton
|
||||||
{...linksGroups[IMDB_LINK_CATEGORY]}
|
{...linksGroups[IMDB_LINK_CATEGORY]}
|
||||||
className={styles['action-button']}
|
className={styles['action-button']}
|
||||||
|
|
@ -191,7 +191,7 @@ const MetaPreview = ({ className, compact, name, logo, background, runtime, rele
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
!compact && linksGroups.hasOwnProperty(SHARE_LINK_CATEGORY) ?
|
!compact && typeof linksGroups[SHARE_LINK_CATEGORY] === 'object' ?
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<ActionButton
|
<ActionButton
|
||||||
className={styles['action-button']}
|
className={styles['action-button']}
|
||||||
|
|
|
||||||
|
|
@ -112,7 +112,7 @@ ModalDialog.propTypes = {
|
||||||
PropTypes.arrayOf(PropTypes.node),
|
PropTypes.arrayOf(PropTypes.node),
|
||||||
PropTypes.node
|
PropTypes.node
|
||||||
]),
|
]),
|
||||||
dataset: PropTypes.objectOf(String),
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
onCloseRequest: PropTypes.func
|
onCloseRequest: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,3 @@
|
||||||
const ModalDialog = require('./ModalDialog');
|
const ModalDialog = require('./ModalDialog');
|
||||||
|
|
||||||
module.exports = ModalDialog;
|
module.exports = ModalDialog;
|
||||||
|
|
|
||||||
|
|
@ -8,7 +8,7 @@ const useBinaryState = require('stremio/common/useBinaryState');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const Multiselect = ({ className, direction, title, disabled, dataset, renderLabelContent, renderLabelText, onOpen, onClose, onSelect, ...props }) => {
|
const Multiselect = ({ className, direction, title, disabled, dataset, renderLabelContent, renderLabelText, onOpen, onClose, onSelect, ...props }) => {
|
||||||
const [menuOpen, openMenu, closeMenu, toggleMenu] = useBinaryState(false);
|
const [menuOpen, , closeMenu, toggleMenu] = useBinaryState(false);
|
||||||
const options = React.useMemo(() => {
|
const options = React.useMemo(() => {
|
||||||
return Array.isArray(props.options) ?
|
return Array.isArray(props.options) ?
|
||||||
props.options.filter((option) => {
|
props.options.filter((option) => {
|
||||||
|
|
@ -143,12 +143,13 @@ Multiselect.propTypes = {
|
||||||
})),
|
})),
|
||||||
selected: PropTypes.arrayOf(PropTypes.string),
|
selected: PropTypes.arrayOf(PropTypes.string),
|
||||||
disabled: PropTypes.bool,
|
disabled: PropTypes.bool,
|
||||||
dataset: PropTypes.objectOf(String),
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
renderLabelContent: PropTypes.func,
|
renderLabelContent: PropTypes.func,
|
||||||
renderLabelText: PropTypes.func,
|
renderLabelText: PropTypes.func,
|
||||||
onOpen: PropTypes.func,
|
onOpen: PropTypes.func,
|
||||||
onClose: PropTypes.func,
|
onClose: PropTypes.func,
|
||||||
onSelect: PropTypes.func
|
onSelect: PropTypes.func,
|
||||||
|
onClick: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = Multiselect;
|
module.exports = Multiselect;
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,7 @@ const useUser = require('stremio/common/useUser');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const NavMenu = ({ className }) => {
|
const NavMenu = ({ className }) => {
|
||||||
const [menuOpen, openMenu, closeMenu, toggleMenu] = useBinaryState(false);
|
const [menuOpen, , closeMenu, toggleMenu] = useBinaryState(false);
|
||||||
const [fullscreen, requestFullscreen, exitFullscreen] = useFullscreen();
|
const [fullscreen, requestFullscreen, exitFullscreen] = useFullscreen();
|
||||||
const [user, logout] = useUser();
|
const [user, logout] = useUser();
|
||||||
const popupLabelOnClick = React.useCallback((event) => {
|
const popupLabelOnClick = React.useCallback((event) => {
|
||||||
|
|
@ -41,7 +41,7 @@ const NavMenu = ({ className }) => {
|
||||||
className={styles['avatar-container']}
|
className={styles['avatar-container']}
|
||||||
style={{
|
style={{
|
||||||
backgroundImage: user === null ?
|
backgroundImage: user === null ?
|
||||||
`url('/images/anonymous.png')`
|
'url(\'/images/anonymous.png\')'
|
||||||
:
|
:
|
||||||
`url('${user.avatar}'), url('/images/default_avatar.png')`
|
`url('${user.avatar}'), url('/images/default_avatar.png')`
|
||||||
}}
|
}}
|
||||||
|
|
|
||||||
|
|
@ -80,7 +80,7 @@ const Notification = ({ className, id, type, name, poster, thumbnail, season, ep
|
||||||
}
|
}
|
||||||
</Button>
|
</Button>
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
Notification.propTypes = {
|
Notification.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,6 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
const PropTypes = require('prop-types');
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const Icon = require('stremio-icons/dom');
|
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const NotificationPlaceholder = ({ className }) => {
|
const NotificationPlaceholder = ({ className }) => {
|
||||||
|
|
@ -20,6 +19,6 @@ const NotificationPlaceholder = ({ className }) => {
|
||||||
|
|
||||||
NotificationPlaceholder.propTypes = {
|
NotificationPlaceholder.propTypes = {
|
||||||
className: PropTypes.string
|
className: PropTypes.string
|
||||||
}
|
};
|
||||||
|
|
||||||
module.exports = NotificationPlaceholder;
|
module.exports = NotificationPlaceholder;
|
||||||
|
|
|
||||||
|
|
@ -57,7 +57,7 @@ const NotificationsList = ({ className, metaItems }) => {
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
NotificationsList.propTypes = {
|
NotificationsList.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
|
|
|
||||||
|
|
@ -5,13 +5,13 @@ const Icon = require('stremio-icons/dom');
|
||||||
const Button = require('stremio/common/Button');
|
const Button = require('stremio/common/Button');
|
||||||
const Popup = require('stremio/common/Popup');
|
const Popup = require('stremio/common/Popup');
|
||||||
const NotificationsList = require('./NotificationsList');
|
const NotificationsList = require('./NotificationsList');
|
||||||
const useNotifications = require('./useNotifications');
|
// const useNotifications = require('./useNotifications');
|
||||||
// const useCatalogs = require('stremio/routes/Board/useCatalogs');
|
// const useCatalogs = require('stremio/routes/Board/useCatalogs');
|
||||||
const useBinaryState = require('stremio/common/useBinaryState');
|
const useBinaryState = require('stremio/common/useBinaryState');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const NotificationsMenu = ({ className, onClearButtonClicked }) => {
|
const NotificationsMenu = ({ className, onClearButtonClicked }) => {
|
||||||
const [menuOpen, openMenu, closeMenu, toggleMenu] = useBinaryState(false);
|
const [menuOpen, , closeMenu, toggleMenu] = useBinaryState(false);
|
||||||
//TODO use useNotifications hook instead of useCatalogs
|
//TODO use useNotifications hook instead of useCatalogs
|
||||||
const metaItems = []; //useCatalogs();
|
const metaItems = []; //useCatalogs();
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -35,6 +35,7 @@ const PaginationInput = ({ className, label, dataset, onSelect, ...props }) => {
|
||||||
PaginationInput.propTypes = {
|
PaginationInput.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
label: PropTypes.string,
|
label: PropTypes.string,
|
||||||
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
onSelect: PropTypes.func
|
onSelect: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,3 @@
|
||||||
const PaginationInput = require('./PaginationInput');
|
const PaginationInput = require('./PaginationInput');
|
||||||
|
|
||||||
module.exports = PaginationInput;
|
module.exports = PaginationInput;
|
||||||
|
|
|
||||||
|
|
@ -96,14 +96,14 @@ const Popup = ({ open, direction, renderLabel, renderMenu, dataset, onCloseReque
|
||||||
:
|
:
|
||||||
null
|
null
|
||||||
});
|
});
|
||||||
}
|
};
|
||||||
|
|
||||||
Popup.propTypes = {
|
Popup.propTypes = {
|
||||||
open: PropTypes.bool,
|
open: PropTypes.bool,
|
||||||
direction: PropTypes.oneOf(['top-left', 'bottom-left', 'top-right', 'bottom-right']),
|
direction: PropTypes.oneOf(['top-left', 'bottom-left', 'top-right', 'bottom-right']),
|
||||||
renderLabel: PropTypes.func.isRequired,
|
renderLabel: PropTypes.func.isRequired,
|
||||||
renderMenu: PropTypes.func.isRequired,
|
renderMenu: PropTypes.func.isRequired,
|
||||||
dataset: PropTypes.objectOf(String),
|
dataset: PropTypes.objectOf(PropTypes.string),
|
||||||
onCloseRequest: PropTypes.func
|
onCloseRequest: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
|
|
@ -30,4 +31,11 @@ const TextInput = React.forwardRef((props, ref) => {
|
||||||
|
|
||||||
TextInput.displayName = 'TextInput';
|
TextInput.displayName = 'TextInput';
|
||||||
|
|
||||||
|
TextInput.propTypes = {
|
||||||
|
className: PropTypes.string,
|
||||||
|
disabled: PropTypes.bool,
|
||||||
|
onKeyDown: PropTypes.func,
|
||||||
|
onSubmit: PropTypes.func
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = TextInput;
|
module.exports = TextInput;
|
||||||
|
|
|
||||||
|
|
@ -8,23 +8,23 @@ const routesRegexp = {
|
||||||
urlParamsNames: []
|
urlParamsNames: []
|
||||||
},
|
},
|
||||||
discover: {
|
discover: {
|
||||||
regexp: /^\/discover(?:\/([^\/]*)\/([^\/]*)\/([^\/]*))?$/,
|
regexp: /^\/discover(?:\/([^/]*)\/([^/]*)\/([^/]*))?$/,
|
||||||
urlParamsNames: ['addonTransportUrl', 'type', 'catalogId']
|
urlParamsNames: ['addonTransportUrl', 'type', 'catalogId']
|
||||||
},
|
},
|
||||||
library: {
|
library: {
|
||||||
regexp: /^\/library(?:\/([^\/]*))?$/,
|
regexp: /^\/library(?:\/([^/]*)\/([^/]*))?$/,
|
||||||
urlParamsNames: ['type']
|
urlParamsNames: ['type', 'sort']
|
||||||
},
|
},
|
||||||
search: {
|
search: {
|
||||||
regexp: /^\/search$/,
|
regexp: /^\/search$/,
|
||||||
urlParamsNames: []
|
urlParamsNames: []
|
||||||
},
|
},
|
||||||
metadetails: {
|
metadetails: {
|
||||||
regexp: /^\/metadetails\/([^\/]*)\/([^\/]*)(?:\/([^\/]*))?$/,
|
regexp: /^\/metadetails\/([^/]*)\/([^/]*)(?:\/([^/]*))?$/,
|
||||||
urlParamsNames: ['type', 'id', 'videoId']
|
urlParamsNames: ['type', 'id', 'videoId']
|
||||||
},
|
},
|
||||||
addons: {
|
addons: {
|
||||||
regexp: /^\/addons(?:\/([^\/]*)\/([^\/]*)\/([^\/]*))?$/,
|
regexp: /^\/addons(?:\/([^/]*)\/([^/]*)\/([^/]*))?$/,
|
||||||
urlParamsNames: ['addonTransportUrl', 'catalogId', 'type']
|
urlParamsNames: ['addonTransportUrl', 'catalogId', 'type']
|
||||||
},
|
},
|
||||||
settings: {
|
settings: {
|
||||||
|
|
@ -32,7 +32,7 @@ const routesRegexp = {
|
||||||
urlParamsNames: []
|
urlParamsNames: []
|
||||||
},
|
},
|
||||||
player: {
|
player: {
|
||||||
regexp: /^\/player\/([^\/]*)\/([^\/]*)\/([^\/]*)\/([^\/]*)$/,
|
regexp: /^\/player\/([^/]*)\/([^/]*)\/([^/]*)\/([^/]*)$/,
|
||||||
urlParamsNames: ['type', 'id', 'videoId', 'stream']
|
urlParamsNames: ['type', 'id', 'videoId', 'stream']
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,60 @@
|
||||||
const useBinaryState = require('stremio/common/useBinaryState');
|
const React = require('react');
|
||||||
|
const { useServices } = require('stremio/services');
|
||||||
|
const useModelState = require('stremio/common/useModelState');
|
||||||
|
|
||||||
const useInLibrary = (id) => {
|
const useInLibrary = (metaItem) => {
|
||||||
const [inLibrary, addToLibrary, removeFromLibrary, toggleInLibrary] = useBinaryState(false);
|
const { core } = useServices();
|
||||||
if (typeof id === 'string') {
|
const initLibraryItemsState = React.useCallback(() => {
|
||||||
return [inLibrary, addToLibrary, removeFromLibrary, toggleInLibrary];
|
return core.getState('library_items');
|
||||||
} else {
|
}, []);
|
||||||
return [false, null, null, null];
|
const libraryItems = useModelState({
|
||||||
}
|
model: 'library_items',
|
||||||
|
init: initLibraryItemsState
|
||||||
|
});
|
||||||
|
const addToLibrary = React.useCallback((metaItem) => {
|
||||||
|
core.dispatch({
|
||||||
|
action: 'UserOp',
|
||||||
|
args: {
|
||||||
|
userOp: 'AddToLibrary',
|
||||||
|
args: {
|
||||||
|
meta_item: metaItem,
|
||||||
|
now: new Date()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
const removeFromLibrary = React.useCallback((id) => {
|
||||||
|
core.dispatch({
|
||||||
|
action: 'UserOp',
|
||||||
|
args: {
|
||||||
|
userOp: 'RemoveFromLibrary',
|
||||||
|
args: {
|
||||||
|
id,
|
||||||
|
now: new Date()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
const inLibrary = React.useMemo(() => {
|
||||||
|
return typeof metaItem === 'object' && metaItem !== null ?
|
||||||
|
libraryItems.ids.includes(metaItem.id)
|
||||||
|
:
|
||||||
|
false;
|
||||||
|
}, [metaItem, libraryItems]);
|
||||||
|
const toggleInLibrary = React.useMemo(() => {
|
||||||
|
if (typeof metaItem !== 'object' || metaItem === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (inLibrary) {
|
||||||
|
removeFromLibrary(metaItem.id);
|
||||||
|
} else {
|
||||||
|
addToLibrary(metaItem);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [metaItem, inLibrary]);
|
||||||
|
return [inLibrary, toggleInLibrary];
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = useInLibrary;
|
module.exports = useInLibrary;
|
||||||
|
|
|
||||||
|
|
@ -95,7 +95,7 @@ const Router = ({ className, onPathNotMatch, ...props }) => {
|
||||||
<div className={classnames(className, 'routes-container')}>
|
<div className={classnames(className, 'routes-container')}>
|
||||||
{
|
{
|
||||||
views
|
views
|
||||||
.filter(view => view !== null)
|
.filter((view) => view !== null)
|
||||||
.map(({ key, component, urlParams, queryParams }, index, views) => (
|
.map(({ key, component, urlParams, queryParams }, index, views) => (
|
||||||
<RouteFocusedProvider key={key} value={index === views.length - 1}>
|
<RouteFocusedProvider key={key} value={index === views.length - 1}>
|
||||||
<Route>
|
<Route>
|
||||||
|
|
|
||||||
|
|
@ -2,24 +2,54 @@ const React = require('react');
|
||||||
const PropTypes = require('prop-types');
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const { Button } = require('stremio/common');
|
const { Button, Image } = require('stremio/common');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const Addon = ({ className, id, name, logo, description, types, version, transportUrl, installed, toggle, onShareButtonClicked }) => {
|
const Addon = ({ className, id, name, version, logo, description, types, installed, onToggle, onShare, dataset }) => {
|
||||||
const onKeyUp = React.useCallback((event) => {
|
const toggleButtonOnClick = React.useCallback((event) => {
|
||||||
if (event.key === 'Enter' && typeof toggle === 'function') {
|
if (typeof onToggle === 'function') {
|
||||||
toggle(event);
|
onToggle({
|
||||||
|
type: 'toggle',
|
||||||
|
nativeEvent: event.nativeEvent,
|
||||||
|
reactEvent: event,
|
||||||
|
dataset: dataset
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}, [toggle]);
|
}, [onToggle, dataset]);
|
||||||
|
const shareButtonOnClick = React.useCallback((event) => {
|
||||||
|
if (typeof onShare === 'function') {
|
||||||
|
onShare({
|
||||||
|
type: 'share',
|
||||||
|
nativeEvent: event.nativeEvent,
|
||||||
|
reactEvent: event,
|
||||||
|
dataset: dataset
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [onShare, dataset]);
|
||||||
|
const onKeyDown = React.useCallback((event) => {
|
||||||
|
if (event.key === 'Enter' && typeof onToggle === 'function') {
|
||||||
|
onToggle({
|
||||||
|
type: 'toggle',
|
||||||
|
nativeEvent: event.nativeEvent,
|
||||||
|
reactEvent: event,
|
||||||
|
dataset: dataset
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [onToggle, dataset]);
|
||||||
|
const renderLogoFallback = React.useMemo(() => () => {
|
||||||
|
return (
|
||||||
|
<Icon className={styles['icon']} icon={'ic_addons'} />
|
||||||
|
);
|
||||||
|
}, []);
|
||||||
return (
|
return (
|
||||||
<Button className={classnames(styles['addon-container'], className)} data-id={id} onKeyUp={onKeyUp}>
|
<Button className={classnames(className, styles['addon-container'])} onKeyDown={onKeyDown}>
|
||||||
<div className={styles['logo-container']}>
|
<div className={styles['logo-container']}>
|
||||||
{
|
<Image
|
||||||
typeof logo === 'string' && logo.length > 0 ?
|
className={styles['logo']}
|
||||||
<img className={styles['logo']} src={logo} alt={' '} />
|
src={logo}
|
||||||
:
|
alt={' '}
|
||||||
<Icon className={styles['icon']} icon={'ic_addons'} />
|
renderFallback={renderLogoFallback}
|
||||||
}
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className={styles['info-container']}>
|
<div className={styles['info-container']}>
|
||||||
<div className={styles['name-container']} title={typeof name === 'string' && name.length > 0 ? name : id}>
|
<div className={styles['name-container']} title={typeof name === 'string' && name.length > 0 ? name : id}>
|
||||||
|
|
@ -32,10 +62,10 @@ const Addon = ({ className, id, name, logo, description, types, version, transpo
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
Array.isArray(types) ?
|
Array.isArray(types) && types.length > 0 ?
|
||||||
<div className={styles['types-container']}>
|
<div className={styles['types-container']}>
|
||||||
{
|
{
|
||||||
types.length <= 1 ?
|
types.length === 1 ?
|
||||||
types.join('')
|
types.join('')
|
||||||
:
|
:
|
||||||
types.slice(0, -1).join(', ') + ' & ' + types[types.length - 1]
|
types.slice(0, -1).join(', ') + ' & ' + types[types.length - 1]
|
||||||
|
|
@ -52,10 +82,10 @@ const Addon = ({ className, id, name, logo, description, types, version, transpo
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
<div className={styles['buttons-container']}>
|
<div className={styles['buttons-container']}>
|
||||||
<Button className={installed ? styles['uninstall-button-container'] : styles['install-button-container']} title={installed ? 'Uninstall' : 'Install'} tabIndex={-1} data-id={id} onClick={toggle}>
|
<Button className={installed ? styles['uninstall-button-container'] : styles['install-button-container']} title={installed ? 'Uninstall' : 'Install'} tabIndex={-1} onClick={toggleButtonOnClick}>
|
||||||
<div className={styles['label']}>{installed ? 'Uninstall' : 'Install'}</div>
|
<div className={styles['label']}>{installed ? 'Uninstall' : 'Install'}</div>
|
||||||
</Button>
|
</Button>
|
||||||
<Button className={styles['share-button-container']} title={'Share addon'} tabIndex={-1} onClick={onShareButtonClicked}>
|
<Button className={styles['share-button-container']} title={'Share addon'} tabIndex={-1} onClick={shareButtonOnClick}>
|
||||||
<Icon className={styles['icon']} icon={'ic_share'} />
|
<Icon className={styles['icon']} icon={'ic_share'} />
|
||||||
<div className={styles['label']}>Share addon</div>
|
<div className={styles['label']}>Share addon</div>
|
||||||
</Button>
|
</Button>
|
||||||
|
|
@ -68,14 +98,14 @@ Addon.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
id: PropTypes.string,
|
id: PropTypes.string,
|
||||||
name: PropTypes.string,
|
name: PropTypes.string,
|
||||||
|
version: PropTypes.string,
|
||||||
logo: PropTypes.string,
|
logo: PropTypes.string,
|
||||||
description: PropTypes.string,
|
description: PropTypes.string,
|
||||||
types: PropTypes.arrayOf(PropTypes.string),
|
types: PropTypes.arrayOf(PropTypes.string),
|
||||||
version: PropTypes.string,
|
|
||||||
transportUrl: PropTypes.string,
|
|
||||||
installed: PropTypes.bool,
|
installed: PropTypes.bool,
|
||||||
toggle: PropTypes.func,
|
onToggle: PropTypes.func,
|
||||||
onShareButtonClicked: PropTypes.func
|
onShare: PropTypes.func,
|
||||||
|
dataset: PropTypes.objectOf(PropTypes.string)
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = Addon;
|
module.exports = Addon;
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,6 @@
|
||||||
.addon-container {
|
.addon-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
flex-wrap: wrap;
|
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
background-color: var(--color-backgroundlighter);
|
background-color: var(--color-backgroundlighter);
|
||||||
|
|
@ -17,6 +16,7 @@
|
||||||
display: block;
|
display: block;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
padding: 0.5rem;
|
||||||
object-fit: contain;
|
object-fit: contain;
|
||||||
object-position: center;
|
object-position: center;
|
||||||
}
|
}
|
||||||
|
|
@ -31,14 +31,13 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.info-container {
|
.info-container {
|
||||||
flex-grow: 1000;
|
flex-grow: 1;
|
||||||
flex-shrink: 1;
|
flex-shrink: 1;
|
||||||
flex-basis: 0;
|
flex-basis: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
align-items: baseline;
|
align-items: baseline;
|
||||||
min-width: 40rem;
|
|
||||||
padding: 0 0.5rem;
|
padding: 0 0.5rem;
|
||||||
|
|
||||||
.name-container {
|
.name-container {
|
||||||
|
|
@ -47,7 +46,7 @@
|
||||||
flex-basis: auto;
|
flex-basis: auto;
|
||||||
padding: 0 0.5rem;
|
padding: 0 0.5rem;
|
||||||
max-height: 3.6em;
|
max-height: 3.6em;
|
||||||
font-size: 1.5rem;
|
font-size: 1.6rem;
|
||||||
color: var(--color-surfacelighter);
|
color: var(--color-surfacelighter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -55,6 +54,7 @@
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
flex-shrink: 1;
|
flex-shrink: 1;
|
||||||
flex-basis: auto;
|
flex-basis: auto;
|
||||||
|
margin-top: 0.5rem;
|
||||||
padding: 0 0.5rem;
|
padding: 0 0.5rem;
|
||||||
max-height: 2.4em;
|
max-height: 2.4em;
|
||||||
color: var(--color-surfacelight);
|
color: var(--color-surfacelight);
|
||||||
|
|
@ -83,22 +83,14 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.buttons-container {
|
.buttons-container {
|
||||||
flex-grow: 1;
|
flex: none;
|
||||||
flex-shrink: 0;
|
width: 17rem;
|
||||||
flex-basis: 0;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
align-items: flex-end;
|
|
||||||
min-width: 17rem;
|
|
||||||
|
|
||||||
.install-button-container, .uninstall-button-container, .share-button-container {
|
.install-button-container, .uninstall-button-container, .share-button-container {
|
||||||
flex: none;
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
width: 17rem;
|
|
||||||
height: 3.5rem;
|
height: 3.5rem;
|
||||||
padding: 0 1rem;
|
padding: 0 1rem;
|
||||||
|
|
||||||
|
|
@ -106,13 +98,8 @@
|
||||||
margin-top: 1rem;
|
margin-top: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:not(:last-child) {
|
|
||||||
margin-right: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.icon {
|
.icon {
|
||||||
flex: none;
|
flex: none;
|
||||||
display: block;
|
|
||||||
width: 1.5rem;
|
width: 1.5rem;
|
||||||
height: 1.5rem;
|
height: 1.5rem;
|
||||||
margin-right: 1rem;
|
margin-right: 1rem;
|
||||||
|
|
|
||||||
|
|
@ -1,49 +1,93 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const { Button, Multiselect, NavBar, TextInput, SharePrompt, ModalDialog } = require('stremio/common');
|
const { Button, Multiselect, NavBar, TextInput, SharePrompt, ModalDialog, useBinaryState } = require('stremio/common');
|
||||||
const Addon = require('./Addon');
|
const Addon = require('./Addon');
|
||||||
const AddonPrompt = require('./AddonPrompt');
|
|
||||||
const useAddons = require('./useAddons');
|
const useAddons = require('./useAddons');
|
||||||
const useSelectedAddon = require('./useSelectedAddon');
|
const useSelectableInputs = require('./useSelectableInputs');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
|
const navigateToAddonDetails = (addonsCatalogRequest, transportUrl) => {
|
||||||
|
const queryParams = new URLSearchParams([['addon', transportUrl]]);
|
||||||
|
if (addonsCatalogRequest !== null) {
|
||||||
|
const addonTransportUrl = encodeURIComponent(addonsCatalogRequest.base);
|
||||||
|
const catalogId = encodeURIComponent(addonsCatalogRequest.path.id);
|
||||||
|
const type = encodeURIComponent(addonsCatalogRequest.path.type_name);
|
||||||
|
window.location.replace(`#/addons/${addonTransportUrl}/${catalogId}/${type}?${queryParams}`);
|
||||||
|
} else {
|
||||||
|
window.location.replace(`#/addons?${queryParams}`);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const Addons = ({ urlParams, queryParams }) => {
|
const Addons = ({ urlParams, queryParams }) => {
|
||||||
const inputRef = React.useRef(null);
|
const addons = useAddons(urlParams);
|
||||||
const [query, setQuery] = React.useState('');
|
const selectInputs = useSelectableInputs(addons);
|
||||||
const queryOnChange = React.useCallback((event) => {
|
const [addAddonModalOpen, openAddAddonModal, closeAddAddonModal] = useBinaryState(false);
|
||||||
setQuery(event.currentTarget.value);
|
const addAddonUrlInputRef = React.useRef(null);
|
||||||
}, []);
|
const addAddonOnSubmit = React.useCallback(() => {
|
||||||
const [[addons, dropdowns, setSelectedAddon, installedAddons, error], installSelectedAddon, uninstallSelectedAddon] = useAddons(urlParams, queryParams);
|
if (addAddonUrlInputRef.current !== null) {
|
||||||
const [addAddonModalOpened, setAddAddonModalOpened] = React.useState(false);
|
const addonsCatalogRequest = addons.catalog_resource !== null ?
|
||||||
const [selectedAddon, clearSelectedAddon] = useSelectedAddon(queryParams.get('addon'));
|
addons.catalog_resource.request
|
||||||
const [sharedAddon, setSharedAddon] = React.useState(null);
|
:
|
||||||
const onAddAddonButtonClicked = React.useCallback(() => {
|
null;
|
||||||
setAddAddonModalOpened(true);
|
navigateToAddonDetails(addonsCatalogRequest, addAddonUrlInputRef.current.value);
|
||||||
}, []);
|
|
||||||
const onAddButtonClicked = React.useCallback(() => {
|
|
||||||
if (inputRef.current.value.length > 0) {
|
|
||||||
setSelectedAddon(inputRef.current.value);
|
|
||||||
setAddAddonModalOpened(false);
|
|
||||||
}
|
}
|
||||||
}, [setSelectedAddon]);
|
}, [addons]);
|
||||||
const installedAddon = React.useCallback((currentAddon) => {
|
const addAddonModalButtons = React.useMemo(() => {
|
||||||
return installedAddons.some((installedAddon) => installedAddon.transportUrl === currentAddon.transportUrl);
|
return [
|
||||||
}, [installedAddons]);
|
{
|
||||||
const toggleAddon = React.useCallback(() => {
|
className: styles['cancel-button'],
|
||||||
installedAddon(selectedAddon) ? uninstallSelectedAddon(selectedAddon) : installSelectedAddon(selectedAddon);
|
label: 'Cancel',
|
||||||
clearSelectedAddon();
|
props: {
|
||||||
}, [selectedAddon]);
|
onClick: closeAddAddonModal
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Add',
|
||||||
|
props: {
|
||||||
|
onClick: addAddonOnSubmit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}, [addAddonOnSubmit]);
|
||||||
|
const [search, setSearch] = React.useState('');
|
||||||
|
const searchInputOnChange = React.useCallback((event) => {
|
||||||
|
setSearch(event.currentTarget.value);
|
||||||
|
}, []);
|
||||||
|
const [sharedTransportUrl, setSharedTransportUrl] = React.useState(null);
|
||||||
|
const clearSharedTransportUrl = React.useCallback(() => {
|
||||||
|
setSharedTransportUrl(null);
|
||||||
|
}, []);
|
||||||
|
const onAddonShare = React.useCallback((event) => {
|
||||||
|
setSharedTransportUrl(event.dataset.transportUrl);
|
||||||
|
}, []);
|
||||||
|
const onAddonToggle = React.useCallback((event) => {
|
||||||
|
const addonsCatalogRequest = addons.catalog_resource !== null ?
|
||||||
|
addons.catalog_resource.request
|
||||||
|
:
|
||||||
|
null;
|
||||||
|
navigateToAddonDetails(addonsCatalogRequest, event.dataset.transportUrl);
|
||||||
|
}, [addons]);
|
||||||
|
React.useLayoutEffect(() => {
|
||||||
|
closeAddAddonModal();
|
||||||
|
setSearch('');
|
||||||
|
clearSharedTransportUrl();
|
||||||
|
}, [urlParams, queryParams]);
|
||||||
return (
|
return (
|
||||||
<div className={styles['addons-container']}>
|
<div className={styles['addons-container']}>
|
||||||
<NavBar className={styles['nav-bar']} backButton={true} title={'Addons'} />
|
<NavBar className={styles['nav-bar']} backButton={true} title={'Addons'} />
|
||||||
<div className={styles['addons-content']}>
|
<div className={styles['addons-content']}>
|
||||||
<div className={styles['top-bar-container']}>
|
<div className={styles['selectable-inputs-container']}>
|
||||||
<Button className={styles['add-button-container']} title={'Add addon'} onClick={onAddAddonButtonClicked}>
|
<Button className={styles['add-button-container']} title={'Add addon'} onClick={openAddAddonModal}>
|
||||||
<Icon className={styles['icon']} icon={'ic_plus'} />
|
<Icon className={styles['icon']} icon={'ic_plus'} />
|
||||||
<div className={styles['add-button-label']}>Add addon</div>
|
<div className={styles['add-button-label']}>Add addon</div>
|
||||||
</Button>
|
</Button>
|
||||||
{dropdowns.map((dropdown, index) => (
|
{selectInputs.map((selectInput, index) => (
|
||||||
<Multiselect {...dropdown} key={index} className={styles['dropdown']} />
|
<Multiselect
|
||||||
|
{...selectInput}
|
||||||
|
key={index}
|
||||||
|
className={styles['select-input-container']}
|
||||||
|
/>
|
||||||
))}
|
))}
|
||||||
<label className={styles['search-bar-container']}>
|
<label className={styles['search-bar-container']}>
|
||||||
<Icon className={styles['icon']} icon={'ic_search'} />
|
<Icon className={styles['icon']} icon={'ic_search'} />
|
||||||
|
|
@ -51,116 +95,105 @@ const Addons = ({ urlParams, queryParams }) => {
|
||||||
className={styles['search-input']}
|
className={styles['search-input']}
|
||||||
type={'text'}
|
type={'text'}
|
||||||
placeholder={'Search addons...'}
|
placeholder={'Search addons...'}
|
||||||
value={query}
|
value={search}
|
||||||
onChange={queryOnChange}
|
onChange={searchInputOnChange}
|
||||||
/>
|
/>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
<div className={styles['addons-list-container']}>
|
{
|
||||||
{
|
addons.selectable.catalogs.length === 0 && addons.catalog_resource === null ?
|
||||||
error !== null ?
|
<div className={styles['message-container']}>
|
||||||
|
No addons
|
||||||
|
</div>
|
||||||
|
:
|
||||||
|
addons.catalog_resource === null ?
|
||||||
<div className={styles['message-container']}>
|
<div className={styles['message-container']}>
|
||||||
{error.type}{error.type === 'Other' ? ` - ${error.content}` : null}
|
No select
|
||||||
</div>
|
</div>
|
||||||
:
|
:
|
||||||
Array.isArray(addons) ?
|
addons.catalog_resource.content.type === 'Err' ?
|
||||||
addons.filter((addon) => query.length === 0 ||
|
|
||||||
((typeof addon.manifest.name === 'string' && addon.manifest.name.toLowerCase().includes(query.toLowerCase())) ||
|
|
||||||
(typeof addon.manifest.description === 'string' && addon.manifest.description.toLowerCase().includes(query.toLowerCase()))
|
|
||||||
))
|
|
||||||
.map((addon, index) => (
|
|
||||||
<Addon
|
|
||||||
{...addon.manifest}
|
|
||||||
key={index}
|
|
||||||
installed={installedAddon(addon)}
|
|
||||||
className={styles['addon']}
|
|
||||||
toggle={() => setSelectedAddon(addon.transportUrl)}
|
|
||||||
onShareButtonClicked={() => setSharedAddon(addon)}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
:
|
|
||||||
<div className={styles['message-container']}>
|
<div className={styles['message-container']}>
|
||||||
Loading
|
Addons could not be loaded
|
||||||
</div>
|
</div>
|
||||||
}
|
:
|
||||||
</div>
|
addons.catalog_resource.content.type === 'Loading' ?
|
||||||
{
|
<div className={styles['message-container']}>
|
||||||
addAddonModalOpened ?
|
Loading
|
||||||
<ModalDialog
|
</div>
|
||||||
className={styles['add-addon-prompt-container']}
|
:
|
||||||
title={'Add addon'}
|
<div className={styles['addons-list-container']}>
|
||||||
buttons={[
|
{
|
||||||
{
|
addons.catalog_resource.content.content
|
||||||
label: 'Cancel',
|
.filter((addon) => {
|
||||||
className: styles['cancel-button'],
|
return search.length === 0 ||
|
||||||
props: {
|
(
|
||||||
title: 'Cancel',
|
(typeof addon.manifest.name === 'string' && addon.manifest.name.toLowerCase().includes(search.toLowerCase())) ||
|
||||||
onClick: () => setAddAddonModalOpened(false)
|
(typeof addon.manifest.description === 'string' && addon.manifest.description.toLowerCase().includes(search.toLowerCase()))
|
||||||
}
|
);
|
||||||
},
|
})
|
||||||
{
|
.map((addon, index) => (
|
||||||
label: 'Add',
|
<Addon
|
||||||
props: {
|
key={index}
|
||||||
title: 'Add',
|
className={styles['addon']}
|
||||||
onClick: onAddButtonClicked
|
id={addon.manifest.id}
|
||||||
}
|
name={addon.manifest.name}
|
||||||
}
|
version={addon.manifest.version}
|
||||||
]}
|
logo={addon.manifest.logo}
|
||||||
onCloseRequest={() => setAddAddonModalOpened(false)}
|
description={addon.manifest.description}
|
||||||
>
|
types={addon.manifest.types}
|
||||||
<TextInput ref={inputRef} className={styles['url-content']} type={'text'} tabIndex={'-1'} placeholder={'Paste url...'} />
|
installed={addon.installed}
|
||||||
</ModalDialog>
|
onToggle={onAddonToggle}
|
||||||
:
|
onShare={onAddonShare}
|
||||||
null
|
dataset={{ transportUrl: addon.transportUrl }}
|
||||||
}
|
/>
|
||||||
{
|
))
|
||||||
selectedAddon !== null ?
|
}
|
||||||
<ModalDialog
|
</div>
|
||||||
className={styles['addon-prompt-container']}
|
|
||||||
buttons={[
|
|
||||||
{
|
|
||||||
label: 'Cancel',
|
|
||||||
className: styles['cancel-button'],
|
|
||||||
props: {
|
|
||||||
title: 'Cancel',
|
|
||||||
onClick: clearSelectedAddon
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: installedAddon(selectedAddon) ? 'Uninstall' : 'Install',
|
|
||||||
props: {
|
|
||||||
title: installedAddon(selectedAddon) ? 'Uninstall' : 'Install',
|
|
||||||
onClick: toggleAddon
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
onCloseRequest={clearSelectedAddon}
|
|
||||||
>
|
|
||||||
<AddonPrompt
|
|
||||||
{...selectedAddon.manifest}
|
|
||||||
transportUrl={selectedAddon.transportUrl}
|
|
||||||
installed={installedAddon(selectedAddon)}
|
|
||||||
official={selectedAddon.flags.official}
|
|
||||||
cancel={clearSelectedAddon}
|
|
||||||
/>
|
|
||||||
</ModalDialog>
|
|
||||||
:
|
|
||||||
null
|
|
||||||
}
|
|
||||||
{
|
|
||||||
sharedAddon !== null ?
|
|
||||||
<ModalDialog className={styles['share-prompt-container']} title={'Share addon'} onCloseRequest={() => setSharedAddon(null)}>
|
|
||||||
<SharePrompt
|
|
||||||
url={sharedAddon.transportUrl}
|
|
||||||
close={() => setSharedAddon(null)}
|
|
||||||
/>
|
|
||||||
</ModalDialog>
|
|
||||||
:
|
|
||||||
null
|
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
{
|
||||||
|
addAddonModalOpen ?
|
||||||
|
<ModalDialog
|
||||||
|
className={styles['add-addon-modal-container']}
|
||||||
|
title={'Add addon'}
|
||||||
|
buttons={addAddonModalButtons}
|
||||||
|
onCloseRequest={closeAddAddonModal}>
|
||||||
|
<TextInput
|
||||||
|
ref={addAddonUrlInputRef}
|
||||||
|
className={styles['addon-url-input']}
|
||||||
|
type={'text'}
|
||||||
|
placeholder={'Paste url...'}
|
||||||
|
onSubmit={addAddonOnSubmit}
|
||||||
|
/>
|
||||||
|
</ModalDialog>
|
||||||
|
:
|
||||||
|
null
|
||||||
|
}
|
||||||
|
{
|
||||||
|
typeof sharedTransportUrl === 'string' ?
|
||||||
|
<ModalDialog
|
||||||
|
className={styles['share-modal-container']}
|
||||||
|
title={'Share addon'}
|
||||||
|
onCloseRequest={clearSharedTransportUrl}>
|
||||||
|
<SharePrompt
|
||||||
|
className={styles['share-prompt-container']}
|
||||||
|
url={sharedTransportUrl}
|
||||||
|
/>
|
||||||
|
</ModalDialog>
|
||||||
|
:
|
||||||
|
null
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Addons.propTypes = {
|
||||||
|
urlParams: PropTypes.exact({
|
||||||
|
addonTransportUrl: PropTypes.string,
|
||||||
|
catalogId: PropTypes.string,
|
||||||
|
type: PropTypes.string
|
||||||
|
}),
|
||||||
|
queryParams: PropTypes.instanceOf(URLSearchParams)
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Addons;
|
module.exports = Addons;
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,7 @@
|
||||||
|
:import('~stremio/common/Multiselect/styles.less') {
|
||||||
|
multiselect-menu-container: menu-container;
|
||||||
|
}
|
||||||
|
|
||||||
.addons-container {
|
.addons-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
@ -16,11 +20,12 @@
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
||||||
.top-bar-container {
|
.selectable-inputs-container {
|
||||||
flex: none;
|
flex: none;
|
||||||
|
align-self: stretch;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
margin: 2rem;
|
padding: 1.5rem;
|
||||||
overflow: visible;
|
overflow: visible;
|
||||||
|
|
||||||
.add-button-container {
|
.add-button-container {
|
||||||
|
|
@ -30,7 +35,7 @@
|
||||||
align-items: center;
|
align-items: center;
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
max-width: 15rem;
|
max-width: 15rem;
|
||||||
margin-right: 1rem;
|
margin-right: 1.5rem;
|
||||||
padding: 0 1rem;
|
padding: 0 1rem;
|
||||||
background-color: var(--color-signal5);
|
background-color: var(--color-signal5);
|
||||||
|
|
||||||
|
|
@ -40,8 +45,8 @@
|
||||||
|
|
||||||
.icon {
|
.icon {
|
||||||
flex: none;
|
flex: none;
|
||||||
width: 1.5rem;
|
width: 1.2rem;
|
||||||
height: 1.5rem;
|
height: 1.2rem;
|
||||||
margin-right: 1rem;
|
margin-right: 1rem;
|
||||||
fill: var(--color-surfacelighter);
|
fill: var(--color-surfacelighter);
|
||||||
}
|
}
|
||||||
|
|
@ -56,12 +61,17 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.dropdown {
|
.select-input-container {
|
||||||
flex-grow: 0;
|
flex-grow: 0;
|
||||||
flex-shrink: 1;
|
flex-shrink: 1;
|
||||||
flex-basis: 15rem;
|
flex-basis: 15rem;
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
margin-right: 1rem;
|
margin-right: 1.5rem;
|
||||||
|
|
||||||
|
.multiselect-menu-container {
|
||||||
|
max-height: calc(3.2rem * 7);
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-bar-container {
|
.search-bar-container {
|
||||||
|
|
@ -72,7 +82,6 @@
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
margin-right: 1rem;
|
|
||||||
padding: 0 1rem;
|
padding: 0 1rem;
|
||||||
background-color: var(--color-backgroundlighter);
|
background-color: var(--color-backgroundlighter);
|
||||||
cursor: text;
|
cursor: text;
|
||||||
|
|
@ -82,7 +91,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon {
|
.icon {
|
||||||
display: block;
|
flex: none;
|
||||||
width: 1.2rem;
|
width: 1.2rem;
|
||||||
height: 1.2rem;
|
height: 1.2rem;
|
||||||
margin-right: 1rem;
|
margin-right: 1rem;
|
||||||
|
|
@ -91,7 +100,6 @@
|
||||||
|
|
||||||
.search-input {
|
.search-input {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
align-self: stretch;
|
|
||||||
color: var(--color-surfacelighter);
|
color: var(--color-surfacelighter);
|
||||||
|
|
||||||
&::placeholder {
|
&::placeholder {
|
||||||
|
|
@ -103,32 +111,31 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.message-container {
|
||||||
|
flex: 1;
|
||||||
|
align-self: stretch;
|
||||||
|
padding: 0 1.5rem;
|
||||||
|
font-size: 2rem;
|
||||||
|
color: var(--color-surfacelighter);
|
||||||
|
}
|
||||||
|
|
||||||
.addons-list-container {
|
.addons-list-container {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
align-self: stretch;
|
align-self: stretch;
|
||||||
padding: 0 2rem;
|
padding: 0 1.5rem;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
|
|
||||||
.addon {
|
.addon {
|
||||||
width: 100%;
|
margin-bottom: 1.5rem;
|
||||||
margin-bottom: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.message-container {
|
|
||||||
padding: 0 2rem;
|
|
||||||
font-size: 2rem;
|
|
||||||
color: var(--color-surfacelighter);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.add-addon-prompt-container {
|
.add-addon-modal-container {
|
||||||
.url-content {
|
.addon-url-input {
|
||||||
flex: 1;
|
width: 25rem;
|
||||||
width: 100%;
|
padding: 0.5rem 1rem;
|
||||||
padding: 0.5rem;
|
|
||||||
font-size: 0.9rem;
|
|
||||||
color: var(--color-surfacedark);
|
color: var(--color-surfacedark);
|
||||||
border: thin solid var(--color-surface);
|
border: thin solid var(--color-surface);
|
||||||
}
|
}
|
||||||
|
|
@ -138,8 +145,8 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.addon-prompt-container {
|
.share-modal-container {
|
||||||
.cancel-button {
|
.share-prompt-container {
|
||||||
background-color: var(--color-surfacedark);
|
width: 25rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
48
src/routes/Addons/useAddonDetails.js
Normal file
48
src/routes/Addons/useAddonDetails.js
Normal file
|
|
@ -0,0 +1,48 @@
|
||||||
|
const React = require('react');
|
||||||
|
const { useModelState } = require('stremio/common');
|
||||||
|
|
||||||
|
const initAddonDetailsState = () => ({
|
||||||
|
descriptor: null
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapAddonDetailsStateWithCtx = (addonDetails, ctx) => {
|
||||||
|
const descriptor = addonDetails.descriptor !== null && addonDetails.descriptor.content.type === 'Ready' ?
|
||||||
|
{
|
||||||
|
...addonDetails.descriptor,
|
||||||
|
content: {
|
||||||
|
...addonDetails.descriptor.content,
|
||||||
|
installed: ctx.content.addons.some((addon) => addon.transportUrl === addonDetails.descriptor.transport_url),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:
|
||||||
|
addonDetails.descriptor;
|
||||||
|
return { descriptor };
|
||||||
|
};
|
||||||
|
|
||||||
|
const useAddonDetails = (queryParams) => {
|
||||||
|
const loadAddonDetailsAction = React.useMemo(() => {
|
||||||
|
if (queryParams.has('addon')) {
|
||||||
|
return {
|
||||||
|
action: 'Load',
|
||||||
|
args: {
|
||||||
|
load: 'AddonDetails',
|
||||||
|
args: {
|
||||||
|
transport_url: queryParams.get('addon')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
action: 'Unload'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [queryParams]);
|
||||||
|
return useModelState({
|
||||||
|
model: 'addon_details',
|
||||||
|
action: loadAddonDetailsAction,
|
||||||
|
mapWithCtx: mapAddonDetailsStateWithCtx,
|
||||||
|
init: initAddonDetailsState,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = useAddonDetails;
|
||||||
|
|
@ -15,8 +15,28 @@ const initAddonsState = () => ({
|
||||||
|
|
||||||
const mapAddonsStateWithCtx = (addons, ctx) => {
|
const mapAddonsStateWithCtx = (addons, ctx) => {
|
||||||
const selectable = addons.selectable;
|
const selectable = addons.selectable;
|
||||||
const catalog_resource = addons.catalog_resource;
|
// TODO replace catalog content if resource catalog id is MY
|
||||||
// TODO add MY catalogId replace catalog content if resource catalog id is MY
|
const catalog_resource = addons.catalog_resource !== null && addons.catalog_resource.content.type === 'Ready' ?
|
||||||
|
{
|
||||||
|
...addons.catalog_resource,
|
||||||
|
content: {
|
||||||
|
...addons.catalog_resource.content,
|
||||||
|
content: addons.catalog_resource.content.content.map((descriptor) => ({
|
||||||
|
transportUrl: descriptor.transportUrl,
|
||||||
|
installed: ctx.content.addons.some((addon) => addon.transportUrl === descriptor.transportUrl),
|
||||||
|
manifest: {
|
||||||
|
id: descriptor.manifest.id,
|
||||||
|
name: descriptor.manifest.name,
|
||||||
|
version: descriptor.manifest.version,
|
||||||
|
logo: descriptor.manifest.logo,
|
||||||
|
description: descriptor.manifest.description,
|
||||||
|
types: descriptor.manifest.types
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:
|
||||||
|
addons.catalog_resource;
|
||||||
return { selectable, catalog_resource };
|
return { selectable, catalog_resource };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -15,18 +15,6 @@ const equalWithouExtra = (request1, request2) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const mapSelectableInputs = (addons) => {
|
const mapSelectableInputs = (addons) => {
|
||||||
const selectedCatalogRequest = addons.catalog_resource !== null ?
|
|
||||||
addons.catalog_resource.request
|
|
||||||
:
|
|
||||||
{
|
|
||||||
base: null,
|
|
||||||
path: {
|
|
||||||
resource: 'addon_catalog',
|
|
||||||
id: null,
|
|
||||||
type_name: null,
|
|
||||||
extra: []
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const catalogSelect = {
|
const catalogSelect = {
|
||||||
title: 'Select catalog',
|
title: 'Select catalog',
|
||||||
options: addons.selectable.catalogs
|
options: addons.selectable.catalogs
|
||||||
|
|
@ -36,7 +24,8 @@ const mapSelectableInputs = (addons) => {
|
||||||
})),
|
})),
|
||||||
selected: addons.selectable.catalogs
|
selected: addons.selectable.catalogs
|
||||||
.filter(({ load_request: { path: { id } } }) => {
|
.filter(({ load_request: { path: { id } } }) => {
|
||||||
return id === selectedCatalogRequest.path.id;
|
return addons.catalog_resource !== null &&
|
||||||
|
addons.catalog_resource.request.path.id === id;
|
||||||
})
|
})
|
||||||
.map(({ load_request }) => JSON.stringify(load_request)),
|
.map(({ load_request }) => JSON.stringify(load_request)),
|
||||||
onSelect: (event) => {
|
onSelect: (event) => {
|
||||||
|
|
@ -52,7 +41,8 @@ const mapSelectableInputs = (addons) => {
|
||||||
})),
|
})),
|
||||||
selected: addons.selectable.types
|
selected: addons.selectable.types
|
||||||
.filter(({ load_request }) => {
|
.filter(({ load_request }) => {
|
||||||
return equalWithouExtra(load_request, selectedCatalogRequest);
|
return addons.catalog_resource !== null &&
|
||||||
|
equalWithouExtra(addons.catalog_resource.request, load_request);
|
||||||
})
|
})
|
||||||
.map(({ load_request }) => JSON.stringify(load_request)),
|
.map(({ load_request }) => JSON.stringify(load_request)),
|
||||||
onSelect: (event) => {
|
onSelect: (event) => {
|
||||||
|
|
|
||||||
|
|
@ -1,35 +0,0 @@
|
||||||
const React = require('react');
|
|
||||||
const UrlUtils = require('url');
|
|
||||||
const { routesRegexp, useLocationHash, useRouteActive } = require('stremio/common');
|
|
||||||
|
|
||||||
const useSelectedAddon = (transportUrl) => {
|
|
||||||
const [addon, setAddon] = React.useState(null);
|
|
||||||
const locationHash = useLocationHash();
|
|
||||||
const active = useRouteActive(routesRegexp.addons.regexp);
|
|
||||||
React.useEffect(() => {
|
|
||||||
if (typeof transportUrl !== 'string') {
|
|
||||||
setAddon(null);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
fetch(transportUrl) // TODO
|
|
||||||
.then((resp) => resp.json())
|
|
||||||
.then((manifest) => setAddon({ manifest, transportUrl, flags: {} }));
|
|
||||||
}, [transportUrl]);
|
|
||||||
const clear = React.useCallback(() => {
|
|
||||||
if (active) {
|
|
||||||
const { pathname, search } = UrlUtils.parse(locationHash.slice(1));
|
|
||||||
const queryParams = new URLSearchParams(search || '');
|
|
||||||
queryParams.delete('addon');
|
|
||||||
if ([...queryParams].length !== 0) {
|
|
||||||
window.location.replace(`#${pathname}?${queryParams.toString()}`);
|
|
||||||
} else {
|
|
||||||
window.location.replace(`#${pathname}`);
|
|
||||||
}
|
|
||||||
setAddon(null);
|
|
||||||
}
|
|
||||||
}, [active, locationHash]);
|
|
||||||
return [addon, clear, setAddon];
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = useSelectedAddon;
|
|
||||||
|
|
@ -32,7 +32,7 @@ const Board = () => {
|
||||||
{board.catalog_resources.map((catalog_resource, index) => {
|
{board.catalog_resources.map((catalog_resource, index) => {
|
||||||
const title = `${catalog_resource.addon_name} - ${catalog_resource.request.path.id} ${catalog_resource.request.path.type_name}`;
|
const title = `${catalog_resource.addon_name} - ${catalog_resource.request.path.id} ${catalog_resource.request.path.type_name}`;
|
||||||
switch (catalog_resource.content.type) {
|
switch (catalog_resource.content.type) {
|
||||||
case 'Ready':
|
case 'Ready': {
|
||||||
return (
|
return (
|
||||||
<MetaRow
|
<MetaRow
|
||||||
key={index}
|
key={index}
|
||||||
|
|
@ -43,7 +43,8 @@ const Board = () => {
|
||||||
limit={10}
|
limit={10}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
case 'Err':
|
}
|
||||||
|
case 'Err': {
|
||||||
const message = `Error(${catalog_resource.content.content.type})${typeof catalog_resource.content.content.content === 'string' ? ` - ${catalog_resource.content.content.content}` : ''}`;
|
const message = `Error(${catalog_resource.content.content.type})${typeof catalog_resource.content.content.content === 'string' ? ` - ${catalog_resource.content.content.content}` : ''}`;
|
||||||
return (
|
return (
|
||||||
<MetaRow
|
<MetaRow
|
||||||
|
|
@ -54,7 +55,8 @@ const Board = () => {
|
||||||
limit={10}
|
limit={10}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
case 'Loading':
|
}
|
||||||
|
case 'Loading': {
|
||||||
return (
|
return (
|
||||||
<MetaRow.Placeholder
|
<MetaRow.Placeholder
|
||||||
key={index}
|
key={index}
|
||||||
|
|
@ -63,6 +65,7 @@ const Board = () => {
|
||||||
limit={10}
|
limit={10}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,7 @@ const DISMISS_OPTION = {
|
||||||
value: 'dismiss'
|
value: 'dismiss'
|
||||||
};
|
};
|
||||||
|
|
||||||
const onSelect = (event) => {
|
const onSelect = () => {
|
||||||
// TODO {{event.value}} {{event.dataset}}
|
// TODO {{event.value}} {{event.dataset}}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const { Button, MainNavBar, MetaItem, MetaPreview, Multiselect, ModalDialog, PaginationInput, useBinaryState } = require('stremio/common');
|
const { Button, MainNavBar, MetaItem, MetaPreview, Multiselect, ModalDialog, PaginationInput, useBinaryState } = require('stremio/common');
|
||||||
|
|
@ -135,4 +136,13 @@ const Discover = ({ urlParams, queryParams }) => {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Discover.propTypes = {
|
||||||
|
urlParams: PropTypes.exact({
|
||||||
|
addonTransportUrl: PropTypes.string,
|
||||||
|
type: PropTypes.string,
|
||||||
|
catalogId: PropTypes.string
|
||||||
|
}),
|
||||||
|
queryParams: PropTypes.instanceOf(URLSearchParams)
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Discover;
|
module.exports = Discover;
|
||||||
|
|
|
||||||
|
|
@ -4,16 +4,20 @@ const classnames = require('classnames');
|
||||||
const { Button, Checkbox } = require('stremio/common');
|
const { Button, Checkbox } = require('stremio/common');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const ConsentCheckbox = React.forwardRef(({ className, checked, label, link, href, toggle, ...props }, ref) => {
|
const ConsentCheckbox = React.forwardRef(({ className, checked, label, link, href, onToggle, ...props }, ref) => {
|
||||||
const checkboxOnClick = React.useCallback((event) => {
|
const checkboxOnClick = React.useCallback((event) => {
|
||||||
if (typeof props.onClick === 'function') {
|
if (typeof props.onClick === 'function') {
|
||||||
props.onClick(event);
|
props.onClick(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!event.nativeEvent.togglePrevented && typeof toggle === 'function') {
|
if (!event.nativeEvent.togglePrevented && typeof onToggle === 'function') {
|
||||||
toggle(event);
|
onToggle({
|
||||||
|
type: 'toggle',
|
||||||
|
reactEvent: event,
|
||||||
|
nativeEvent: event.nativeEvent
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}, [toggle]);
|
}, [onToggle]);
|
||||||
const linkOnClick = React.useCallback((event) => {
|
const linkOnClick = React.useCallback((event) => {
|
||||||
event.nativeEvent.togglePrevented = true;
|
event.nativeEvent.togglePrevented = true;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
@ -42,7 +46,8 @@ ConsentCheckbox.propTypes = {
|
||||||
label: PropTypes.string,
|
label: PropTypes.string,
|
||||||
link: PropTypes.string,
|
link: PropTypes.string,
|
||||||
href: PropTypes.string,
|
href: PropTypes.string,
|
||||||
toggle: PropTypes.func
|
onToggle: PropTypes.func,
|
||||||
|
onClick: PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = ConsentCheckbox;
|
module.exports = ConsentCheckbox;
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const { TextInput } = require('stremio/common');
|
const { TextInput } = require('stremio/common');
|
||||||
|
|
||||||
const CredentialsTextInput = React.forwardRef((props, ref) => {
|
const CredentialsTextInput = React.forwardRef((props, ref) => {
|
||||||
|
|
@ -23,4 +24,8 @@ const CredentialsTextInput = React.forwardRef((props, ref) => {
|
||||||
|
|
||||||
CredentialsTextInput.displayName = 'CredentialsTextInput';
|
CredentialsTextInput.displayName = 'CredentialsTextInput';
|
||||||
|
|
||||||
|
CredentialsTextInput.propTypes = {
|
||||||
|
onKeyDown: PropTypes.func
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = CredentialsTextInput;
|
module.exports = CredentialsTextInput;
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const { useRouteFocused } = require('stremio-router');
|
const { useRouteFocused } = require('stremio-router');
|
||||||
|
|
@ -73,14 +74,17 @@ const Intro = ({ queryParams }) => {
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
const onEvent = ({ event, args }) => {
|
const onEvent = ({ event, args }) => {
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case 'CtxActionErr':
|
case 'CtxActionErr': {
|
||||||
const [_action, error] = args;
|
const [, error] = args;
|
||||||
dispatch({ type: 'error', error: error.args.message });
|
dispatch({ type: 'error', error: error.args.message });
|
||||||
case 'CtxChanged':
|
break;
|
||||||
|
}
|
||||||
|
case 'CtxChanged': {
|
||||||
const state = core.getState();
|
const state = core.getState();
|
||||||
if (state.ctx.content.auth !== null) {
|
if (state.ctx.content.auth !== null) {
|
||||||
window.location.replace('#/');
|
window.location.replace('#/');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
if (routeFocused) {
|
if (routeFocused) {
|
||||||
|
|
@ -113,9 +117,7 @@ const Intro = ({ queryParams }) => {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch(() => { });
|
||||||
console.error(err);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, [state.email, state.password]);
|
}, [state.email, state.password]);
|
||||||
|
|
@ -256,7 +258,7 @@ const Intro = ({ queryParams }) => {
|
||||||
<Icon className={styles['icon']} icon={'ic_facebook'} />
|
<Icon className={styles['icon']} icon={'ic_facebook'} />
|
||||||
<div className={styles['label']}>Continue with Facebook</div>
|
<div className={styles['label']}>Continue with Facebook</div>
|
||||||
</Button>
|
</Button>
|
||||||
<div className={styles['facebook-statement']}>We won't post anything on your behalf</div>
|
<div className={styles['facebook-statement']}>We won't post anything on your behalf</div>
|
||||||
<CredentialsTextInput
|
<CredentialsTextInput
|
||||||
ref={emailRef}
|
ref={emailRef}
|
||||||
className={styles['credentials-text-input']}
|
className={styles['credentials-text-input']}
|
||||||
|
|
@ -294,7 +296,7 @@ const Intro = ({ queryParams }) => {
|
||||||
link={'Terms and conditions'}
|
link={'Terms and conditions'}
|
||||||
href={'https://www.stremio.com/tos'}
|
href={'https://www.stremio.com/tos'}
|
||||||
checked={state.termsAccepted}
|
checked={state.termsAccepted}
|
||||||
toggle={toggleTermsAccepted}
|
onToggle={toggleTermsAccepted}
|
||||||
/>
|
/>
|
||||||
<ConsentCheckbox
|
<ConsentCheckbox
|
||||||
ref={privacyPolicyRef}
|
ref={privacyPolicyRef}
|
||||||
|
|
@ -303,14 +305,14 @@ const Intro = ({ queryParams }) => {
|
||||||
link={'Privacy Policy'}
|
link={'Privacy Policy'}
|
||||||
href={'https://www.stremio.com/privacy'}
|
href={'https://www.stremio.com/privacy'}
|
||||||
checked={state.privacyPolicyAccepted}
|
checked={state.privacyPolicyAccepted}
|
||||||
toggle={togglePrivacyPolicyAccepted}
|
onToggle={togglePrivacyPolicyAccepted}
|
||||||
/>
|
/>
|
||||||
<ConsentCheckbox
|
<ConsentCheckbox
|
||||||
ref={marketingRef}
|
ref={marketingRef}
|
||||||
className={styles['consent-checkbox']}
|
className={styles['consent-checkbox']}
|
||||||
label={'I agree to receive marketing communications from Stremio'}
|
label={'I agree to receive marketing communications from Stremio'}
|
||||||
checked={state.marketingAccepted}
|
checked={state.marketingAccepted}
|
||||||
toggle={toggleMarketingAccepted}
|
onToggle={toggleMarketingAccepted}
|
||||||
/>
|
/>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
:
|
:
|
||||||
|
|
@ -343,4 +345,8 @@ const Intro = ({ queryParams }) => {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Intro.propTypes = {
|
||||||
|
queryParams: PropTypes.instanceOf(URLSearchParams)
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Intro;
|
module.exports = Intro;
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const classnames = require('classnames');
|
const classnames = require('classnames');
|
||||||
const { Button, Multiselect, MainNavBar, MetaItem } = require('stremio/common');
|
const { Button, Multiselect, MainNavBar, MetaItem } = require('stremio/common');
|
||||||
const useLibrary = require('./useLibrary');
|
const useLibrary = require('./useLibrary');
|
||||||
|
|
@ -6,8 +7,8 @@ const useSelectableInputs = require('./useSelectableInputs');
|
||||||
const useItemOptions = require('./useItemOptions');
|
const useItemOptions = require('./useItemOptions');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const Library = ({ urlParams, queryParams }) => {
|
const Library = ({ urlParams }) => {
|
||||||
const library = useLibrary(urlParams, queryParams);
|
const library = useLibrary(urlParams);
|
||||||
const [typeSelect, sortPropSelect] = useSelectableInputs(library);
|
const [typeSelect, sortPropSelect] = useSelectableInputs(library);
|
||||||
const [options, optionOnSelect] = useItemOptions();
|
const [options, optionOnSelect] = useItemOptions();
|
||||||
return (
|
return (
|
||||||
|
|
@ -67,6 +68,13 @@ const Library = ({ urlParams, queryParams }) => {
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
|
Library.propTypes = {
|
||||||
|
urlParams: PropTypes.exact({
|
||||||
|
type: PropTypes.string,
|
||||||
|
sort: PropTypes.string
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Library;
|
module.exports = Library;
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,7 @@ const DISMISS_OPTION = {
|
||||||
value: 'dismiss'
|
value: 'dismiss'
|
||||||
};
|
};
|
||||||
|
|
||||||
const onSelect = (event) => {
|
const onSelect = () => {
|
||||||
// TODO {{event.value}} {{event.dataset}}
|
// TODO {{event.value}} {{event.dataset}}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -46,20 +46,17 @@ const onNewLibraryState = (library) => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const useLibrary = (urlParams, queryParams) => {
|
const useLibrary = (urlParams) => {
|
||||||
const { core } = useServices();
|
const { core } = useServices();
|
||||||
const loadLibraryAction = React.useMemo(() => {
|
const loadLibraryAction = React.useMemo(() => {
|
||||||
if (typeof urlParams.type === 'string') {
|
if (typeof urlParams.type === 'string' && typeof urlParams.sort === 'string') {
|
||||||
return {
|
return {
|
||||||
action: 'Load',
|
action: 'Load',
|
||||||
args: {
|
args: {
|
||||||
load: 'LibraryFiltered',
|
load: 'LibraryFiltered',
|
||||||
args: {
|
args: {
|
||||||
type_name: urlParams.type,
|
type_name: urlParams.type,
|
||||||
sort_prop: queryParams.has('sort_prop') ?
|
sort_prop: urlParams.sort
|
||||||
queryParams.get('sort_prop')
|
|
||||||
:
|
|
||||||
null
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -82,7 +79,7 @@ const useLibrary = (urlParams, queryParams) => {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, [urlParams, queryParams]);
|
}, [urlParams]);
|
||||||
return useModelState({
|
return useModelState({
|
||||||
model: 'library',
|
model: 'library',
|
||||||
action: loadLibraryAction,
|
action: loadLibraryAction,
|
||||||
|
|
@ -90,6 +87,6 @@ const useLibrary = (urlParams, queryParams) => {
|
||||||
init: initLibraryState,
|
init: initLibraryState,
|
||||||
onNewState: onNewLibraryState
|
onNewState: onNewLibraryState
|
||||||
});
|
});
|
||||||
}
|
};
|
||||||
|
|
||||||
module.exports = useLibrary;
|
module.exports = useLibrary;
|
||||||
|
|
|
||||||
|
|
@ -16,13 +16,7 @@ const mapSelectableInputs = (library) => {
|
||||||
options: library.type_names
|
options: library.type_names
|
||||||
.map((type) => ({ label: type, value: type })),
|
.map((type) => ({ label: type, value: type })),
|
||||||
onSelect: (event) => {
|
onSelect: (event) => {
|
||||||
const queryParams = new URLSearchParams(
|
window.location.replace(`#/library/${encodeURIComponent(event.value)}/${encodeURIComponent(library.selected.sort_prop)}`);
|
||||||
library.selected !== null ?
|
|
||||||
[['sort_prop', library.selected.sort_prop]]
|
|
||||||
:
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
window.location.replace(`#/library/${encodeURIComponent(event.value)}?${queryParams.toString()}`);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
const sortPropSelect = {
|
const sortPropSelect = {
|
||||||
|
|
@ -33,9 +27,8 @@ const mapSelectableInputs = (library) => {
|
||||||
[],
|
[],
|
||||||
options: SORT_PROP_OPTIONS,
|
options: SORT_PROP_OPTIONS,
|
||||||
onSelect: (event) => {
|
onSelect: (event) => {
|
||||||
const queryParams = new URLSearchParams([['sort_prop', event.value]]);
|
|
||||||
if (library.selected !== null) {
|
if (library.selected !== null) {
|
||||||
window.location.replace(`#/library/${encodeURIComponent(library.selected.type_name)}?${queryParams.toString()}`);
|
window.location.replace(`#/library/${encodeURIComponent(library.selected.type_name)}/${encodeURIComponent(event.value)}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const { NavBar, MetaPreview, useInLibrary } = require('stremio/common');
|
const { NavBar, MetaPreview, useInLibrary } = require('stremio/common');
|
||||||
const VideosList = require('./VideosList');
|
const VideosList = require('./VideosList');
|
||||||
const StreamsList = require('./StreamsList');
|
const StreamsList = require('./StreamsList');
|
||||||
|
|
@ -11,7 +12,20 @@ const MetaDetails = ({ urlParams }) => {
|
||||||
const [metaResourceRef, metaResources, selectedMetaResource] = useSelectableResource(metaDetails.selected.meta_resource_ref, metaDetails.meta_resources);
|
const [metaResourceRef, metaResources, selectedMetaResource] = useSelectableResource(metaDetails.selected.meta_resource_ref, metaDetails.meta_resources);
|
||||||
const streamsResourceRef = metaDetails.selected.streams_resource_ref;
|
const streamsResourceRef = metaDetails.selected.streams_resource_ref;
|
||||||
const streamsResources = metaDetails.streams_resources;
|
const streamsResources = metaDetails.streams_resources;
|
||||||
const [inLibrary, , , toggleInLibrary] = useInLibrary(metaResourceRef !== null ? metaResourceRef.id : null);
|
const metaItem = React.useMemo(() => {
|
||||||
|
return selectedMetaResource !== null ?
|
||||||
|
selectedMetaResource.content.content
|
||||||
|
:
|
||||||
|
metaResourceRef !== null ?
|
||||||
|
{
|
||||||
|
id: metaResourceRef.id,
|
||||||
|
type: metaResourceRef.type_name,
|
||||||
|
name: ''
|
||||||
|
}
|
||||||
|
:
|
||||||
|
null;
|
||||||
|
}, [metaResourceRef, selectedMetaResource]);
|
||||||
|
const [inLibrary, toggleInLibrary] = useInLibrary(metaItem);
|
||||||
return (
|
return (
|
||||||
<div className={styles['metadetails-container']}>
|
<div className={styles['metadetails-container']}>
|
||||||
<NavBar
|
<NavBar
|
||||||
|
|
@ -98,4 +112,12 @@ const MetaDetails = ({ urlParams }) => {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
MetaDetails.propTypes = {
|
||||||
|
urlParams: PropTypes.exact({
|
||||||
|
type: PropTypes.string,
|
||||||
|
id: PropTypes.string,
|
||||||
|
videoId: PropTypes.string
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = MetaDetails;
|
module.exports = MetaDetails;
|
||||||
|
|
|
||||||
|
|
@ -44,7 +44,7 @@ const StreamsList = ({ className, streamsResources }) => {
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
StreamsList.propTypes = {
|
StreamsList.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
|
|
|
||||||
|
|
@ -16,11 +16,9 @@ const SeasonsBar = ({ className, seasons, season, onSelect }) => {
|
||||||
const selected = React.useMemo(() => {
|
const selected = React.useMemo(() => {
|
||||||
return [String(season)];
|
return [String(season)];
|
||||||
}, [season]);
|
}, [season]);
|
||||||
const renderMultiselectLabelContent = React.useMemo(() => {
|
const renderMultiselectLabelContent = React.useMemo(() => () => (
|
||||||
return () => (
|
<div className={styles['season-label']}>Season {season}</div>
|
||||||
<div className={styles['season-label']}>Season {season}</div>
|
), [season]);
|
||||||
);
|
|
||||||
}, [season]);
|
|
||||||
const prevNextButtonOnClick = React.useCallback((event) => {
|
const prevNextButtonOnClick = React.useCallback((event) => {
|
||||||
if (typeof onSelect === 'function') {
|
if (typeof onSelect === 'function') {
|
||||||
const seasonIndex = seasons.indexOf(season);
|
const seasonIndex = seasons.indexOf(season);
|
||||||
|
|
|
||||||
|
|
@ -6,7 +6,7 @@ const Icon = require('stremio-icons/dom');
|
||||||
const VideoPlaceholder = require('./VideoPlaceholder');
|
const VideoPlaceholder = require('./VideoPlaceholder');
|
||||||
const styles = require('./styles');
|
const styles = require('./styles');
|
||||||
|
|
||||||
const Video = ({ className, title, thumbnail, episode, released, upcoming, watched, progress, ...props }) => {
|
const Video = ({ className, id, title, thumbnail, episode, released, upcoming, watched, progress, ...props }) => {
|
||||||
return (
|
return (
|
||||||
<Button {...props} className={classnames(className, styles['video-container'])} title={title}>
|
<Button {...props} className={classnames(className, styles['video-container'])} title={title}>
|
||||||
{
|
{
|
||||||
|
|
@ -76,6 +76,7 @@ Video.Placeholder = VideoPlaceholder;
|
||||||
|
|
||||||
Video.propTypes = {
|
Video.propTypes = {
|
||||||
className: PropTypes.string,
|
className: PropTypes.string,
|
||||||
|
id: PropTypes.string,
|
||||||
title: PropTypes.string,
|
title: PropTypes.string,
|
||||||
thumbnail: PropTypes.string,
|
thumbnail: PropTypes.string,
|
||||||
episode: PropTypes.number,
|
episode: PropTypes.number,
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
const React = require('react');
|
const React = require('react');
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
const Icon = require('stremio-icons/dom');
|
const Icon = require('stremio-icons/dom');
|
||||||
const { MainNavBar, MetaRow } = require('stremio/common');
|
const { MainNavBar, MetaRow } = require('stremio/common');
|
||||||
const useSearch = require('./useSearch');
|
const useSearch = require('./useSearch');
|
||||||
|
|
@ -68,6 +69,10 @@ const Search = ({ queryParams }) => {
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
|
Search.propTypes = {
|
||||||
|
queryParams: PropTypes.instanceOf(URLSearchParams)
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = Search;
|
module.exports = Search;
|
||||||
|
|
|
||||||
|
|
@ -6,7 +6,7 @@ function Core() {
|
||||||
let error = null;
|
let error = null;
|
||||||
let starting = false;
|
let starting = false;
|
||||||
let stremio_core = null;
|
let stremio_core = null;
|
||||||
let events = new EventEmitter();
|
const events = new EventEmitter();
|
||||||
events.on('error', () => { });
|
events.on('error', () => { });
|
||||||
|
|
||||||
function onStateChanged() {
|
function onStateChanged() {
|
||||||
|
|
@ -26,6 +26,7 @@ function Core() {
|
||||||
try {
|
try {
|
||||||
events.emit(name, args);
|
events.emit(name, args);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
/* eslint-disable-next-line no-console */
|
||||||
console.error(e);
|
console.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -96,6 +97,6 @@ function Core() {
|
||||||
this.getState = getState;
|
this.getState = getState;
|
||||||
|
|
||||||
Object.freeze(this);
|
Object.freeze(this);
|
||||||
};
|
}
|
||||||
|
|
||||||
module.exports = Core;
|
module.exports = Core;
|
||||||
|
|
|
||||||
|
|
@ -41,6 +41,6 @@ function KeyboardNavigation() {
|
||||||
this.stop = stop;
|
this.stop = stop;
|
||||||
|
|
||||||
Object.freeze(this);
|
Object.freeze(this);
|
||||||
};
|
}
|
||||||
|
|
||||||
module.exports = KeyboardNavigation;
|
module.exports = KeyboardNavigation;
|
||||||
|
|
|
||||||
|
|
@ -4,4 +4,4 @@ const useServices = require('./useServices');
|
||||||
module.exports = {
|
module.exports = {
|
||||||
ServicesProvider,
|
ServicesProvider,
|
||||||
useServices
|
useServices
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -4,7 +4,7 @@ function Shell() {
|
||||||
let active = false;
|
let active = false;
|
||||||
let error = null;
|
let error = null;
|
||||||
let starting = false;
|
let starting = false;
|
||||||
let events = new EventEmitter();
|
const events = new EventEmitter();
|
||||||
events.on('error', () => { });
|
events.on('error', () => { });
|
||||||
|
|
||||||
function onStateChanged() {
|
function onStateChanged() {
|
||||||
|
|
@ -66,6 +66,6 @@ function Shell() {
|
||||||
this.dispatch = dispatch;
|
this.dispatch = dispatch;
|
||||||
|
|
||||||
Object.freeze(this);
|
Object.freeze(this);
|
||||||
};
|
}
|
||||||
|
|
||||||
module.exports = Shell;
|
module.exports = Shell;
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue