lbry-desktop/src/ui/page/settings/view.jsx

667 lines
24 KiB
React
Raw Normal View History

2018-03-26 23:32:43 +02:00
// @flow
2019-08-25 22:43:44 +02:00
/* eslint react/no-unescaped-entities:0 */
/* eslint react/jsx-no-comment-textnodes:0 */
2018-11-26 02:21:25 +01:00
import * as SETTINGS from 'constants/settings';
import * as PAGES from 'constants/pages';
2018-03-26 23:32:43 +02:00
import * as React from 'react';
2019-09-27 22:03:05 +02:00
import { FormField, FormFieldPrice } from 'component/common/form';
2018-03-26 23:32:43 +02:00
import Button from 'component/button';
import I18nMessage from 'component/i18nMessage';
2018-03-26 23:32:43 +02:00
import Page from 'component/page';
import SettingLanguage from 'component/settingLanguage';
2018-03-26 23:32:43 +02:00
import FileSelector from 'component/common/file-selector';
2019-10-15 23:23:51 +02:00
import SyncToggle from 'component/syncToggle';
2019-09-27 22:03:05 +02:00
import Card from 'component/common/card';
2019-08-28 04:35:07 +02:00
import { getSavedPassword } from 'util/saved-passwords';
2018-03-26 23:32:43 +02:00
2019-04-24 16:02:08 +02:00
type Price = {
2018-03-26 23:32:43 +02:00
currency: string,
amount: number,
};
type SetDaemonSettingArg = boolean | string | number | Price;
2019-08-18 19:01:04 +02:00
type DarkModeTimes = {
from: { hour: string, min: string, formattedTime: string },
to: { hour: string, min: string, formattedTime: string },
};
type OptionTimes = {
fromTo: string,
time: string,
};
2018-03-26 23:32:43 +02:00
type DaemonSettings = {
download_dir: string,
2018-03-26 23:32:43 +02:00
share_usage_data: boolean,
2018-10-13 17:49:47 +02:00
max_key_fee?: Price,
2019-08-02 08:28:14 +02:00
max_connections_per_download?: number,
save_files: boolean,
save_blobs: boolean,
2018-03-26 23:32:43 +02:00
};
type Props = {
setDaemonSetting: (string, ?SetDaemonSettingArg) => void,
setClientSetting: (string, SetDaemonSettingArg) => void,
2018-03-26 23:32:43 +02:00
clearCache: () => Promise<any>,
daemonSettings: DaemonSettings,
showNsfw: boolean,
2019-10-08 18:25:33 +02:00
showAnonymous: boolean,
2018-03-26 23:32:43 +02:00
instantPurchaseEnabled: boolean,
instantPurchaseMax: Price,
currentTheme: string,
themes: Array<string>,
automaticDarkModeEnabled: boolean,
2018-05-30 05:18:41 +02:00
autoplay: boolean,
2019-10-15 06:20:12 +02:00
// autoDownload: boolean,
2018-07-18 21:48:30 +02:00
encryptWallet: () => void,
decryptWallet: () => void,
2018-10-13 17:49:47 +02:00
updateWalletStatus: () => void,
2018-07-18 21:48:30 +02:00
walletEncrypted: boolean,
osNotificationsEnabled: boolean,
supportOption: boolean,
userBlockedChannelsCount?: number,
hideBalance: boolean,
confirmForgetPassword: ({}) => void,
2019-08-13 07:35:13 +02:00
floatingPlayer: boolean,
clearPlayingUri: () => void,
2019-08-18 19:01:04 +02:00
darkModeTimes: DarkModeTimes,
2019-08-18 18:54:55 +02:00
setDarkTime: (string, {}) => void,
2018-03-26 23:32:43 +02:00
};
type State = {
clearingCache: boolean,
2019-08-20 14:29:59 +02:00
storedPassword: boolean,
2018-03-26 23:32:43 +02:00
};
class SettingsPage extends React.PureComponent<Props, State> {
constructor(props: Props) {
2017-05-17 10:10:25 +02:00
super(props);
this.state = {
clearingCache: false,
2019-08-20 14:29:59 +02:00
storedPassword: false,
2017-06-06 23:19:12 +02:00
};
2018-03-26 23:32:43 +02:00
(this: any).onKeyFeeChange = this.onKeyFeeChange.bind(this);
2019-08-02 08:28:14 +02:00
(this: any).onMaxConnectionsChange = this.onMaxConnectionsChange.bind(this);
(this: any).onKeyFeeDisableChange = this.onKeyFeeDisableChange.bind(this);
2018-03-26 23:32:43 +02:00
(this: any).onInstantPurchaseMaxChange = this.onInstantPurchaseMaxChange.bind(this);
(this: any).onThemeChange = this.onThemeChange.bind(this);
(this: any).onAutomaticDarkModeChange = this.onAutomaticDarkModeChange.bind(this);
(this: any).clearCache = this.clearCache.bind(this);
2019-08-18 18:54:55 +02:00
(this: any).onChangeTime = this.onChangeTime.bind(this);
2019-08-28 04:35:07 +02:00
(this: any).onConfirmForgetPassword = this.onConfirmForgetPassword.bind(this);
}
2018-03-26 23:32:43 +02:00
componentDidMount() {
2018-10-13 17:49:47 +02:00
this.props.updateWalletStatus();
2019-08-28 04:35:07 +02:00
getSavedPassword().then(p => {
if (p) {
this.setState({ storedPassword: true });
}
});
2017-05-17 10:10:25 +02:00
}
2018-03-26 23:32:43 +02:00
onKeyFeeChange(newValue: Price) {
this.setDaemonSetting('max_key_fee', newValue);
2017-05-17 10:10:25 +02:00
}
2019-08-02 08:28:14 +02:00
onMaxConnectionsChange(event: SyntheticInputEvent<*>) {
const { value } = event.target;
this.setDaemonSetting('max_connections_per_download', value);
}
2018-03-26 23:32:43 +02:00
onKeyFeeDisableChange(isDisabled: boolean) {
if (isDisabled) this.setDaemonSetting('max_key_fee');
2017-05-17 10:10:25 +02:00
}
2018-03-26 23:32:43 +02:00
onThemeChange(event: SyntheticInputEvent<*>) {
const { value } = event.target;
if (value === 'dark') {
this.onAutomaticDarkModeChange(false);
}
2018-11-26 02:21:25 +01:00
this.props.setClientSetting(SETTINGS.THEME, value);
2017-08-05 03:36:36 +02:00
}
2018-03-26 23:32:43 +02:00
onAutomaticDarkModeChange(value: boolean) {
2018-11-26 02:21:25 +01:00
this.props.setClientSetting(SETTINGS.AUTOMATIC_DARK_MODE_ENABLED, value);
2018-01-14 10:14:15 +01:00
}
2018-03-26 23:32:43 +02:00
onInstantPurchaseEnabledChange(enabled: boolean) {
2018-11-26 02:21:25 +01:00
this.props.setClientSetting(SETTINGS.INSTANT_PURCHASE_ENABLED, enabled);
}
2018-03-26 23:32:43 +02:00
onInstantPurchaseMaxChange(newValue: Price) {
2018-11-26 02:21:25 +01:00
this.props.setClientSetting(SETTINGS.INSTANT_PURCHASE_MAX, newValue);
}
2018-07-18 21:48:30 +02:00
onChangeEncryptWallet() {
2018-10-13 17:49:47 +02:00
const { decryptWallet, walletEncrypted, encryptWallet } = this.props;
if (walletEncrypted) {
decryptWallet();
} else {
encryptWallet();
}
}
2019-08-20 14:29:59 +02:00
onConfirmForgetPassword() {
const { confirmForgetPassword } = this.props;
confirmForgetPassword({
callback: () => {
this.setState({ storedPassword: false });
},
});
2019-08-20 14:29:59 +02:00
}
2019-08-18 19:01:04 +02:00
onChangeTime(event: SyntheticInputEvent<*>, options: OptionTimes) {
2019-08-18 18:54:55 +02:00
const { value } = event.target;
this.props.setDarkTime(value, options);
}
to12Hour(time: string) {
const now = new Date(0, 0, 0, Number(time));
const hour = now.toLocaleTimeString('en-US', { hour12: true, hour: '2-digit' });
return hour;
}
setDaemonSetting(name: string, value: ?SetDaemonSettingArg): void {
2018-10-13 17:49:47 +02:00
this.props.setDaemonSetting(name, value);
}
2018-03-26 23:32:43 +02:00
clearCache() {
this.setState({
clearingCache: true,
});
const success = () => {
this.setState({ clearingCache: false });
window.location.reload();
2018-03-26 23:32:43 +02:00
};
const clear = () => this.props.clearCache().then(success);
setTimeout(clear, 1000, { once: true });
}
2017-08-07 05:01:10 +02:00
2017-05-17 10:10:25 +02:00
render() {
2017-09-07 03:53:42 +02:00
const {
daemonSettings,
showNsfw,
2019-10-08 18:25:33 +02:00
showAnonymous,
instantPurchaseEnabled,
instantPurchaseMax,
2018-03-26 23:32:43 +02:00
currentTheme,
2017-09-07 03:53:42 +02:00
themes,
2018-01-14 10:14:15 +01:00
automaticDarkModeEnabled,
2018-05-30 05:18:41 +02:00
autoplay,
2018-07-18 21:48:30 +02:00
walletEncrypted,
osNotificationsEnabled,
2019-10-15 06:20:12 +02:00
// autoDownload,
2019-02-13 17:27:20 +01:00
setDaemonSetting,
setClientSetting,
supportOption,
hideBalance,
userBlockedChannelsCount,
2019-08-13 07:35:13 +02:00
floatingPlayer,
2019-08-14 05:04:08 +02:00
clearPlayingUri,
2019-08-18 18:54:55 +02:00
darkModeTimes,
2017-09-07 03:53:42 +02:00
} = this.props;
2018-03-26 23:32:43 +02:00
const noDaemonSettings = !daemonSettings || Object.keys(daemonSettings).length === 0;
const defaultMaxKeyFee = { currency: 'USD', amount: 50 };
2019-08-02 08:28:14 +02:00
const disableMaxKeyFee = !(daemonSettings && daemonSettings.max_key_fee);
2019-08-13 07:35:13 +02:00
const connectionOptions = [1, 2, 4, 6, 10, 20];
2019-08-18 18:54:55 +02:00
const startHours = ['18', '19', '20', '21'];
const endHours = ['5', '6', '7', '8'];
2016-04-10 02:00:56 +02:00
return (
2018-03-26 23:32:43 +02:00
<Page>
2019-08-26 20:32:45 +02:00
{!IS_WEB && noDaemonSettings ? (
2018-03-26 23:32:43 +02:00
<section className="card card--section">
<div className="card__title">{__('Failed to load settings.')}</div>
</section>
) : (
2019-08-22 23:14:51 +02:00
<div>
2019-09-27 22:03:05 +02:00
<Card title={__('Language')} actions={<SettingLanguage />} />
2019-10-15 23:23:51 +02:00
<Card title={__('Sync')} actions={<SyncToggle />} />
2019-08-26 20:32:45 +02:00
{/* @if TARGET='app' */}
2019-09-27 22:03:05 +02:00
<Card
title={__('Download Directory')}
actions={
<React.Fragment>
<FileSelector
type="openDirectory"
currentPath={daemonSettings.download_dir}
onFileChosen={(newDirectory: string) => {
setDaemonSetting('download_dir', newDirectory);
}}
/>
<p className="help">{__('LBRY downloads will be saved here.')}</p>
</React.Fragment>
}
/>
<Card
title={__('Network and Data Settings')}
actions={
<React.Fragment>
<FormField
type="checkbox"
name="save_files"
onChange={() => setDaemonSetting('save_files', !daemonSettings.save_files)}
checked={daemonSettings.save_files}
label={__('Save all viewed content to your downloads directory')}
helper={__(
'Paid content and some file types are saved by default. Changing this setting will not affect previously downloaded content.'
)}
/>
2019-09-27 22:03:05 +02:00
<FormField
type="checkbox"
name="save_blobs"
onChange={() => setDaemonSetting('save_blobs', !daemonSettings.save_blobs)}
checked={daemonSettings.save_blobs}
label={__('Save hosting data to help the LBRY network')}
helper={
<React.Fragment>
{__("If disabled, LBRY will be very sad and you won't be helping improve the network.")}{' '}
<Button button="link" label={__('Learn more')} href="https://lbry.com/faq/host-content" />.
</React.Fragment>
}
/>
</React.Fragment>
}
/>
<Card
title={__('Max Purchase Price')}
actions={
<React.Fragment>
<FormField
type="radio"
name="no_max_purchase_no_limit"
checked={disableMaxKeyFee}
label={__('No Limit')}
onChange={() => {
this.onKeyFeeDisableChange(true);
}}
/>
<FormField
type="radio"
name="max_purchase_limit"
checked={!disableMaxKeyFee}
onChange={() => {
this.onKeyFeeDisableChange(false);
this.onKeyFeeChange(defaultMaxKeyFee);
}}
label={__('Choose limit')}
/>
2019-02-13 17:27:20 +01:00
2019-09-27 22:03:05 +02:00
{!disableMaxKeyFee && (
<FormFieldPrice
name="max_key_fee"
min={0}
onChange={this.onKeyFeeChange}
price={daemonSettings.max_key_fee ? daemonSettings.max_key_fee : defaultMaxKeyFee}
/>
)}
<p className="help">
{__('This will prevent you from purchasing any content over a certain cost, as a safety measure.')}
</p>
</React.Fragment>
}
/>
<Card
title={__('Purchase Confirmations')}
actions={
<React.Fragment>
<FormField
type="radio"
name="confirm_all_purchases"
checked={!instantPurchaseEnabled}
label={__('Always confirm before purchasing content')}
onChange={() => {
this.onInstantPurchaseEnabledChange(false);
}}
/>
<FormField
type="radio"
name="instant_purchases"
checked={instantPurchaseEnabled}
label={__('Only confirm purchases over a certain price')}
onChange={() => {
this.onInstantPurchaseEnabledChange(true);
}}
2018-04-11 00:05:30 +02:00
/>
2019-09-27 22:03:05 +02:00
{instantPurchaseEnabled && (
<FormFieldPrice
name="confirmation_price"
min={0.1}
onChange={this.onInstantPurchaseMaxChange}
price={instantPurchaseMax}
/>
)}
2019-09-27 22:03:05 +02:00
<p className="help">
{__("When this option is chosen, LBRY won't ask you to confirm downloads below your chosen price.")}
</p>
</React.Fragment>
}
/>
{/* @endif */}
<Card
title={__('Content Settings')}
actions={
<React.Fragment>
<FormField
type="checkbox"
name="floating_player"
onChange={() => {
setClientSetting(SETTINGS.FLOATING_PLAYER, !floatingPlayer);
clearPlayingUri();
}}
checked={floatingPlayer}
label={__('Floating video player')}
helper={__('Keep content playing in the corner when navigating to a different page.')}
/>
2019-02-20 06:20:29 +01:00
2019-09-27 22:03:05 +02:00
<FormField
type="checkbox"
name="autoplay"
onChange={() => setClientSetting(SETTINGS.AUTOPLAY, !autoplay)}
checked={autoplay}
label={__('Autoplay media files')}
helper={__(
'Autoplay video and audio files when navigating to a file, as well as the next related item when a file finishes playing.'
)}
2018-04-11 00:05:30 +02:00
/>
2019-10-08 18:25:33 +02:00
<FormField
type="checkbox"
name="show_anonymous"
onChange={() => setClientSetting(SETTINGS.SHOW_ANONYMOUS, !showAnonymous)}
checked={showAnonymous}
label={__('Show anonymous content')}
2019-10-15 19:29:27 +02:00
helper={__('Anonymous content is published without a channel.')}
2019-10-08 18:25:33 +02:00
/>
2019-09-27 22:03:05 +02:00
<FormField
type="checkbox"
name="show_nsfw"
onChange={() => setClientSetting(SETTINGS.SHOW_MATURE, !showNsfw)}
checked={showNsfw}
label={__('Show mature content')}
helper={__(
'Mature content may include nudity, intense sexuality, profanity, or other adult content. By displaying mature content, you are affirming you are of legal age to view mature content in your country or jurisdiction. '
)}
/>
</React.Fragment>
}
/>
<Card
title={__('Blocked Channels')}
actions={
<p>
{__('You have')} {userBlockedChannelsCount} {__('blocked')}{' '}
{userBlockedChannelsCount === 1 && __('channel')}
{userBlockedChannelsCount !== 1 && __('channels')}.{' '}
<Button button="link" label={__('Manage')} navigate={`/$/${PAGES.BLOCKED}`} />
</p>
2019-09-27 22:03:05 +02:00
}
/>
2019-08-26 20:32:45 +02:00
{/* @if TARGET='app' */}
2019-09-27 22:03:05 +02:00
<Card
title={__('Notifications')}
actions={
2019-08-25 22:43:44 +02:00
<FormField
type="checkbox"
name="desktopNotification"
onChange={() => setClientSetting(SETTINGS.OS_NOTIFICATIONS_ENABLED, !osNotificationsEnabled)}
checked={osNotificationsEnabled}
label={__('Show Desktop Notifications')}
helper={__('Get notified when a publish is confirmed, or when new content is available to watch.')}
/>
2019-09-27 22:03:05 +02:00
}
/>
<Card
title={__('Share Diagnostic Data')}
actions={
<FormField
2019-07-21 23:31:22 +02:00
type="checkbox"
name="share_usage_data"
2019-05-07 23:38:29 +02:00
onChange={() => setDaemonSetting('share_usage_data', !daemonSettings.share_usage_data)}
checked={daemonSettings.share_usage_data}
2019-07-17 20:04:50 +02:00
label={
<React.Fragment>
{__('Help make LBRY better by contributing analytics and diagnostic data about my usage.')}{' '}
<Button button="link" label={__('Learn more')} href="https://lbry.com/privacypolicy" />.
</React.Fragment>
}
2019-05-07 23:38:29 +02:00
helper={__('You will be ineligible to earn rewards while diagnostics are not being shared.')}
/>
2019-09-27 22:03:05 +02:00
}
/>
2019-08-26 20:32:45 +02:00
{/* @endif */}
2019-09-27 22:03:05 +02:00
<Card
title={__('Appearance')}
actions={
<React.Fragment>
<fieldset-section>
<FormField
name="theme_select"
type="select"
label={__('Theme')}
onChange={this.onThemeChange}
value={currentTheme}
disabled={automaticDarkModeEnabled}
>
{themes.map(theme => (
<option key={theme} value={theme}>
{theme}
</option>
))}
</FormField>
</fieldset-section>
<fieldset-section>
<FormField
type="checkbox"
name="automatic_dark_mode"
onChange={() => this.onAutomaticDarkModeChange(!automaticDarkModeEnabled)}
checked={automaticDarkModeEnabled}
label={__('Automatic dark mode')}
/>
{automaticDarkModeEnabled && (
<fieldset-group class="fieldset-group--smushed">
<FormField
type="select"
name="automatic_dark_mode_range"
onChange={value => this.onChangeTime(value, { fromTo: 'from', time: 'hour' })}
value={darkModeTimes.from.hour}
label={__('From')}
>
{startHours.map(time => (
<option key={time} value={time}>
{this.to12Hour(time)}
</option>
))}
</FormField>
<FormField
type="select"
name="automatic_dark_mode_range"
label={__('To')}
onChange={value => this.onChangeTime(value, { fromTo: 'to', time: 'hour' })}
value={darkModeTimes.to.hour}
>
{endHours.map(time => (
<option key={time} value={time}>
{this.to12Hour(time)}
</option>
))}
</FormField>
</fieldset-group>
)}
</fieldset-section>
</React.Fragment>
}
/>
<Card
title={__('Wallet Security')}
actions={
<React.Fragment>
2019-10-01 07:20:05 +02:00
{/* @if TARGET='app' */}
2019-02-13 17:27:20 +01:00
<FormField
2019-10-03 23:40:54 +02:00
disabled
2019-07-21 23:31:22 +02:00
type="checkbox"
2019-09-27 22:03:05 +02:00
name="encrypt_wallet"
onChange={() => this.onChangeEncryptWallet()}
checked={walletEncrypted}
label={__('Encrypt my wallet with a custom password')}
helper={
<React.Fragment>
2019-10-03 23:40:54 +02:00
<I18nMessage
tokens={{
learn_more: (
<Button button="link" label={__('Learn more')} href="https://lbry.com/faq/account-sync" />
),
}}
>
Wallet encryption is currently unavailable until it's supported for synced accounts. It will
be added back soon. %learn_more%
</I18nMessage>
{/* {__('Secure your local wallet data with a custom password.')}{' '}
2019-09-27 22:03:05 +02:00
<strong>{__('Lost passwords cannot be recovered.')} </strong>
2019-10-03 23:40:54 +02:00
<Button button="link" label={__('Learn more')} href="https://lbry.com/faq/wallet-encryption" />. */}
2019-09-27 22:03:05 +02:00
</React.Fragment>
}
2019-02-13 17:27:20 +01:00
/>
2019-08-26 20:32:45 +02:00
2019-09-27 22:03:05 +02:00
{walletEncrypted && this.state.storedPassword && (
<FormField
type="checkbox"
name="save_password"
onChange={this.onConfirmForgetPassword}
checked={this.state.storedPassword}
label={__('Save Password')}
helper={<React.Fragment>{__('Automatically unlock your wallet on startup')}</React.Fragment>}
/>
)}
2019-10-01 07:20:05 +02:00
{/* @endif */}
<FormField
type="checkbox"
2019-09-27 22:03:05 +02:00
name="hide_balance"
onChange={() => setClientSetting(SETTINGS.HIDE_BALANCE, !hideBalance)}
checked={hideBalance}
label={__('Hide wallet balance in header')}
/>
</React.Fragment>
}
/>
<Card
title={__('Experimental Settings')}
actions={
<React.Fragment>
<FormField
type="checkbox"
name="support_option"
onChange={() => setClientSetting(SETTINGS.SUPPORT_OPTION, !supportOption)}
checked={supportOption}
label={__('Enable claim support')}
helper={
<I18nMessage
tokens={{
discovery_link: (
<Button button="link" label={__('discovery')} href="https://lbry.com/faq/trending" />
),
vanity_names_link: (
<Button button="link" label={__('vanity names')} href="https://lbry.com/faq/naming" />
),
}}
>
This will add a Support button along side tipping. Similar to tips, supports help
%discovery_link% but the LBC is returned to your wallet if revoked. Both also help secure your
%vanity_names_link%.
</I18nMessage>
}
/>
2019-02-13 17:27:20 +01:00
2019-10-01 07:20:05 +02:00
{/* @if TARGET='app' */}
2019-10-15 17:47:43 +02:00
{/*
Disabling below until we get downloads to work with shared subscriptions code
<FormField
2019-09-27 22:03:05 +02:00
type="checkbox"
name="auto_download"
onChange={() => setClientSetting(SETTINGS.AUTO_DOWNLOAD, !autoDownload)}
checked={autoDownload}
label={__('Automatically download new content from my subscriptions')}
2019-08-13 07:35:13 +02:00
helper={__(
2019-09-27 22:03:05 +02:00
"The latest file from each of your subscriptions will be downloaded for quick access as soon as it's published."
2019-08-13 07:35:13 +02:00
)}
2019-10-15 06:20:12 +02:00
/> */}
2019-09-27 22:03:05 +02:00
<fieldset-section>
<FormField
name="max_connections"
type="select"
label={__('Max Connections')}
helper={__(
'For users with good bandwidth, try a higher value to improve streaming and download speeds. Low bandwidth users may benefit from a lower setting. Default is 4.'
)}
min={1}
max={100}
onChange={this.onMaxConnectionsChange}
value={daemonSettings.max_connections_per_download}
>
{connectionOptions.map(connectionOption => (
<option key={connectionOption} value={connectionOption}>
{connectionOption}
</option>
))}
</FormField>
</fieldset-section>
2019-10-01 07:20:05 +02:00
{/* @endif */}
2019-09-27 22:03:05 +02:00
</React.Fragment>
}
/>
<Card
title={__('Application Cache')}
subtitle={
2019-10-16 03:14:21 +02:00
<p className="card__subtitle--status">
2019-09-27 22:03:05 +02:00
{__(
'This will clear the application cache. Your wallet will not be affected. Currently, followed tags and blocked channels will be cleared.'
)}
2019-10-16 03:14:21 +02:00
</p>
2019-09-27 22:03:05 +02:00
}
actions={
<Button
button="inverse"
label={this.state.clearingCache ? __('Clearing') : __('Clear Cache')}
onClick={this.clearCache}
disabled={this.state.clearingCache}
/>
}
/>
2019-03-18 06:06:41 +01:00
</div>
2018-03-26 23:32:43 +02:00
)}
</Page>
2016-04-10 02:00:56 +02:00
);
}
2017-05-17 10:10:25 +02:00
}
2016-11-22 21:19:08 +01:00
export default SettingsPage;