Skip to content

Commit 6905f49

Browse files
committed
finish documentation, ensure we're passing through style props
1 parent 1ad3191 commit 6905f49

File tree

3 files changed

+131
-6
lines changed

3 files changed

+131
-6
lines changed

README.md

+112-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ A React Native module for parsing and displaying git diffs. This library was hea
44

55
## Overview
66

7-
The purpose of this library is to parse and render a unified diff view for any provided diff(s). The flexible widget system also allows for rendering of custom elements on a per-line basis. The end result will look something like this:
7+
The purpose of this library is to parse and render a unified diff view for any provided diff(s). The flexible widget system also allows for rendering of custom elements on a per-line (or "change") basis. The end result will look something like this:
88

99
![Example diff with a comment widget](./example.png)
1010

@@ -16,16 +16,127 @@ The purpose of this library is to parse and render a unified diff view for any p
1616

1717
### Parsing Diffs
1818

19+
The top-level `parseDiff(diff: string): IFile[]` export is a wrapper around [gitdiff-parser](https://www.npmjs.com/package/gitdiff-parser), but strongly typed
20+
and with some extra options:
21+
22+
- `nearbySequence: 'zip' | undefined` — the action to take when nearby sequences are encountered.
23+
1924
### Rendering Diff Hunks
2025

26+
The top-level `Diff` export is a component to be used for rendering a single diff. Here's a simple example:
27+
28+
```tsx
29+
import React from 'react';
30+
import { View, ScrollView } from 'react-native';
31+
import { parseDiff, Diff, Hunk } from 'react-native-diff-view';
32+
33+
const App = ({ diffText }) => {
34+
const files = parseDiff(diffText);
35+
36+
const renderFile = ({ oldRevision, newRevision, type, hunks }) => (
37+
<ScrollView key={oldRevision + '-' + newRevision} horizontal={true} bounces={false}>
38+
<Diff diffType={type} hunks={hunks}>
39+
{(hunks) => hunks.map((hunk) => <Hunk key={hunk.content} hunk={hunk} />)}
40+
</Diff>
41+
</ScrollView>
42+
);
43+
44+
return (
45+
<View>
46+
{files.map(renderFile)}
47+
</View>
48+
);
49+
};
50+
```
51+
52+
`props.children`, in this case, is a function that takes an array of `IHunk` and returns the rendered element(s). This is optional,
53+
and if not provided the hunks will be rendered as default `<Hunk/ >` components.
54+
2155
### Wrapping Hunks in Decorations
2256

57+
A decoration is a way to wrap a `<Hunk />` component with customized content.
58+
59+
A `<Decoration />` component is a simple passthrough of `props.children`, which can be either a single element or an array of two:
60+
61+
- A single element: this will be rendered in the entire row.
62+
- An array containing two elements: The first element will be rendered in gutter position, the second will be rendered in code position.
63+
64+
A very simple use case of `Decoration` is to provide a summary infomation of hunk:
65+
66+
```tsx
67+
import React from 'react';
68+
import { Diff, Hunk, Decoration } from 'react-native-diff-view';
69+
70+
const renderHunk = (hunk) => [
71+
<Decoration key={'decoration-' + hunk.content}>
72+
{hunk.content}
73+
</Decoration>,
74+
<Hunk key={'hunk-' + hunk.content}> hunk={hunk} />,
75+
];
76+
77+
const DiffFile = ({ diffType, hunks }) => (
78+
<Diff diffType={diffType}>
79+
{hunks.map(renderHunk)}
80+
</Diff>
81+
);
82+
```
83+
2384
### Rendering Widgets
2485

86+
As mentioned above, widgets can be used to render custom element(s) on a per-change, or per-line, basis. These will be rendered
87+
immediately below their corresponding line. Only the first match will be rendered.
88+
89+
Here's a basic example that adds a warning on long lines:
90+
91+
```tsx
92+
import React from 'react';
93+
import { Text } from 'react-native';
94+
import { parseDiff, getChangeKey, Diff } from 'react-native-diff-view';
95+
96+
const getWidgets = (hunks) => {
97+
const changes = hunks.reduce((result, {changes}) => [...result, ...changes], []);
98+
const longLines = changes.filter(({content}) => content.length > 120);
99+
100+
return longLines.reduce(
101+
(widgets, change) => {
102+
const changeKey = getChangeKey(change);
103+
104+
return {
105+
...widgets,
106+
[changeKey]: <Text>Line too long</Text>
107+
};
108+
},
109+
{},
110+
);
111+
};
112+
113+
const App = ({diffText}) => {
114+
const files = parseDiff(diffText);
115+
116+
return (
117+
<div>
118+
{files.map(({hunks}, i) => <Diff key={i} hunks={hunks} widgets={getWidgets(hunks)} viewType="split" />)}
119+
</div>
120+
);
121+
};
122+
```
123+
25124
### Styling
26125

126+
The following props are supported but optional on the top-level `<Diff />` component:
127+
128+
- `style: ViewStyle` &mdash; styling to be applied to the top-level `Diff` view.
129+
- `lineStyle: ViewStyle` &mdash; styling to be applied on each individual change line.
130+
- `gutterStyle: ViewStyle` &mdash; styling to be applied on the gutter of each individual change.
131+
- `contentStyle: ViewStyle` &mdash; styling to be applied to the code content of each individual change.
132+
27133
### Events
28134

135+
The following events are supported but optional on the top-level `<Diff />` component:
136+
137+
- `onChangePress(change: IChange) => any` &mdash; if provided, this will be triggered any time a user presses on a specific
138+
line or change in a diff.
139+
29140
## Contributing
30141

31142
🎊 Thanks for considering contributing!

src/components/diff/index.tsx

+17-4
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@ interface IDiffProps {
1010
hunks: IHunk[];
1111
gutterType?: GutterType;
1212
style?: ViewStyle;
13+
lineStyle?: ViewStyle;
14+
gutterStyle?: ViewStyle;
15+
contentStyle?: ViewStyle;
1316
selectedChanges?: string[];
1417
widgets?: Widgets;
1518
optimizeSelection?: boolean;
@@ -21,7 +24,17 @@ interface IDiffProps {
2124

2225
const Diff: React.FunctionComponent<IDiffProps> = React.memo(
2326
(props): JSX.Element => {
24-
const { diffType, children, style, optimizeSelection, hunks, ...remainings } = props;
27+
const {
28+
diffType,
29+
children,
30+
style,
31+
lineStyle,
32+
gutterStyle,
33+
contentStyle,
34+
optimizeSelection,
35+
hunks,
36+
...remainings
37+
} = props;
2538
const hideGutter = remainings.gutterType === 'none';
2639
const monotonous = diffType === 'add' || diffType === 'delete';
2740
const cols = (() => {
@@ -38,7 +51,7 @@ const Diff: React.FunctionComponent<IDiffProps> = React.memo(
3851
<Provider value={{ ...remainings, monotonous } as IDiffSettings}>
3952
<View style={style}>
4053
{cols}
41-
{children(hunks)}
54+
{children(hunks, { lineStyle, gutterStyle, contentStyle })}
4255
</View>
4356
</Provider>
4457
);
@@ -55,10 +68,10 @@ Diff.defaultProps = {
5568
renderGutter({ textStyle, renderDefault }) {
5669
return renderDefault(textStyle);
5770
},
58-
children(hunks) {
71+
children(hunks, hunkProps = {}) {
5972
const key = (hunk: IHunk): string => `-${hunk.oldStart},${hunk.oldLines} +${hunk.newStart},${hunk.newLines}`;
6073

61-
return hunks.map((hunk) => <Hunk key={key(hunk)} hunk={hunk} />);
74+
return hunks.map((hunk) => <Hunk key={key(hunk)} {...hunkProps} hunk={hunk} />);
6275
},
6376
};
6477

src/types.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import { TextStyle, ViewStyle } from 'react-native';
2+
import { IHunkProps } from './components/hunk';
23

34
export interface IChange {
45
content?: string;
@@ -44,7 +45,7 @@ export type Widgets = Record<string, JSX.Element>;
4445

4546
export type Events = Record<string, (arg: any) => any>;
4647

47-
export type DiffChildren = (hunks: IHunk[]) => JSX.Element | JSX.Element[];
48+
export type DiffChildren = (hunks: IHunk[], hunkProps?: Partial<IHunkProps>) => JSX.Element | JSX.Element[];
4849

4950
export interface IGutterOptions {
5051
change: IChange;

0 commit comments

Comments
 (0)