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. export interface Props {
  10. navModel: NavModel;
  11. dataSourceTypes: Plugin[];
  12. isLoading: boolean;
  13. addDataSource: typeof addDataSource;
  14. loadDataSourceTypes: typeof loadDataSourceTypes;
  15. dataSourceTypeSearchQuery: string;
  16. setDataSourceTypeSearchQuery: typeof setDataSourceTypeSearchQuery;
  17. }
  18. class NewDataSourcePage extends PureComponent<Props> {
  19. componentDidMount() {
  20. this.props.loadDataSourceTypes();
  21. }
  22. onDataSourceTypeClicked = (plugin: Plugin) => {
  23. this.props.addDataSource(plugin);
  24. };
  25. onSearchQueryChange = (event: React.ChangeEvent<HTMLInputElement>) => {
  26. this.props.setDataSourceTypeSearchQuery(event.target.value);
  27. };
  28. render() {
  29. const { navModel, dataSourceTypes, dataSourceTypeSearchQuery, isLoading } = this.props;
  30. return (
  31. <Page navModel={navModel}>
  32. <Page.Contents isLoading={isLoading}>
  33. <h2 className="add-data-source-header">Choose data source type</h2>
  34. <div className="add-data-source-search">
  35. <label className="gf-form--has-input-icon">
  36. <input
  37. type="text"
  38. className="gf-form-input width-20"
  39. value={dataSourceTypeSearchQuery}
  40. onChange={this.onSearchQueryChange}
  41. placeholder="Filter by name or type"
  42. />
  43. <i className="gf-form-input-icon fa fa-search" />
  44. </label>
  45. </div>
  46. <div className="add-data-source-grid">
  47. {dataSourceTypes.map((plugin, index) => {
  48. return (
  49. <div
  50. onClick={() => this.onDataSourceTypeClicked(plugin)}
  51. className="add-data-source-grid-item"
  52. key={`${plugin.id}-${index}`}
  53. >
  54. <img className="add-data-source-grid-item-logo" src={plugin.info.logos.small} />
  55. <span className="add-data-source-grid-item-text">{plugin.name}</span>
  56. </div>
  57. );
  58. })}
  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));