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

Provide tools to search bar as nodes #458

Merged
merged 3 commits into from
Mar 9, 2018

Conversation

cjcenizal
Copy link
Contributor

@cjcenizal cjcenizal commented Mar 3, 2018

Let's compare and contrast this with the config-object-based approach of #452:

    const toolbar = {
      leftTools: [
        {
          type: 'button',
          name: 'Delete Items',
          icon: 'trash',
          color: 'danger',
          available: (table) => table.selection && table.selection.length > 0,
          onClick: (table) => {
            store.deleteUsers(...table.selection.map(user => user.id));
            table.refresh();
          }
        }
      ],
      search: {
        box: {
          incremental: this.state.incremental
        },
        filters: !this.state.filters ? undefined : [
          {
            type: 'is',
            field: 'online',
            name: 'Online',
            negatedName: 'Offline'
          },
          {
            type: 'field_value_selection',
            field: 'nationality',
            name: 'Nationality',
            multiSelect: false,
            options: store.countries.map(country => ({
              value: country.code,
              name: country.name,
              view: `${country.flag} ${country.name}`
            }))
          }
        ]
      }
    };

@nreese
Copy link
Contributor

nreese commented Mar 5, 2018

I like the flexibility that passing in actions as nodes provides. This gives the table consumer a lot of control and future proofs the component for future use cases.

@@ -105,6 +106,22 @@ export class SearchBar extends Component {
this.setState(prevState => ({ incremental: !prevState.incremental }));
};

renderToolsLeft() {
return null;
Copy link
Contributor

Choose a reason for hiding this comment

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

2 returns in method. Second return that renders node never gets used

@uboness
Copy link
Contributor

uboness commented Mar 5, 2018

Can you paste an example snippet in the description (so you can compare)?

Flexibility comes in different shapes any forms... But I'll wait for the snippet

@chrisronline
Copy link
Contributor

I'll let @cjcenizal clarify exactly, but I think the discussion is around:

{
  type: 'button',
  name: 'Delete Items',
  icon: 'trash',
  color: 'danger',
  available: (table) => table.selection && table.selection.length > 0,
  onClick: (table) => {
    store.deleteUsers(...table.selection.map(user => user.id));
    table.refresh();
  }
}

is less expressive than:

table.selection && table.selection.length > 0
? <Button
    iconType="trash"
    color="danger"
    onClick={(table) => {
      store.deleteUsers(...table.selection.map(user => user.id));
      table.refresh();
    }}
  >
    Delete Items
  </Button>
: null

With the general spirit being that we can create expressive UIs through React componentry and that expressiveness leads to better clarity, rather than potentially needing to infer meaning based on config property names (obviously, React component props can suffer from this too, so it's not perfect).

Like in this example, I actually wasn't clear on what available meant but seeing it in the form I provided, for me at least, makes it much more clear. I don't know if we can/should create a steadfast rule for this, but this example feels like a good candidate for a more expressive option.

@cjcenizal
Copy link
Contributor Author

Thanks @chrisronline, I couldn't have put it nearly as well as you just did.

@cjcenizal cjcenizal force-pushed the experiment/tool_bar branch from 878f740 to 083ae96 Compare March 7, 2018 21:05
@nreese
Copy link
Contributor

nreese commented Mar 9, 2018

Is this still an experiment or can it be merged?

@cjcenizal
Copy link
Contributor Author

@nreese I'd like to update a couple other examples and add some tests and I then I think it will be ready for a final review & merge.

@cjcenizal cjcenizal force-pushed the experiment/tool_bar branch from a55420b to f21d167 Compare March 9, 2018 02:58
@cjcenizal cjcenizal changed the title Experiment with providing tools to search bar as nodes Provide tools to search bar as nodes Mar 9, 2018
@cjcenizal
Copy link
Contributor Author

@bmcconaghy @jen-huang @chrisronline @nreese Well, turns out I just needed to add a test. This is ready for review. @uboness mentioned lingering concerns about the necessary boilerplate. The only boilerplate I can see that we need is the check against selection to see if we need to hide or show tools based on whether any rows have been selected. Can anyone spot anything we might be missing here?

Copy link
Contributor

@nreese nreese left a comment

Choose a reason for hiding this comment

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

lgtm

Copy link
Contributor

@chrisronline chrisronline left a comment

Choose a reason for hiding this comment

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

LGTM

@uboness
Copy link
Contributor

uboness commented Mar 9, 2018

@cjcenizal where is the comparison your referring to? am I missing something? (we said we're going to have a comprehensive example and compare).

@uboness
Copy link
Contributor

uboness commented Mar 9, 2018

ah... I see it now... yea... that's a boilerplate I'd rather avoid... I'll take this:

leftTools: [
        {
          type: 'button',
          name: 'Delete Items',
          icon: 'trash',
          color: 'danger',
          available: (table) => table.selection && table.selection.length > 0,
          onClick: (table) => {
            store.deleteUsers(...table.selection.map(user => user.id));
            table.refresh();
          }
        }
      ],

over this:

renderToolsLeft() {
    const selection = this.state.selection;

    if (selection.length === 0) {
      return;
    }

    const onClick = () => {
      store.deleteUsers(...selection.map(user => user.id));
      this.setState({ selection: [] });
    };

    return (
      <EuiButton
        color="danger"
        iconType="trash"
        onClick={onClick}
      >
        Delete {selection.length} Users
      </EuiButton>
    );
  }

any day of the week. And... you now need to track selection state yet again... more boilerplate...

but... seems like kibana likes to boilerplate... 🤷‍♂️... do whatever you think is right

@cjcenizal
Copy link
Contributor Author

Thanks @uboness. We're going to try the approach of passing in nodes first. If the boilerplate proves to be too burdensome then it will be good that we have your config-based approach to fall back to.

@cjcenizal cjcenizal merged commit 8e4305c into elastic:master Mar 9, 2018
@cjcenizal cjcenizal deleted the experiment/tool_bar branch March 9, 2018 20:01
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants