Skip to content

Commit 6fe2db0

Browse files
committed
Update cookiecutter template
1 parent 25577b3 commit 6fe2db0

File tree

1 file changed

+50
-64
lines changed
  • cookiecutter/{{ cookiecutter.package_name }}/{{ cookiecutter.import_name }}/frontend-react/src

1 file changed

+50
-64
lines changed

cookiecutter/{{ cookiecutter.package_name }}/{{ cookiecutter.import_name }}/frontend-react/src/MyComponent.tsx

+50-64
Original file line numberDiff line numberDiff line change
@@ -1,84 +1,70 @@
11
import {
22
Streamlit,
3-
StreamlitComponentBase,
43
withStreamlitConnection,
4+
ComponentProps,
55
} from "streamlit-component-lib"
6-
import React, { ReactNode } from "react"
7-
8-
interface State {
9-
numClicks: number
10-
isFocused: boolean
11-
}
6+
import React, { useCallback, useEffect, useMemo, useState, ReactElement } from "react"
127

138
/**
14-
* This is a React-based component template. The `render()` function is called
15-
* automatically when your component should be re-rendered.
9+
* This is a React-based component template. The passed props are coming from the
10+
* Streamlit library. Your custom args can be accessed via the `args` props.
1611
*/
17-
class MyComponent extends StreamlitComponentBase<State> {
18-
public state = { numClicks: 0, isFocused: false }
12+
function MyComponent({ args, disabled, theme }: ComponentProps): ReactElement {
13+
const { name } = args
1914

20-
public render = (): ReactNode => {
21-
// Arguments that are passed to the plugin in Python are accessible
22-
// via `this.props.args`. Here, we access the "name" arg.
23-
const name = this.props.args["name"]
15+
const [isFocused, setIsFocused] = useState(false)
16+
const [numClicks, setNumClicks] = useState(0)
2417

25-
// Streamlit sends us a theme object via props that we can use to ensure
26-
// that our component has visuals that match the active theme in a
27-
// streamlit app.
28-
const { theme } = this.props
29-
const style: React.CSSProperties = {}
18+
const style: React.CSSProperties = useMemo(() => {
19+
if (!theme) return {}
3020

31-
// Maintain compatibility with older versions of Streamlit that don't send
32-
// a theme object.
33-
if (theme) {
34-
// Use the theme object to style our button border. Alternatively, the
35-
// theme style is defined in CSS vars.
36-
const borderStyling = `1px solid ${
37-
this.state.isFocused ? theme.primaryColor : "gray"
38-
}`
39-
style.border = borderStyling
40-
style.outline = borderStyling
41-
}
21+
// Use the theme object to style our button border. Alternatively, the
22+
// theme style is defined in CSS vars.
23+
const borderStyling = `1px solid ${isFocused ? theme.primaryColor : "gray"}`
24+
return { border: borderStyling, outline: borderStyling }
25+
}, [theme, isFocused])
4226

43-
// Show a button and some text.
44-
// When the button is clicked, we'll increment our "numClicks" state
45-
// variable, and send its new value back to Streamlit, where it'll
46-
// be available to the Python program.
47-
return (
48-
<span>
49-
Hello, {name}! &nbsp;
50-
<button
51-
style={style}
52-
onClick={this.onClicked}
53-
disabled={this.props.disabled}
54-
onFocus={this._onFocus}
55-
onBlur={this._onBlur}
56-
>
57-
Click Me!
58-
</button>
59-
</span>
60-
)
61-
}
27+
useEffect(() => {
28+
Streamlit.setComponentValue(numClicks)
29+
}, [numClicks])
30+
31+
useEffect(() => {
32+
Streamlit.setFrameHeight()
33+
}, [style, numClicks])
6234

6335
/** Click handler for our "Click Me!" button. */
64-
private onClicked = (): void => {
65-
// Increment state.numClicks, and pass the new value back to
66-
// Streamlit via `Streamlit.setComponentValue`.
67-
this.setState(
68-
prevState => ({ numClicks: prevState.numClicks + 1 }),
69-
() => Streamlit.setComponentValue(this.state.numClicks)
70-
)
71-
}
36+
const onClicked = useCallback((): void => {
37+
setNumClicks((prevNumClicks) => prevNumClicks + 1)
38+
}, [setNumClicks])
7239

7340
/** Focus handler for our "Click Me!" button. */
74-
private _onFocus = (): void => {
75-
this.setState({ isFocused: true })
76-
}
41+
const onFocus = useCallback((): void => {
42+
setIsFocused(true)
43+
}, [setIsFocused])
7744

7845
/** Blur handler for our "Click Me!" button. */
79-
private _onBlur = (): void => {
80-
this.setState({ isFocused: false })
81-
}
46+
const onBlur = useCallback((): void => {
47+
setIsFocused(false)
48+
}, [setIsFocused])
49+
50+
// Show a button and some text.
51+
// When the button is clicked, we'll increment our "numClicks" state
52+
// variable, and send its new value back to Streamlit, where it'll
53+
// be available to the Python program.
54+
return (
55+
<span>
56+
Hello, {name}! &nbsp;
57+
<button
58+
style={style}
59+
onClick={onClicked}
60+
disabled={disabled}
61+
onFocus={onFocus}
62+
onBlur={onBlur}
63+
>
64+
Click Me!
65+
</button>
66+
</span>
67+
)
8268
}
8369

8470
// "withStreamlitConnection" is a wrapper function. It bootstraps the

0 commit comments

Comments
 (0)