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

docs: Component Rules #1055

Merged
merged 16 commits into from
Dec 16, 2024
122 changes: 122 additions & 0 deletions plugins/ui/docs/describing/component_rules.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,122 @@
# Component Rules

Here are some important rules to understand about `deephaven.ui` components.
dgodinez-dh marked this conversation as resolved.
Show resolved Hide resolved

## Children and props

Arguments passed to a component may be either `children` or `props`. `Children` refers to `child` components that are passed to a `parent` component as positional arguments. `Props` are properties passed as keyword arguments that determine the behavior and rendering style of the component. Positional arguments must be included in the correct order. Keyword arguments are included with a keyword and equals sign.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Worth noting that keyword argument ordering doesn't matter, but that ordering of keyword arguments can affect ordering of positional arguments and make them out of order?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added


```python
margaretkennedy marked this conversation as resolved.
Show resolved Hide resolved
from deephaven import ui

my_flex = ui.flex(
ui.heading("Heading"),
ui.button("Button"),
ui.text("Text"),
direction="column",
wrap=True,
width="200px",
)
```

In the above example, the `flex` component is the `parent`. It has three `children`: a `heading`, a `button`, and a `text` component. These `children` will be rendered inside the `flex`. It also has three props: `direction`, `wrap`, and `width`. These three props indicate that the flex should be rendered as a 200 pixel column with wrap enabled.

## Comparison with JSX
margaretkennedy marked this conversation as resolved.
Show resolved Hide resolved

Here is a component written in React JSX.

```html
<MyComponent prop1="value1">Hello World</MyComponent>
```

Here is the same component written in `deephaven.ui`.

```python
my_component("Hello World", prop1="value1")
```

## Defining your own children and props
dgodinez-dh marked this conversation as resolved.
Show resolved Hide resolved

To define `children` and `props` for a custom component, add them as arguments to the component function. As a convention, you may declare the children using the `*` symbol to take any number of arguments.

```python
margaretkennedy marked this conversation as resolved.
Show resolved Hide resolved
from deephaven import ui


@ui.component
def custom_flex(*children, is_column):
return ui.flex(
ui.heading("My Component"),
children,
direction="column" if is_column else "row",
)


my_custom_flex = custom_flex(ui.text("text"), ui.button("button"), is_column=True)
```

## Component return values

A `deephaven.ui` component usually returns a component. It may also return a list or tuple of components. It may return `None` if it should perform logic but does not need to be rendered. It may also return a single value like a `string` or `int`.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this differ at all from any other Python UDF?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes. A Python UDF can return any value. A deephaven.ui component is limited in valid return values.

dgodinez-dh marked this conversation as resolved.
Show resolved Hide resolved

```python
from deephaven import ui


@ui.component
def return_component():
return ui.text("component")


@ui.component
def list_of_components():
return [ui.text("list"), ui.text("of"), ui.text("components")]


@ui.component
def return_tuple():
return (ui.text("a"), ui.text("b"))


@ui.component
def return_none():
print("return none")
return None


@ui.component
def return_string():
return "string"


@ui.component
def return_int():
return 1


my_return_component = return_component()
my_list_of_components = list_of_components()
my_return_tuple = return_tuple()
my_return_none = return_none()
my_return_string = return_string()
my_return_int = return_int()
```

## Conditional return
margaretkennedy marked this conversation as resolved.
Show resolved Hide resolved

Return statements can be conditional in order to render different components based on inputs.

```python
from deephaven import ui


@ui.component
def return_conditional(is_button):
if is_button:
return ui.button("button")
return ui.text("text")


my_button = return_conditional(True)
my_text = return_conditional(False)
```
margaretkennedy marked this conversation as resolved.
Show resolved Hide resolved
4 changes: 4 additions & 0 deletions plugins/ui/docs/sidebar.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,10 @@
{
"label": "Importing and Exporting Components",
"path": "describing/importing_and_exporting_components.md"
},
{
"label": "Component Rules",
"path": "describing/component_rules.md"
}
]
},
Expand Down
Loading