Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Mobile: Resolves #10883: Remove slider component module and replace integer settings with validated text inputs and ensure all validation error flows work correctly #11069

Open
wants to merge 33 commits into
base: dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
e55db5a
Create and utilise new component for note history and trashed item tt…
mrjo118 Sep 16, 2024
454da7c
Revert "Create and utilise new component for note history and trashed…
mrjo118 Oct 2, 2024
3d35235
Merge remote-tracking branch 'origin/dev' into note-history-and-trash…
mrjo118 Oct 2, 2024
d001a1e
Allow slider component values to be incremented by tapping the value …
mrjo118 Oct 2, 2024
a0f95a9
Merge branch 'dev' into note-history-and-trash-ttl-text-input
personalizedrefrigerator Oct 5, 2024
dc84c0c
Remove slider component mode and replace integer settings with valida…
mrjo118 Nov 12, 2024
70dbbb3
Make switching to the plugins screen on desktop trigger the switching…
mrjo118 Nov 13, 2024
11b05af
Merge branch 'note-history-and-trash-ttl-text-input2' into note-histo…
mrjo118 Nov 13, 2024
d563598
Remove unused import
mrjo118 Nov 13, 2024
10cb151
Remove license changes with incorrect formatting
mrjo118 Nov 13, 2024
3d7e0c9
Add license change back in
mrjo118 Nov 13, 2024
4a116c3
Revert "Add license change back in"
mrjo118 Nov 13, 2024
e754232
Add license change back in
mrjo118 Nov 13, 2024
47c2fc4
Avoid validating enum values as integers, as they are already restricted
mrjo118 Nov 13, 2024
1aa395e
Fix validation issue
mrjo118 Nov 13, 2024
1e4b87e
Add automated tests
mrjo118 Nov 19, 2024
ad8a277
Merge remote-tracking branch 'origin/dev' into note-history-and-trash…
mrjo118 Nov 19, 2024
57304d2
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Nov 19, 2024
1aa63f0
Resolved compile error due to merging in updates
mrjo118 Nov 19, 2024
960af00
Merge branch 'note-history-and-trash-ttl-text-input' of https://githu…
mrjo118 Nov 19, 2024
bf039c3
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Nov 28, 2024
5428a45
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Dec 9, 2024
2595f61
Change formatting of values back to always using integer type and add…
mrjo118 Dec 10, 2024
c7e97f5
Merge remote-tracking branch 'origin/dev' into note-history-and-trash…
mrjo118 Dec 19, 2024
de1f860
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 1, 2025
0031b96
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 9, 2025
1d24940
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 9, 2025
703f660
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 10, 2025
f2b15f2
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 10, 2025
df99617
Resolve issues from merged in code and missed section of slider compo…
mrjo118 Jan 11, 2025
151a8ca
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 11, 2025
d6955b0
Merge branch 'dev' into note-history-and-trash-ttl-text-input
mrjo118 Jan 11, 2025
622d848
Disable full screen ui for number text inputs, to avoid undesirable b…
mrjo118 Jan 11, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view

This file was deleted.

1 change: 0 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,6 @@
"[email protected]": "patch:app-builder-lib@npm%3A24.4.0#./.yarn/patches/app-builder-lib-npm-24.4.0-05322ff057.patch",
"nanoid": "patch:nanoid@npm%3A3.3.7#./.yarn/patches/nanoid-npm-3.3.7-98824ba130.patch",
"pdfjs-dist": "patch:pdfjs-dist@npm%3A3.11.174#./.yarn/patches/pdfjs-dist-npm-3.11.174-67f2fee6d6.patch",
"@react-native-community/slider": "patch:@react-native-community/slider@npm%3A4.4.4#./.yarn/patches/@react-native-community-slider-npm-4.4.4-d78e472f48.patch",
"husky": "patch:husky@npm%3A3.1.0#./.yarn/patches/husky-npm-3.1.0-5cc13e4e34.patch",
"chokidar@^2.0.0": "3.5.3",
"[email protected]": "patch:react-native@npm%3A0.74.1#./.yarn/patches/react-native-npm-0.74.1-754c02ae9e.patch",
Expand Down
12 changes: 9 additions & 3 deletions packages/app-desktop/gui/ConfigScreen/ConfigScreen.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -141,18 +141,24 @@ class ConfigScreenComponent extends React.Component<any, any> {
public async switchSection(name: string) {
const section = this.sectionByName(name);
let screenName = '';
if (section.isScreen) {
screenName = section.name;
if (section.isScreen || name === 'plugins') {
if (section.isScreen) {
screenName = section.name;
}

if (this.hasChanges()) {
const ok = await shim.showConfirmationDialog(_('This will open a new screen. Save your current changes?'));
if (ok) {
await shared.saveSettings(this);
const result = await shared.saveSettings(this);
if (result === false) return false;
} else {
return false;
}
}
}

this.setState({ selectedSectionName: section.name, screenName: screenName });
return true;
}

// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import Setting, { AppType, SettingItemSubType } from '@joplin/lib/models/Setting';
import { themeStyle } from '@joplin/lib/theme';
import * as React from 'react';
import { useCallback, useId } from 'react';
import { useCallback, useId, useState } from 'react';
import control_PluginsStates from './plugins/PluginsStates';
import bridge from '../../../services/bridge';
import { _ } from '@joplin/lib/locale';
Expand Down Expand Up @@ -70,6 +70,7 @@ const SettingComponent: React.FC<Props> = props => {
const inputId = useId();
const descriptionId = useId();
const descriptionComp = <SettingDescription id={descriptionId} text={descriptionText}/>;
const [valueState, setValueState] = useState(props.value?.toString());

if (key in settingKeyToControl) {
const CustomSettingComponent = settingKeyToControl[key];
Expand Down Expand Up @@ -321,10 +322,9 @@ const SettingComponent: React.FC<Props> = props => {
);
}
} else if (md.type === Setting.TYPE_INT) {
const value = props.value as number;

const onNumChange: React.ChangeEventHandler<HTMLInputElement> = (event) => {
updateSettingValue(key, event.target.value);
setValueState(event.target.value);
updateSettingValue(key, Number.isInteger(Number(event.target.value)) ? event.target.value : ''); // Prevent invalid values being mapped to 0
};

const label = [md.label()];
Expand All @@ -336,7 +336,7 @@ const SettingComponent: React.FC<Props> = props => {
<input
type="number"
style={textInputBaseStyle}
value={value}
value={valueState}
onChange={onNumChange}
min={md.minimum}
max={md.maximum}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -131,11 +131,13 @@ class ConfigScreenComponent extends BaseScreenComponent<ConfigScreenProps, Confi
const shouldSetIgnoreTlsErrors = this.state.changedSettingKeys.includes('net.ignoreTlsErrors');

const done = await shared.saveSettings(this);
if (!done) return;
if (!done) return false;

if (shouldSetIgnoreTlsErrors) {
await setIgnoreTlsErrors(Setting.value('net.ignoreTlsErrors'));
}

return true;
};

private syncStatusButtonPress_ = () => {
Expand Down Expand Up @@ -263,22 +265,21 @@ class ConfigScreenComponent extends BaseScreenComponent<ConfigScreenProps, Confi
return this.state.changedSettingKeys.length > 0;
}

private async promptSaveChanges(): Promise<void> {
private async promptSaveChanges(): Promise<boolean> {
if (this.hasUnsavedChanges()) {
const response = await shim.showMessageBox(_('There are unsaved changes.'), {
buttons: [_('Save changes'), _('Discard changes')],
});
if (response === 0) {
await this.saveButton_press();
return await this.saveButton_press();
}
}

return true;
}

private handleNavigateToNewScreen = async (): Promise<boolean> => {
await this.promptSaveChanges();

// Continue navigation
return false;
return !(await this.promptSaveChanges());
};

private handleBackButtonPress = (): boolean => {
Expand All @@ -301,8 +302,10 @@ class ConfigScreenComponent extends BaseScreenComponent<ConfigScreenProps, Confi

if (this.hasUnsavedChanges()) {
void (async () => {
await this.promptSaveChanges();
await goBack();
const result = await this.promptSaveChanges();
if (result) {
await goBack();
}
})();
return true;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,11 @@ import { View, Text, TextInput } from 'react-native';
import Setting, { AppType } from '@joplin/lib/models/Setting';
import Dropdown from '../../Dropdown';
import { ConfigScreenStyles } from './configScreenStyles';
import Slider from '@react-native-community/slider';
import SettingsToggle from './SettingsToggle';
import FileSystemPathSelector from './FileSystemPathSelector';
import shim from '@joplin/lib/shim';
import { themeStyle } from '../../global-style';
import { useId } from 'react';
import { useId, useState } from 'react';

interface Props {
settingId: string;
Expand Down Expand Up @@ -41,6 +40,7 @@ const SettingComponent: React.FunctionComponent<Props> = props => {
const containerStyles = props.styles.getContainerStyle(!!settingDescription);

const labelId = useId();
const [valueState, setValueState] = useState(props.value?.toString());

if (md.isEnum) {
const value = props.value?.toString();
Expand Down Expand Up @@ -92,33 +92,35 @@ const SettingComponent: React.FunctionComponent<Props> = props => {
/>
);
} else if (md.type === Setting.TYPE_INT) {
const unitLabel = md.unitLabel ? md.unitLabel(props.value) : props.value;
const minimum = 'minimum' in md ? md.minimum : 0;
const maximum = 'maximum' in md ? md.maximum : 10;
const label = md.label();
const label = md.unitLabel?.toString() !== undefined ? `${md.label()} (${md.unitLabel(md.value)})` : `${md.label()}`;

// Note: Do NOT add the minimumTrackTintColor and maximumTrackTintColor props
// on the Slider as they are buggy and can crash the app on certain devices.
// https://github.com/laurent22/joplin/issues/2733
// https://github.com/react-native-community/react-native-slider/issues/161
return (
<View key={props.settingId} style={styleSheet.settingContainer}>
<Text key="label" style={styleSheet.settingText} nativeID={labelId}>
{label}
</Text>
<View style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', flex: 1 }}>
<Text style={styleSheet.sliderUnits}>{unitLabel}</Text>
<Slider
<View key={props.settingId} style={containerStyles.outerContainer}>
<View key={props.settingId} style={containerStyles.innerContainer}>
<Text key="label" style={styleSheet.settingText} nativeID={labelId}>
{label}
</Text>
<TextInput
keyboardType="numeric"
autoCorrect={false}
autoComplete="off"
selectionColor={theme.textSelectionColor}
keyboardAppearance={theme.keyboardAppearance}
autoCapitalize="none"
key="control"
style={{ flex: 1 }}
step={md.step}
minimumValue={minimum}
maximumValue={maximum}
value={props.value}
onValueChange={newValue => void props.updateSettingValue(props.settingId, newValue)}
accessibilityHint={label}
style={styleSheet.settingControl}
value={valueState}
onChangeText={newValue => {
setValueState(newValue);
void props.updateSettingValue(props.settingId, Number.isInteger(Number(newValue)) ? newValue : ''); // Prevent invalid values being mapped to 0
}}
maxLength={15}
secureTextEntry={!!md.secure}
aria-labelledby={labelId}
disableFullscreenUI={true}
/>
</View>
{descriptionComp}
</View>
);
} else if (md.type === Setting.TYPE_STRING) {
Expand Down
25 changes: 0 additions & 25 deletions packages/app-mobile/ios/Podfile.lock
Original file line number Diff line number Diff line change
Expand Up @@ -1060,27 +1060,6 @@ PODS:
- React-Core
- react-native-safe-area-context (4.10.8):
- React-Core
- react-native-slider (4.4.4):
- DoubleConversion
- glog
- hermes-engine
- RCT-Folly (= 2024.01.01.00)
- RCTRequired
- RCTTypeSafety
- React-Codegen
- React-Core
- React-debug
- React-Fabric
- React-featureflags
- React-graphics
- React-ImageManager
- React-NativeModulesApple
- React-RCTFabric
- React-rendererdebug
- React-utils
- ReactCommon/turbomodule/bridging
- ReactCommon/turbomodule/core
- Yoga
- react-native-sqlite-storage (6.0.1):
- React-Core
- react-native-version-info (1.1.1):
Expand Down Expand Up @@ -1442,7 +1421,6 @@ DEPENDENCIES:
- react-native-rsa-native (from `../node_modules/react-native-rsa-native`)
- "react-native-saf-x (from `../node_modules/@joplin/react-native-saf-x`)"
- react-native-safe-area-context (from `../node_modules/react-native-safe-area-context`)
- "react-native-slider (from `../node_modules/@react-native-community/slider`)"
- react-native-sqlite-storage (from `../node_modules/react-native-sqlite-storage`)
- react-native-version-info (from `../node_modules/react-native-version-info`)
- react-native-webview (from `../node_modules/react-native-webview`)
Expand Down Expand Up @@ -1594,8 +1572,6 @@ EXTERNAL SOURCES:
:path: "../node_modules/@joplin/react-native-saf-x"
react-native-safe-area-context:
:path: "../node_modules/react-native-safe-area-context"
react-native-slider:
:path: "../node_modules/@react-native-community/slider"
react-native-sqlite-storage:
:path: "../node_modules/react-native-sqlite-storage"
react-native-version-info:
Expand Down Expand Up @@ -1730,7 +1706,6 @@ SPEC CHECKSUMS:
react-native-rsa-native: 12132eb627797529fdb1f0d22fd0f8f9678df64a
react-native-saf-x: 2b561a9a31413dc594cf2b56a3a8736836b0cf9f
react-native-safe-area-context: b7daa1a8df36095a032dff095a1ea8963cb48371
react-native-slider: 03f213d3ffbf919b16298c7896c1b60101d8e137
react-native-sqlite-storage: f6d515e1c446d1e6d026aa5352908a25d4de3261
react-native-version-info: a106f23009ac0db4ee00de39574eb546682579b9
react-native-webview: 553abd09f58e340fdc7746c9e2ae096839e99911
Expand Down
1 change: 0 additions & 1 deletion packages/app-mobile/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,6 @@
"@react-native-community/geolocation": "3.3.0",
"@react-native-community/netinfo": "11.3.3",
"@react-native-community/push-notification-ios": "1.11.0",
"@react-native-community/slider": "4.5.3",
"assert-browserify": "2.0.0",
"buffer": "6.0.3",
"color": "3.2.1",
Expand Down
8 changes: 6 additions & 2 deletions packages/lib/components/shared/config/config-shared.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ const ObjectUtils = require('../../../ObjectUtils');
const { _ } = require('../../../locale');
import { createSelector } from 'reselect';
import Logger from '@joplin/utils/Logger';

import shim, { MessageBoxType } from '../../../shim';
import { type ReactNode } from 'react';
import { type Registry } from '../../../registry';
import settingValidations from '../../../models/settings/settingValidations';
Expand Down Expand Up @@ -155,7 +155,11 @@ export const saveSettings = async (comp: ConfigScreenComponent) => {

const validationMessage = await settingValidations(savedSettingKeys, comp.state.settings);
if (validationMessage) {
alert(validationMessage);
await shim.showMessageBox(validationMessage, {
type: MessageBoxType.Error,
buttons: [_('OK')],
});

return false;
}

Expand Down
15 changes: 15 additions & 0 deletions packages/lib/models/Setting.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -456,4 +456,19 @@ describe('models/Setting', () => {
await Setting.saveAll();
}
});

test.each([
['1', 1],
['-1', -1],
['+1', 1],
['', null],
[null, 0],
[undefined, 0],
['1e3', 1000],
['1e20', 1e20],
['99999999999999999999', 1e20], // Value exceeds integer limit, output exhibits a rounding issue but still retains integer type
])('should format integer type setting as an integer or null', (async (input, expectedOutput) => {
const value = Setting.formatValue('revisionService.ttlDays', input);
expect(value).toBe(expectedOutput);
}));
});
9 changes: 8 additions & 1 deletion packages/lib/models/Setting.ts
Original file line number Diff line number Diff line change
Expand Up @@ -810,7 +810,14 @@ class Setting extends BaseModel {
public static formatValue(key: string | SettingItemType, value: any) {
const type = typeof key === 'string' ? this.settingMetadata(key).type : key;

if (type === SettingItemType.Int) return !value ? 0 : Math.floor(Number(value));
if (type === SettingItemType.Int) {
if (value === '') {
// Set empty string as null instead of zero, so that validations will fail
return null;
} else {
return !value ? 0 : Math.floor(Number(value));
}
}

if (type === SettingItemType.Bool) {
if (typeof value === 'string') {
Expand Down
Loading
Loading