Skip to content

Commit

Permalink
Merge branch 'main' into blazejkustra/typescript-poc
Browse files Browse the repository at this point in the history
  • Loading branch information
blazejkustra committed Jul 26, 2023
2 parents f48892e + 23a8a34 commit 404cefb
Show file tree
Hide file tree
Showing 132 changed files with 2,872 additions and 1,646 deletions.
4 changes: 2 additions & 2 deletions android/app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -106,8 +106,8 @@ android {
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
multiDexEnabled rootProject.ext.multiDexEnabled
versionCode 1001034400
versionName "1.3.44-0"
versionCode 1001034508
versionName "1.3.45-8"
}

splits {
Expand Down
1 change: 1 addition & 0 deletions assets/animations/ExpensifyLounge.json

Large diffs are not rendered by default.

1 change: 1 addition & 0 deletions assets/animations/WorkspacePlanet.json

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion config/webpack/webpack.dev.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ const {merge} = require('webpack-merge');
const {TimeAnalyticsPlugin} = require('time-analytics-webpack-plugin');
const getCommonConfig = require('./webpack.common');

const BASE_PORT = 8080;
const BASE_PORT = 8082;

/**
* Configuration for the local dev server
Expand Down
2 changes: 1 addition & 1 deletion contributingGuides/CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ If you've found a vulnerability, please email [email protected] with the su
## Payment for Contributions
We hire and pay external contributors via Upwork.com. If you'd like to be paid for contributing or reporting a bug, please create an Upwork account, apply for an available job in [GitHub](https://github.com/Expensify/App/issues?q=is%3Aopen+is%3Aissue+label%3A%22Help+Wanted%22), and finally apply for the job in Upwork once your proposal gets selected in GitHub. If you think your compensation should be increased for a specific job, you can request a reevaluation by commenting in the Github issue where the Upwork job was posted.

Payment for your contributions and bug reports will be made no less than 7 days after the pull request is deployed to production to allow for regression testing. If a regression occurs, payment will be issued 7 days after all regressions are fixed. If you have not received payment after 8 days of the PR being deployed to production and there being no regressions, please email [email protected] referencing the GH issue and your GH handle.
Payment for your contributions and bug reports will be made no less than 7 days after the pull request is deployed to production to allow for regression testing. If a regression occurs, payment will be issued 7 days after all regressions are fixed (ie: deployed to production). A 50% penalty will be applied to the Contributor and Contributor+ for each regression on an issue. If you have not received payment after 8 days of the PR being deployed to production, and there are no regressions, please email [email protected] referencing the GH issue and your GH handle.

New contributors are limited to working on one job at a time, however experienced contributors may work on numerous jobs simultaneously.

Expand Down
2 changes: 1 addition & 1 deletion contributingGuides/FORMS.md
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ Form inputs will NOT store draft values by default. This is to avoid accidentall

### Validate on Blur, on Change and Submit

Each individual form field that requires validation will have its own validate test defined. When the form field loses focus (blur) we will run that validate test and show feedback. A blur on one field will not cause other fields to validate or show errors unless they have already been blurred.
Each individual form field that requires validation will have its own validate test defined. When the form field loses focus (blur) we will run that validate test and show feedback. A blur on one field will not cause other fields to validate or show errors unless they have already been blurred. To prevent server errors from being cleared inadvertently, we only run validation on blur if any form data has changed since the last validation/submit.

Once a user has “touched” an input, i.e. blurred the input, we will also start validating that input on change when the user goes back to editing it.

Expand Down
142 changes: 142 additions & 0 deletions contributingGuides/PROPTYPES_CONVERSION_TABLE.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
# Expensify PropTypes Conversion Table

## Table of Contents

- [Important Considerations](#important-considerations)
- [Don't Rely on `isRequired`](#dont-rely-on-isrequired)
- [PropTypes Conversion Table](#proptypes-conversion-table)
- [Conversion Example](#conversion-example)

## Important Considerations

### Don't Rely on `isRequired`

Regardless of `isRequired` is present or not on props in `PropTypes`, read through the component implementation to check if props without `isRequired` can actually be optional. The use of `isRequired` is not consistent in the current codebase. Just because `isRequired` is not present, it does not necessarily mean that the prop is optional.

One trick is to mark the prop in question with optional modifier `?`. See if the "possibly `undefined`" error is raised by TypeScript. If any error is raised, the implementation assumes the prop not to be optional.

```ts
// Before
const propTypes = {
isVisible: PropTypes.bool.isRequired,
// `confirmText` prop is not marked as required here, theoretically it is optional.
confirmText: PropTypes.string,
};

// After
type Props = {
isVisible: boolean;
// Consider it as required unless you have proof that it is indeed an optional prop.
confirmText: string; // vs. confirmText?: string;
};
```

## PropTypes Conversion Table

| PropTypes | TypeScript | Instructions |
| -------------------------------------------------------------------- | --------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `PropTypes.any` | `T`, `Record<string, unknown>` or `unknown` | Figure out what would be the correct data type and use it.<br><br>If you know that it's a object but isn't possible to determine the internal structure, use `Record<string, unknown>`. |
| `PropTypes.array` or `PropTypes.arrayOf(T)` | `T[]` or `Array<T>` | Convert to `T[]`, where `T` is the data type of the array.<br><br>If `T` isn't a primitive type, create a separate `type` for the object structure of your prop and use it. |
| `PropTypes.bool` | `boolean` | Convert to `boolean`. |
| `PropTypes.func` | `(arg1: Type1, arg2: Type2...) => ReturnType` | Convert to the function signature. |
| `PropTypes.number` | `number` | Convert to `number`. |
| `PropTypes.object`, `PropTypes.shape(T)` or `PropTypes.exact(T)` | `T` | If `T` isn't a primitive type, create a separate `type` for the `T` object structure of your prop and use it.<br><br>If you want an object but it isn't possible to determine the internal structure, use `Record<string, unknown>`. |
| `PropTypes.objectOf(T)` | `Record<string, T>` | Convert to a `Record<string, T>` where `T` is the data type of values stored in the object.<br><br>If `T` isn't a primitive type, create a separate `type` for the object structure and use it. |
| `PropTypes.string` | `string` | Convert to `string`. |
| `PropTypes.node` | `React.ReactNode` | Convert to `React.ReactNode`. `ReactNode` includes `ReactElement` as well as other types such as `strings`, `numbers`, `arrays` of the same, `null`, and `undefined` In other words, anything that can be rendered in React is a `ReactNode`. |
| `PropTypes.element` | `React.ReactElement` | Convert to `React.ReactElement`. |
| `PropTypes.symbol` | `symbol` | Convert to `symbol`. |
| `PropTypes.elementType` | `React.ElementType` | Convert to `React.ElementType`. |
| `PropTypes.instanceOf(T)` | `T` | Convert to `T`. |
| `PropTypes.oneOf([T, U, ...])` or `PropTypes.oneOfType([T, U, ...])` | `T \| U \| ...` | Convert to a union type e.g. `T \| U \| ...`. |

## Conversion Example

```ts
// Before
const propTypes = {
unknownData: PropTypes.any,
anotherUnknownData: PropTypes.any,
indexes: PropTypes.arrayOf(PropTypes.number),
items: PropTypes.arrayOf(
PropTypes.shape({
value: PropTypes.string,
label: PropTypes.string,
})
),
shouldShowIcon: PropTypes.bool,
onChangeText: PropTypes.func,
count: PropTypes.number,
session: PropTypes.shape({
authToken: PropTypes.string,
accountID: PropTypes.number,
}),
errors: PropTypes.objectOf(PropTypes.string),
inputs: PropTypes.objectOf(
PropTypes.shape({
id: PropTypes.string,
label: PropTypes.string,
})
),
label: PropTypes.string,
anchor: PropTypes.node,
footer: PropTypes.element,
uniqSymbol: PropTypes.symbol,
icon: PropTypes.elementType,
date: PropTypes.instanceOf(Date),
size: PropTypes.oneOf(["small", "medium", "large"]),

optionalString: PropTypes.string,
/**
* Note that all props listed above are technically optional because they lack the `isRequired` attribute.
* However, in most cases, props are actually required but the `isRequired` attribute is left out by mistake.
*
* For each prop that appears to be optional, determine whether the component implementation assumes that
* the prop has a value (making it non-optional) or not. Only those props that are truly optional should be
* labeled with a `?` in their type definition.
*/
};

// After
type Item = {
value: string;
label: string;
};

type Session = {
authToken: string;
accountID: number;
};

type Input = {
id: string;
label: string;
};

type Size = "small" | "medium" | "large";

type Props = {
unknownData: string[];

// It's not possible to infer the data as it can be anything because of reasons X, Y and Z.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
anotherUnknownData: unknown;

indexes: number[];
items: Item[];
shouldShowIcon: boolean;
onChangeText: (value: string) => void;
count: number;
session: Session;
errors: Record<string, string>;
inputs: Record<string, Input>;
label: string;
anchor: React.ReactNode;
footer: React.ReactElement;
uniqSymbol: symbol;
icon: React.ElementType;
date: Date;
size: Size;
optionalString?: string;
};
```
Loading

0 comments on commit 404cefb

Please sign in to comment.