conflict resolved

This commit is contained in:
svetlagasheva 2019-12-16 18:06:15 +02:00
commit 0aa6b06df2
52 changed files with 1256 additions and 541 deletions

85
.eslintrc Normal file
View 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
}
]
}
}

View file

@ -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"
} }

View file

@ -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;

View file

@ -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;

View file

@ -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;

View file

@ -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;

View file

@ -23,6 +23,8 @@ const MainNavBar = React.memo(({ className }) => {
); );
}); });
MainNavBar.displayName = 'MainNavBar';
MainNavBar.propTypes = { MainNavBar.propTypes = {
className: PropTypes.string className: PropTypes.string
}; };

View file

@ -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;

View file

@ -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']}

View file

@ -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
}; };

View file

@ -1,3 +1,3 @@
const ModalDialog = require('./ModalDialog'); const ModalDialog = require('./ModalDialog');
module.exports = ModalDialog; module.exports = ModalDialog;

View file

@ -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;

View file

@ -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')`
}} }}

View file

@ -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,

View file

@ -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;

View file

@ -57,7 +57,7 @@ const NotificationsList = ({ className, metaItems }) => {
} }
</div> </div>
); );
} };
NotificationsList.propTypes = { NotificationsList.propTypes = {
className: PropTypes.string, className: PropTypes.string,

View file

@ -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();

View file

@ -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
}; };

View file

@ -1,3 +1,3 @@
const PaginationInput = require('./PaginationInput'); const PaginationInput = require('./PaginationInput');
module.exports = PaginationInput; module.exports = PaginationInput;

View file

@ -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
}; };

View file

@ -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;

View file

@ -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']
} }
}; };

View file

@ -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;

View file

@ -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>

View file

@ -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;

View file

@ -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;

View file

@ -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;

View file

@ -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;
} }
} }

View 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;

View file

@ -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 };
}; };

View file

@ -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) => {

View file

@ -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;

View file

@ -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>

View file

@ -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}}
}; };

View file

@ -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;

View file

@ -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;

View file

@ -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;

View file

@ -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&#39;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;

View file

@ -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;

View file

@ -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}}
}; };

View file

@ -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;

View file

@ -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)}`);
} }
} }
}; };

View file

@ -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;

View file

@ -44,7 +44,7 @@ const StreamsList = ({ className, streamsResources }) => {
</Button> </Button>
</div> </div>
); );
} };
StreamsList.propTypes = { StreamsList.propTypes = {
className: PropTypes.string, className: PropTypes.string,

View file

@ -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);

View file

@ -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,

View file

@ -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;

View file

@ -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;

View file

@ -41,6 +41,6 @@ function KeyboardNavigation() {
this.stop = stop; this.stop = stop;
Object.freeze(this); Object.freeze(this);
}; }
module.exports = KeyboardNavigation; module.exports = KeyboardNavigation;

View file

@ -4,4 +4,4 @@ const useServices = require('./useServices');
module.exports = { module.exports = {
ServicesProvider, ServicesProvider,
useServices useServices
}; };

View file

@ -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;

792
yarn.lock

File diff suppressed because it is too large Load diff