/* * SonarQube * Copyright (C) 2009-2019 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 { differenceWith } from 'lodash'; import { translate } from 'sonar-ui-common/helpers/l10n'; import { AsyncSelect } from 'sonar-ui-common/components/controls/Select'; import ProjectNotifications from './ProjectNotifications'; import { NotificationProject } from './types'; import { getSuggestions } from '../../../api/components'; import Organization from '../../../components/shared/Organization'; export interface Props { addNotification: (n: T.Notification) => void; channels: string[]; notificationsByProject: T.Dict; projects: NotificationProject[]; removeNotification: (n: T.Notification) => void; types: string[]; } interface State { addedProjects: NotificationProject[]; } export default class Projects extends React.PureComponent { state: State = { addedProjects: [] }; componentWillReceiveProps(nextProps: Props) { // remove all projects from `this.state.addedProjects` // that already exist in `nextProps.projects` this.setState(state => ({ addedProjects: differenceWith( state.addedProjects, Object.keys(nextProps.projects), (stateProject, propsProjectKey) => stateProject.key !== propsProjectKey ) })); } loadOptions = (query: string) => { if (query.length < 2) { return Promise.resolve({ options: [] }); } return getSuggestions(query) .then(r => { const projects = r.results.find(domain => domain.q === 'TRK'); return projects ? projects.items : []; }) .then(projects => { return projects .filter( project => !this.props.projects.find(p => p.key === project.key) && !this.state.addedProjects.find(p => p.key === project.key) ) .map(project => ({ value: project.key, label: project.name, organization: project.organization })); }) .then(options => { return { options }; }); }; handleAddProject = (selected: { label: string; organization: string; value: string }) => { const project = { key: selected.value, name: selected.label, organization: selected.organization }; this.setState(state => ({ addedProjects: [...state.addedProjects, project] })); }; renderOption = (option: { label: string; organization: string; value: string }) => { return ( {option.label} ); }; render() { const allProjects = [...this.props.projects, ...this.state.addedProjects]; return (

{translate('my_profile.per_project_notifications.title')}

{allProjects.length === 0 && (
{translate('my_account.no_project_notifications')}
)} {allProjects.map(project => ( ))}
{translate('my_account.set_notifications_for')}:
); } }