diff --git a/server/build.gradle b/server/build.gradle index 0607d5c68a..5879128f30 100755 --- a/server/build.gradle +++ b/server/build.gradle @@ -7,7 +7,7 @@ plugins { id "jacoco" } -version "0.8.6" +version "0.8.7" group "com.objectcomputing.checkins" repositories { diff --git a/server/src/main/java/com/objectcomputing/checkins/services/memberprofile/birthday/BirthDayServicesImpl.java b/server/src/main/java/com/objectcomputing/checkins/services/memberprofile/birthday/BirthDayServicesImpl.java index bea821aa58..d241cd3786 100644 --- a/server/src/main/java/com/objectcomputing/checkins/services/memberprofile/birthday/BirthDayServicesImpl.java +++ b/server/src/main/java/com/objectcomputing/checkins/services/memberprofile/birthday/BirthDayServicesImpl.java @@ -23,6 +23,13 @@ public BirthDayServicesImpl(MemberProfileServices memberProfileServices) { public List findByValue(String[] months, Integer[] daysOfMonth) { Set memberProfiles = memberProfileServices.findByValues(null, null, null, null, null, null, false); List memberProfileAll = new ArrayList<>(memberProfiles); + if (months == null && daysOfMonth == null) { + // If nothing was passed in, get all members without birthdays. + memberProfileAll = memberProfileAll + .stream() + .filter(member -> member.getBirthDate() == null) + .toList(); + } if (months != null) { for (String month : months) { if (month != null) { @@ -66,7 +73,9 @@ private List profileToBirthDateResponseDto(List { export const getBirthdays = async (months, cookie) => { const results = []; - for (let month of months) { + if (months) { + for (let month of months) { + const res = await resolve({ + url: `${birthdayReportUrl}?month=${month}`, + headers: { 'X-CSRF-Header': cookie, Accept: 'application/json' } + }); + if (res.error) { + console.error(res.error); + } else { + results.push(...res.payload.data); + } + } + } else { const res = await resolve({ - url: `${birthdayReportUrl}?month=${month}`, + url: birthdayReportUrl, headers: { 'X-CSRF-Header': cookie, Accept: 'application/json' } }); - results.push(...res.payload.data); + if (res.error) { + console.error(res.error); + } else { + results.push(...res.payload.data); + } } return results; }; diff --git a/web-ui/src/api/notifications.js b/web-ui/src/api/notifications.js index 2410be5eda..bc40d1b07f 100644 --- a/web-ui/src/api/notifications.js +++ b/web-ui/src/api/notifications.js @@ -3,7 +3,7 @@ import { resolve } from './api.js'; const emailNotificationURL = '/services/email-notifications'; const emailURL = '/services/email'; const testEmailURL = import.meta.env.VITE_APP_API_URL - ? import.meta.env.VITE_APP_URL + '/feedback/submit?request=' + ? import.meta.env.VITE_APP_API_URL + '/feedback/submit?request=' : 'http://localhost:8080/feedback/submit?request='; export const sendReminderNotification = async ( diff --git a/web-ui/src/components/guild-results/EditGuildModal.spec.jsx b/web-ui/src/components/guild-results/EditGuildModal.spec.jsx index 0473bb1e01..dced872d30 100644 --- a/web-ui/src/components/guild-results/EditGuildModal.spec.jsx +++ b/web-ui/src/components/guild-results/EditGuildModal.spec.jsx @@ -34,12 +34,14 @@ const testGuild = { guildMembers: [ { id: 125, name: 'Guild Member' }, { id: 126, name: 'Other Member' } - ] + ], + active: true, }; const emptyGuild = { name: 'Test Guild', - description: 'A guild used for testing.' + description: 'A guild used for testing.', + active: true, }; const currentUserProfile = { diff --git a/web-ui/src/components/kudos_dialog/KudosDialog.jsx b/web-ui/src/components/kudos_dialog/KudosDialog.jsx index 79b6aa5842..92cc98a41a 100644 --- a/web-ui/src/components/kudos_dialog/KudosDialog.jsx +++ b/web-ui/src/components/kudos_dialog/KudosDialog.jsx @@ -29,7 +29,7 @@ import { AppContext } from '../../context/AppContext'; import { selectCsrfToken, selectCurrentUser, - selectNormalizedTeams, + selectActiveTeams, selectOrderedCurrentMemberProfiles, selectProfile } from '../../context/selectors'; @@ -65,7 +65,7 @@ const KudosDialog = ({ open, recipient, teamId, onClose }) => { ); const currentUser = selectCurrentUser(state); - const teams = selectNormalizedTeams(state, ''); + const teams = selectActiveTeams(state); const memberProfiles = selectOrderedCurrentMemberProfiles(state); const handleSubmit = useCallback(() => { diff --git a/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.jsx b/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.jsx index 79cd42f71b..a24fc85109 100644 --- a/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.jsx +++ b/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.jsx @@ -37,14 +37,14 @@ import { AppContext } from '../../../context/AppContext'; import { selectCsrfToken, selectCurrentMembers, - selectGuilds, + selectActiveGuilds, selectMappedUserRoles, selectRoles, selectSkills, selectSubordinates, selectSupervisors, selectTeamMembersBySupervisorId, - selectTeams + selectActiveTeams, } from '../../../context/selectors'; import { UPDATE_TOAST } from '../../../context/actions'; import { getMembersByTeam } from '../../../api/team'; @@ -172,14 +172,14 @@ const MemberSelectorDialog = ({ const getFilterOptions = () => { switch (filterType) { case FilterType.TEAM: - const teams = selectTeams(state); + const teams = selectActiveTeams(state); return { options: teams, label: team => team.name, equals: (team1, team2) => team1.id === team2.id }; case FilterType.GUILD: - const guilds = selectGuilds(state); + const guilds = selectActiveGuilds(state); return { options: guilds, label: guild => guild.name, diff --git a/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.spec.jsx b/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.spec.jsx index 5e5f8cc157..9d36237959 100644 --- a/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.spec.jsx +++ b/web-ui/src/components/member_selector/member_selector_dialog/MemberSelectorDialog.spec.jsx @@ -39,7 +39,8 @@ const testGuild = { name: 'Test Guild', description: 'A guild used for testing.', guildLeads: [{ id: 124, name: managerProfile.name }], - guildMembers: [] + guildMembers: [], + active: true, }; const initialState = { diff --git a/web-ui/src/components/reviews/TeamReviews.jsx b/web-ui/src/components/reviews/TeamReviews.jsx index aa2489d488..5a6d92235f 100644 --- a/web-ui/src/components/reviews/TeamReviews.jsx +++ b/web-ui/src/components/reviews/TeamReviews.jsx @@ -647,9 +647,9 @@ const TeamReviews = ({ onBack, periodId }) => { if (!period.closeDate) return 'No close date was specified.'; if (!period.periodStartDate) return 'No period-start-date was specified.'; if (!period.periodEndDate) return 'No period-end-date was specified.'; - if (teamMembers.length === 0) return 'No members were added.'; - const haveReviewers = teamMembers.every( - member => getReviewers(member).length > 0 + if (assignments.length === 0) return 'No members were added.'; + const haveReviewers = assignments.every( + assignment => assignment.reviewerId != null ); if (!haveReviewers) return 'One or more members have no reviewer.'; return null; // no validation errors diff --git a/web-ui/src/components/team-results/TeamSummaryCard.jsx b/web-ui/src/components/team-results/TeamSummaryCard.jsx index c60abe631a..450390b4bc 100644 --- a/web-ui/src/components/team-results/TeamSummaryCard.jsx +++ b/web-ui/src/components/team-results/TeamSummaryCard.jsx @@ -16,7 +16,8 @@ import { DialogContent, DialogContentText, DialogTitle, - Tooltip + Tooltip, + Typography, } from '@mui/material'; import PropTypes from 'prop-types'; import { updateTeam } from '../../api/team.js'; @@ -34,7 +35,8 @@ const StyledCard = styled(Card)({ width: '340px', display: 'flex', flexDirection: 'column', - justifyContent: 'space-between' + justifyContent: 'space-between', + position: 'relative', }, [`& .${classes.header}`]: { width: '100%' @@ -46,6 +48,11 @@ const StyledCard = styled(Card)({ } }); +const inactiveStyle = { + 'color': 'var(--action-disabled)', + 'font-size': '0.75em', +}; + const propTypes = { team: PropTypes.shape({ id: PropTypes.string, @@ -101,7 +108,7 @@ const TeamSummaryCard = ({ team, index, onTeamSelect, selectedTeamId }) => { title: classes.title, subheader: classes.title }} - title={team.name + (team.active ? ' (Active)' : ' (Inactive)')} + title={team.name} subheader={ { } /> + {!team.active && ( + + Inactive + + )} {team.teamMembers == null ? ( Team Leads: None Assigned diff --git a/web-ui/src/components/team-results/__snapshots__/TeamSummaryCard.test.jsx.snap b/web-ui/src/components/team-results/__snapshots__/TeamSummaryCard.test.jsx.snap index be253e7eb9..6b55c57066 100644 --- a/web-ui/src/components/team-results/__snapshots__/TeamSummaryCard.test.jsx.snap +++ b/web-ui/src/components/team-results/__snapshots__/TeamSummaryCard.test.jsx.snap @@ -3,7 +3,7 @@ exports[`renders correctly 1`] = `
- string (Inactive) + string +

+ Inactive +

Team Leads: @@ -52,7 +57,7 @@ exports[`renders correctly 1`] = ` exports[`renders correctly for ADMIN 1`] = `
- string (Inactive) + string +

+ Inactive +

Team Leads: @@ -142,7 +152,7 @@ exports[`renders correctly for ADMIN 1`] = ` exports[`renders correctly for team lead 1`] = `
- stuff (Inactive) + stuff +

+ Inactive +

Team Leads: diff --git a/web-ui/src/context/selectors.js b/web-ui/src/context/selectors.js index 725aa4a776..9ecfb34ed9 100644 --- a/web-ui/src/context/selectors.js +++ b/web-ui/src/context/selectors.js @@ -686,11 +686,21 @@ export const selectNormalizedTeams = createSelector( }) ); +export const selectActiveTeams = createSelector( + selectTeams, + (teams, searchText) => teams?.filter(team => team.active) +); + +export const selectActiveGuilds = createSelector( + selectGuilds, + (guilds, searchText) => guilds?.filter(guild => guild.active) +); + export const selectMyGuilds = createSelector( selectCurrentUserId, selectGuilds, (id, guilds) => - guilds?.filter(guild => + guilds?.filter(guild => guild.active && guild.guildMembers?.some(member => member.memberId === id) ) ); @@ -699,7 +709,7 @@ export const selectMyTeams = createSelector( selectCurrentUserId, selectTeams, (id, teams) => - teams?.filter(team => + teams?.filter(team => team.active && team.teamMembers?.some(member => member.memberId === id) ) ); diff --git a/web-ui/src/pages/BirthdayAnniversaryReportPage.css b/web-ui/src/pages/BirthdayAnniversaryReportPage.css index 5e40155dc2..b706346b30 100644 --- a/web-ui/src/pages/BirthdayAnniversaryReportPage.css +++ b/web-ui/src/pages/BirthdayAnniversaryReportPage.css @@ -20,7 +20,6 @@ .select-month { display: flex; - justify-content: space-between; align-items: center; margin-left: 15px; margin-right: 10px; diff --git a/web-ui/src/pages/BirthdayReportPage.jsx b/web-ui/src/pages/BirthdayReportPage.jsx index c9aa348b92..af16520155 100644 --- a/web-ui/src/pages/BirthdayReportPage.jsx +++ b/web-ui/src/pages/BirthdayReportPage.jsx @@ -2,7 +2,7 @@ import React, { useContext, useState } from 'react'; import { AppContext } from '../context/AppContext'; -import { Button, TextField } from '@mui/material'; +import { FormControlLabel, Switch, Button, TextField } from '@mui/material'; import Autocomplete from '@mui/material/Autocomplete'; import './BirthdayAnniversaryReportPage.css'; @@ -45,6 +45,7 @@ const BirthdayReportPage = () => { const [selectedMonths, setSelectedMonths] = useState(defaultMonths); const [hasSearched, setHasSearched] = useState(false); const [loading, setLoading] = useState(false); + const [noBirthday, setNoBirthday] = useState(false); useQueryParameters([ { @@ -61,7 +62,8 @@ const BirthdayReportPage = () => { const handleSearch = async monthsToSearch => { setLoading(true); try { - const birthdayResults = await getBirthdays(monthsToSearch, csrf); + const birthdayResults = await getBirthdays(noBirthday ? null : + monthsToSearch, csrf); setSearchBirthdayResults(sortBirthdays(birthdayResults)); setHasSearched(true); } catch(e) { @@ -101,12 +103,25 @@ const BirthdayReportPage = () => { placeholder="Choose a month" /> )} + disabled={noBirthday} + /> + { + const { checked } = event.target; + setNoBirthday(checked); + }} + /> + } + label="No Birthday Registered" />
+