/* * SonarQube * Copyright (C) 2009-2017 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ import * as React from 'react'; import * as Select from 'react-select'; import { sortBy } from 'lodash'; import BulkApplyTemplateModal from './BulkApplyTemplateModal'; import DeleteModal from './DeleteModal'; import { QUALIFIERS_ORDER } from './utils'; import { Project } from './utils'; import { Organization } from '../../app/types'; import Checkbox from '../../components/controls/Checkbox'; import { translate } from '../../helpers/l10n'; import QualifierIcon from '../../components/shared/QualifierIcon'; import Tooltip from '../../components/controls/Tooltip'; export interface Props { onAllDeselected: () => void; onAllSelected: () => void; onDeleteProjects: () => void; onProvisionedChanged: (provisioned: boolean) => void; onQualifierChanged: (qualifier: string) => void; onSearch: (query: string) => void; organization: Organization; projects: Project[]; provisioned: boolean; qualifiers: string; query: string; ready: boolean; selection: any[]; topLevelQualifiers: string[]; total: number; } interface State { bulkApplyTemplateModal: boolean; deleteModal: boolean; } export default class Search extends React.PureComponent { input: HTMLInputElement; mounted: boolean; state: State = { bulkApplyTemplateModal: false, deleteModal: false }; onSubmit = (event: React.SyntheticEvent) => { event.preventDefault(); this.search(); }; search = (event?: React.SyntheticEvent) => { const q = event ? event.currentTarget.value : this.input.value; this.props.onSearch(q); }; getQualifierOptions = () => { const options = this.props.topLevelQualifiers.map(q => ({ label: translate('qualifiers', q), value: q })); return sortBy(options, option => QUALIFIERS_ORDER.indexOf(option.value)); }; onCheck = (checked: boolean) => { if (checked) { this.props.onAllSelected(); } else { this.props.onAllDeselected(); } }; handleDeleteClick = (event: React.SyntheticEvent) => { event.preventDefault(); event.currentTarget.blur(); this.setState({ deleteModal: true }); }; closeDeleteModal = () => { this.setState({ deleteModal: false }); }; handleDeleteConfirm = () => { this.closeDeleteModal(); this.props.onDeleteProjects(); }; handleBulkApplyTemplateClick = (event: React.SyntheticEvent) => { event.preventDefault(); event.currentTarget.blur(); this.setState({ bulkApplyTemplateModal: true }); }; closeBulkApplyTemplateModal = () => { this.setState({ bulkApplyTemplateModal: false }); }; handleQualifierChange = ({ value }: { value: string }) => this.props.onQualifierChanged(value); renderCheckbox = () => { const isAllChecked = this.props.projects.length > 0 && this.props.selection.length === this.props.projects.length; const thirdState = this.props.projects.length > 0 && this.props.selection.length > 0 && this.props.selection.length < this.props.projects.length; const checked = isAllChecked || thirdState; return ( ); }; renderQualifierOption = (option: { label: string; value: string }) => {option.label} ; renderQualifierFilter = () => { const options = this.getQualifierOptions(); if (options.length < 2) { return null; } return ( (this.input = node!)} className="search-box-input input-medium" type="search" placeholder="Search" /> {this.state.bulkApplyTemplateModal && } {this.state.deleteModal && } ); } }