NewDataSourcePage.tsx 2.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. import React, { PureComponent } from 'react';
  2. import { connect } from 'react-redux';
  3. import { hot } from 'react-hot-loader';
  4. import Page from 'app/core/components/Page/Page';
  5. import { NavModel, Plugin, StoreState } from 'app/types';
  6. import { addDataSource, loadDataSourceTypes, setDataSourceTypeSearchQuery } from './state/actions';
  7. import { getNavModel } from 'app/core/selectors/navModel';
  8. import { getDataSourceTypes } from './state/selectors';
  9. import { FilterInput } from 'app/core/components/FilterInput/FilterInput';
  10. export interface Props {
  11. navModel: NavModel;
  12. dataSourceTypes: Plugin[];
  13. isLoading: boolean;
  14. addDataSource: typeof addDataSource;
  15. loadDataSourceTypes: typeof loadDataSourceTypes;
  16. dataSourceTypeSearchQuery: string;
  17. setDataSourceTypeSearchQuery: typeof setDataSourceTypeSearchQuery;
  18. }
  19. class NewDataSourcePage extends PureComponent<Props> {
  20. componentDidMount() {
  21. this.props.loadDataSourceTypes();
  22. }
  23. onDataSourceTypeClicked = (plugin: Plugin) => {
  24. this.props.addDataSource(plugin);
  25. };
  26. onSearchQueryChange = (value: string) => {
  27. this.props.setDataSourceTypeSearchQuery(value);
  28. };
  29. render() {
  30. const { navModel, dataSourceTypes, dataSourceTypeSearchQuery, isLoading } = this.props;
  31. return (
  32. <Page navModel={navModel}>
  33. <Page.Contents isLoading={isLoading}>
  34. <div className="page-container page-body">
  35. <h2 className="add-data-source-header">Choose data source type</h2>
  36. <div className="add-data-source-search">
  37. <FilterInput
  38. labelClassName="gf-form--has-input-icon"
  39. inputClassName="gf-form-input width-20"
  40. value={dataSourceTypeSearchQuery}
  41. onChange={this.onSearchQueryChange}
  42. placeholder="Filter by name or type"
  43. />
  44. </div>
  45. <div className="add-data-source-grid">
  46. {dataSourceTypes.map((plugin, index) => {
  47. return (
  48. <div
  49. onClick={() => this.onDataSourceTypeClicked(plugin)}
  50. className="add-data-source-grid-item"
  51. key={`${plugin.id}-${index}`}
  52. >
  53. <img className="add-data-source-grid-item-logo" src={plugin.info.logos.small} />
  54. <span className="add-data-source-grid-item-text">{plugin.name}</span>
  55. </div>
  56. );
  57. })}
  58. </div>
  59. </div>
  60. </Page.Contents>
  61. </Page>
  62. );
  63. }
  64. }
  65. function mapStateToProps(state: StoreState) {
  66. return {
  67. navModel: getNavModel(state.navIndex, 'datasources'),
  68. dataSourceTypes: getDataSourceTypes(state.dataSources),
  69. isLoading: state.dataSources.isLoadingDataSources,
  70. };
  71. }
  72. const mapDispatchToProps = {
  73. addDataSource,
  74. loadDataSourceTypes,
  75. setDataSourceTypeSearchQuery,
  76. };
  77. export default hot(module)(connect(mapStateToProps, mapDispatchToProps)(NewDataSourcePage));