From 816e7615d1b28857b6fba467db367234e62dd3d1 Mon Sep 17 00:00:00 2001 From: Akhilender Bongirwar <112749383+akhilender-bongirwar@users.noreply.github.com> Date: Sun, 19 Nov 2023 00:17:44 +0530 Subject: [PATCH 1/4] test: Achieved 100% test coverage and fixed uncovered lines (#1068) * test: Achieved 100% test coverage and fixed uncovered lines - Improved the test coverage for the User-Password-Update component, addressing the previously uncovered lines and ensuring that all tests pass successfully. - Added two new tests 1. Empty Password Field Test: - The first test ensures that an error is displayed when attempting to save changes with an empty password field. 2. Mismatched New and Confirm Passwords Test - The second test covers the scenario where the new and confirm password fields do not match. With these new tests, I now have 100% test coverage, and there are no more uncovered lines. Signed-off-by: Akhilender * Altered the formData - Altered the formData to make sure all are related to the organization name. Signed-off-by: Akhilender --------- Signed-off-by: Akhilender --- .../UserPasswordUpdate.test.tsx | 66 +++++++++++++++++-- 1 file changed, 62 insertions(+), 4 deletions(-) diff --git a/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx b/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx index 7b07db0cc7..285b430c59 100644 --- a/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx +++ b/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx @@ -3,11 +3,18 @@ import { act, render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; - import { UPDATE_USER_PASSWORD_MUTATION } from 'GraphQl/Mutations/mutations'; import i18nForTest from 'utils/i18nForTest'; import UserPasswordUpdate from './UserPasswordUpdate'; import { StaticMockLink } from 'utils/StaticMockLink'; +import { toast as mockToast } from 'react-toastify'; + +jest.mock('react-toastify', () => ({ + toast: { + error: jest.fn(), + success: jest.fn(), + }, +})); const MOCKS = [ { @@ -48,9 +55,10 @@ describe('Testing User Password Update', () => { }; const formData = { - previousPassword: 'anshgoyal', - newPassword: 'anshgoyalansh', - confirmNewPassword: 'anshgoyalansh', + previousPassword: 'Palisadoes', + newPassword: 'ThePalisadoesFoundation', + wrongPassword: 'This is wrong passoword', + confirmNewPassword: 'ThePalisadoesFoundation', }; global.alert = jest.fn(); @@ -89,4 +97,54 @@ describe('Testing User Password Update', () => { screen.getByPlaceholderText(/Confirm New Password/i) ).toBeInTheDocument(); }); + + test('displays an error when the password field is empty', async () => { + render( + + + + + + ); + + userEvent.click(screen.getByText(/Save Changes/i)); + + await wait(); + expect(mockToast.error).toHaveBeenCalledWith( + 'The password field cannot be empty.' + ); + }); + + test('displays an error when new and confirm password field does not match', async () => { + render( + + + + + + ); + + await wait(); + + userEvent.type( + screen.getByPlaceholderText(/Previous Password/i), + formData.previousPassword + ); + userEvent.type( + screen.getAllByPlaceholderText(/New Password/i)[0], + formData.wrongPassword + ); + userEvent.type( + screen.getByPlaceholderText(/Confirm New Password/i), + formData.confirmNewPassword + ); + + userEvent.click(screen.getByText(/Save Changes/i)); + + expect(screen.getByText(/Cancel/i)).toBeTruthy(); + await wait(); + expect(mockToast.error).toHaveBeenCalledWith( + 'New and Confirm password do not match.' + ); + }); }); From 61cd6356e8a8b36cf2ceb7d621fe643511d012bf Mon Sep 17 00:00:00 2001 From: Kanhaiya yadav <93936630+kanhaiya04@users.noreply.github.com> Date: Tue, 21 Nov 2023 20:10:42 +0530 Subject: [PATCH 2/4] created test for src/components/UserPortal/EventCard/EventCard.tsx (#1079) * created test for eventCard of User portal * corrected the start and end time --- .../UserPortal/EventCard/EventCard.test.tsx | 185 ++++++++++++++++++ 1 file changed, 185 insertions(+) create mode 100644 src/components/UserPortal/EventCard/EventCard.test.tsx diff --git a/src/components/UserPortal/EventCard/EventCard.test.tsx b/src/components/UserPortal/EventCard/EventCard.test.tsx new file mode 100644 index 0000000000..8e5079767e --- /dev/null +++ b/src/components/UserPortal/EventCard/EventCard.test.tsx @@ -0,0 +1,185 @@ +import React from 'react'; +import { MockedProvider } from '@apollo/react-testing'; +import { I18nextProvider } from 'react-i18next'; +import { BrowserRouter } from 'react-router-dom'; +import { ToastContainer } from 'react-toastify'; +import i18nForTest from 'utils/i18nForTest'; +import EventCard from './EventCard'; +import { render, screen, waitFor } from '@testing-library/react'; +import { REGISTER_EVENT } from 'GraphQl/Mutations/mutations'; +import { Provider } from 'react-redux'; +import { store } from 'state/store'; +import { StaticMockLink } from 'utils/StaticMockLink'; +import userEvent from '@testing-library/user-event'; + +const MOCKS = [ + { + request: { + query: REGISTER_EVENT, + variables: { eventId: '123' }, + }, + result: { + data: { + registerForEvent: [ + { + _id: '123', + }, + ], + }, + }, + }, +]; + +const link = new StaticMockLink(MOCKS, true); + +afterEach(() => { + localStorage.clear(); +}); + +describe('Testing Event Card In User portal', () => { + const props = { + id: '123', + title: 'Test Event', + description: 'This is a test event', + location: 'Virtual', + startDate: '2023-04-13', + endDate: '2023-04-15', + isRegisterable: true, + isPublic: true, + endTime: '19:49:12Z', + startTime: '17:49:12Z', + recurring: false, + allDay: true, + creator: { + firstName: 'Joe', + lastName: 'David', + id: '123', + }, + registrants: [ + { + id: '234', + }, + ], + }; + + test('The card should be rendered properly, and all the details should be displayed correct', async () => { + const { queryByText } = render( + + + + + + + + + + + ); + await waitFor(() => expect(queryByText('Test Event')).toBeInTheDocument()); + await waitFor(() => + expect(queryByText('This is a test event')).toBeInTheDocument() + ); + await waitFor(() => expect(queryByText('Location')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('Virtual')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('Starts')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('5:49:12 PM')).toBeInTheDocument()); + await waitFor(() => + expect(queryByText(`13 April '23`)).toBeInTheDocument() + ); + await waitFor(() => expect(queryByText('Ends')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('7:49:12 PM')).toBeInTheDocument()); + await waitFor(() => + expect(queryByText(`15 April '23`)).toBeInTheDocument() + ); + await waitFor(() => expect(queryByText('Creator')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('Joe David')).toBeInTheDocument()); + await waitFor(() => expect(queryByText('Register')).toBeInTheDocument()); + }); + + test('When the user is already registered', async () => { + localStorage.setItem('userId', '234'); + const { queryByText } = render( + + + + + + + + + + + ); + await waitFor(() => + expect(queryByText('Already registered')).toBeInTheDocument() + ); + }); + + test('Handle register should work properly', async () => { + localStorage.setItem('userId', '456'); + const { queryByText } = render( + + + + + + + + + + + ); + userEvent.click(screen.getByText('Register')); + await waitFor(() => + expect( + queryByText('Successfully registered for Test Event') + ).toBeInTheDocument() + ); + }); +}); + +describe('Event card when start and end time are not given', () => { + const props = { + id: '123', + title: 'Test Event', + description: 'This is a test event', + location: 'Virtual', + startDate: '2023-04-13', + endDate: '2023-04-15', + isRegisterable: true, + isPublic: true, + endTime: '', + startTime: '', + recurring: false, + allDay: true, + creator: { + firstName: 'Joe', + lastName: 'David', + id: '123', + }, + registrants: [ + { + id: '234', + }, + ], + }; + + test('Card is rendered correctly', async () => { + const { container } = render( + + + + + + + + + + + ); + + await waitFor(() => + expect(container.querySelector(':empty')).toBeInTheDocument() + ); + }); +}); From 602358416075026b75be66ccb45d8f3afaa87d22 Mon Sep 17 00:00:00 2001 From: Siddhesh Bhupendra Kuakde Date: Thu, 23 Nov 2023 09:02:04 +0530 Subject: [PATCH 3/4] Feature request: Adding advertisement screen (#994) * Add/ test for OrgPost.tsx * fix: org post back to default * Added Dialog 2 * Updated Dialog UI * Removed Extra code * Updated Plugin store * fix: warnings and solves #951 & #948 * fix: warnings and solves #951 & #948 * fix: warnings and solves #951 & #948 * Fix: UI Redesign * fix: merge * fix * Update AddOnStore.tsx * Fixed Merge Errors * Add test: for OrgEntry * Test 3 * fix test 4 * chores: version changes * Add: Initial Websocket setup on talawa mobile web * Add: plugin logic * Add: plugin logic * removed extra * removed extra * Added: Tests * fix * Added WEBSOCKET_URL in .env.example * Advertisement Management Screen * Feature: Create and Delete advertisement * Only current OrgIDs are visible to admin * Showing advertisements in the User end app * Message: fix * formatting * update test * Fix # 1071 * Added test for entry file * Added test for entry file2 * Added test for entry file2 --- .env.example | 5 +- public/locales/en.json | 20 + public/locales/fr.json | 8 + public/locales/hi.json | 8 + public/locales/sp.json | 8 + public/locales/zh.json | 8 + schema.graphql | 1152 +++++++++++++++++ src/App.tsx | 2 + src/GraphQl/Mutations/mutations.ts | 29 +- src/GraphQl/Queries/Queries.ts | 14 +- .../AddOn/core/AddOnStore/AddOnStore.tsx | 2 +- .../Advertisements/Advertisement.module.css | 31 + .../Advertisements/Advertisement.test.tsx | 95 ++ .../Advertisements/Advertisements.tsx | 197 +++ .../AdvertisementEntry.module.css | 20 + .../AdvertisementEntry.test.tsx | 60 + .../AdvertisementEntry/AdvertisementEntry.tsx | 103 ++ .../AdvertisementRegister.module.css | 9 + .../AdvertisementRegister.tsx | 207 +++ .../IconComponent/IconComponent.tsx | 2 + .../OrganizationScreen/OrganizationScreen.tsx | 1 - .../PromotedPost/PromotedPost.module.css | 56 + .../UserPortal/PromotedPost/PromotedPost.tsx | 32 + src/screens/UserPortal/Home/Home.tsx | 37 +- src/state/reducers/routesReducer.test.ts | 19 + src/state/reducers/routesReducer.ts | 2 + 26 files changed, 2120 insertions(+), 7 deletions(-) create mode 100644 schema.graphql create mode 100644 src/components/Advertisements/Advertisement.module.css create mode 100644 src/components/Advertisements/Advertisement.test.tsx create mode 100644 src/components/Advertisements/Advertisements.tsx create mode 100644 src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.module.css create mode 100644 src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx create mode 100644 src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx create mode 100644 src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.module.css create mode 100644 src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx create mode 100644 src/components/UserPortal/PromotedPost/PromotedPost.module.css create mode 100644 src/components/UserPortal/PromotedPost/PromotedPost.tsx diff --git a/.env.example b/.env.example index d7e32ef01d..bd9529ea79 100644 --- a/.env.example +++ b/.env.example @@ -17,4 +17,7 @@ REACT_APP_USE_RECAPTCHA= # from here for reCAPTCHA v2 and "I'm not a robot" Checkbox, and paste the key here. # Note: In domains, fill localhost -REACT_APP_RECAPTCHA_SITE_KEY= \ No newline at end of file +REACT_APP_RECAPTCHA_SITE_KEY= + +# has to be inserted in the env file to use plugins and other websocket based features. +REACT_APP_BACKEND_WEBSOCKET_URL=ws://localhost:4000/graphql \ No newline at end of file diff --git a/public/locales/en.json b/public/locales/en.json index fb5d78bc2f..50945e8dde 100644 --- a/public/locales/en.json +++ b/public/locales/en.json @@ -65,6 +65,7 @@ "Block/Unblock": "Block/Unblock", "Plugins": "Plugins", "Plugin Store": "Plugin Store", + "Advertisement": "Advertisements", "allOrganizations": "All Organizations", "yourOrganization": "Your Organization", "notification": "Notification", @@ -656,6 +657,25 @@ "event": "Event", "organization": "Organization" }, + "advertisement": { + "title": "Advertisements", + "pHeading": "Manage Ads", + "activeAds": "Active Campaigns", + "archievedAds": "Completed Campaigns", + "pMessage": "Ads not present for this campaign.", + "delete": "Delete", + "Rname": "Enter name of Advertisement", + "Rtype": "Select type of Advertisement", + "Rlink": "Provide a link for content to be displayed", + "RstartDate": "Select Start Date", + "RendDate": "Select End Date", + "RClose": "Close the window", + "addNew": "Create new advertisement", + "EXname": "Ex. Cookie Shop", + "EXlink": "Ex. http://yourwebsite.com/photo", + "register": "Create Advertisement", + "close": "Close " + }, "userChat": { "chat": "Chat", "search": "Search", diff --git a/public/locales/fr.json b/public/locales/fr.json index 97116881e9..75522a0fed 100644 --- a/public/locales/fr.json +++ b/public/locales/fr.json @@ -646,6 +646,14 @@ "event": "Événement", "organization": "Organisation" }, + "advertisement": { + "title": "Publicités", + "pHeading": "Gérer les publicités", + "activeAds": "Campagnes actives", + "archievedAds": "Campagnes terminées", + "pMessage": "Aucune publicité n'est présente pour cette campagne.", + "delete": "Supprimer" + }, "userChat": { "chat": "Chat", "search": "Recherche", diff --git a/public/locales/hi.json b/public/locales/hi.json index 490b5b5d04..d952ff7f30 100644 --- a/public/locales/hi.json +++ b/public/locales/hi.json @@ -646,6 +646,14 @@ "event": "आयोजन", "organization": "संगठन" }, + "advertisement": { + "title": "विज्ञापन", + "pHeading": "विज्ञापन प्रबंधन", + "activeAds": "सक्रिय अभियान", + "archievedAds": "संपन्न अभियान", + "pMessage": "इस अभियान के लिए कोई विज्ञापन नहीं हैं।", + "delete": "हटाएँ" + }, "userChat": { "chat": "बात", "search": "खोज", diff --git a/public/locales/sp.json b/public/locales/sp.json index 33cb77ed64..9b107afd44 100644 --- a/public/locales/sp.json +++ b/public/locales/sp.json @@ -646,6 +646,14 @@ "event": "Evento", "organization": "Organización" }, + "advertisement": { + "title": "Anuncios", + "pHeading": "Gestionar anuncios", + "activeAds": "Campañas activas", + "archievedAds": "Campañas completadas", + "pMessage": "No hay anuncios disponibles para esta campaña.", + "delete": "Eliminar" + }, "userChat": { "chat": "Charlar", "search": "Buscar", diff --git a/public/locales/zh.json b/public/locales/zh.json index 8de1d7c5d9..92e8c14c83 100644 --- a/public/locales/zh.json +++ b/public/locales/zh.json @@ -646,6 +646,14 @@ "event": "事件", "organization": "組織" }, + "advertisement": { + "title": "广告", + "pHeading": "管理广告", + "activeAds": "活动广告", + "archievedAds": "已完成的广告活动", + "pMessage": "此广告活动没有相关广告。", + "delete": "删除" + }, "userChat": { "chat": "聊天", "search": "搜尋", diff --git a/schema.graphql b/schema.graphql new file mode 100644 index 0000000000..d200ee7349 --- /dev/null +++ b/schema.graphql @@ -0,0 +1,1152 @@ +directive @auth on FIELD_DEFINITION + +directive @role(requires: UserType) on FIELD_DEFINITION + +type Advertisement { + _id: ID + endDate: Date! + link: String! + name: String! + orgId: ID + startDate: Date! + type: String! +} + +type AggregatePost { + count: Int! +} + +type AggregateUser { + count: Int! +} + +type AndroidFirebaseOptions { + apiKey: String + appId: String + messagingSenderId: String + projectId: String + storageBucket: String +} + +type AuthData { + accessToken: String! + androidFirebaseOptions: AndroidFirebaseOptions! + iosFirebaseOptions: IOSFirebaseOptions! + refreshToken: String! + user: User! +} + +type CheckIn { + _id: ID! + allotedRoom: String + allotedSeat: String + event: Event! + feedbackSubmitted: Boolean! + time: DateTime! + user: User! +} + +input CheckInInput { + allotedRoom: String + allotedSeat: String + eventId: ID! + userId: ID! +} + +type CheckInStatus { + _id: ID! + checkIn: CheckIn + user: User! +} + +type Comment { + _id: ID + createdAt: DateTime + creator: User! + likeCount: Int + likedBy: [User] + post: Post! + text: String! +} + +input CommentInput { + text: String! +} + +union ConnectionError = InvalidCursor | MaximumValueError + +type ConnectionPageInfo { + endCursor: String + hasNextPage: Boolean! + hasPreviousPage: Boolean! + startCursor: String +} + +input CreateUserTagInput { + name: String! + organizationId: ID! + parentTagId: ID +} + +input CursorPaginationInput { + cursor: String + direction: PaginationDirection! + limit: PositiveInt! +} + +scalar Date + +scalar DateTime + +type DeletePayload { + success: Boolean! +} + +type DirectChat { + _id: ID! + creator: User! + messages: [DirectChatMessage] + organization: Organization! + users: [User!]! +} + +type DirectChatMessage { + _id: ID! + createdAt: DateTime! + directChatMessageBelongsTo: DirectChat! + messageContent: String! + receiver: User! + sender: User! +} + +type Donation { + _id: ID! + amount: Float! + nameOfOrg: String! + nameOfUser: String! + orgId: ID! + payPalId: String! + userId: ID! +} + +input DonationWhereInput { + id: ID + id_contains: ID + id_in: [ID!] + id_not: ID + id_not_in: [ID!] + id_starts_with: ID + name_of_user: String + name_of_user_contains: String + name_of_user_in: [String!] + name_of_user_not: String + name_of_user_not_in: [String!] + name_of_user_starts_with: String +} + +scalar EmailAddress + +interface Error { + message: String! +} + +type Event { + _id: ID! + admins(adminId: ID): [User] + allDay: Boolean! + attendees: [User!]! + attendeesCheckInStatus: [CheckInStatus!]! + averageFeedbackScore: Float + creator: User! + description: String! + endDate: Date! + endTime: Time + feedback: [Feedback!]! + isPublic: Boolean! + isRegisterable: Boolean! + latitude: Latitude + location: String + longitude: Longitude + organization: Organization + projects: [EventProject] + recurrance: Recurrance + recurring: Boolean! + startDate: Date! + startTime: Time + status: Status! + title: String! +} + +input EventAttendeeInput { + eventId: ID! + userId: ID! +} + +input EventInput { + allDay: Boolean! + description: String! + endDate: Date + endTime: Time + isPublic: Boolean! + isRegisterable: Boolean! + latitude: Latitude + location: String + longitude: Longitude + organizationId: ID! + recurrance: Recurrance + recurring: Boolean! + startDate: Date! + startTime: Time + title: String! +} + +enum EventOrderByInput { + allDay_ASC + allDay_DESC + description_ASC + description_DESC + endDate_ASC + endDate_DESC + endTime_ASC + endTime_DESC + id_ASC + id_DESC + location_ASC + location_DESC + recurrance_ASC + recurrance_DESC + startDate_ASC + startDate_DESC + startTime_ASC + startTime_DESC + title_ASC + title_DESC +} + +type EventProject { + _id: ID! + description: String! + event: Event! + tasks: [Task] + title: String! +} + +input EventProjectInput { + description: String! + eventId: ID! + title: String! +} + +input EventWhereInput { + description: String + description_contains: String + description_in: [String!] + description_not: String + description_not_in: [String!] + description_starts_with: String + id: ID + id_contains: ID + id_in: [ID!] + id_not: ID + id_not_in: [ID!] + id_starts_with: ID + location: String + location_contains: String + location_in: [String!] + location_not: String + location_not_in: [String!] + location_starts_with: String + organization_id: ID + title: String + title_contains: String + title_in: [String!] + title_not: String + title_not_in: [String!] + title_starts_with: String +} + +type ExtendSession { + accessToken: String! + refreshToken: String! +} + +type Feedback { + _id: ID! + event: Event! + rating: Int! + review: String +} + +input FeedbackInput { + eventId: ID! + rating: Int! + review: String +} + +interface FieldError { + message: String! + path: [String!]! +} + +input ForgotPasswordData { + newPassword: String! + otpToken: String! + userOtp: String! +} + +type Group { + _id: ID + admins: [User] + createdAt: DateTime + description: String + organization: Organization! + title: String +} + +type GroupChat { + _id: ID! + creator: User! + messages: [GroupChatMessage] + organization: Organization! + users: [User!]! +} + +type GroupChatMessage { + _id: ID! + createdAt: DateTime! + groupChatMessageBelongsTo: GroupChat! + messageContent: String! + sender: User! +} + +type IOSFirebaseOptions { + apiKey: String + appId: String + iosBundleId: String + iosClientId: String + messagingSenderId: String + projectId: String + storageBucket: String +} + +type InvalidCursor implements FieldError { + message: String! + path: [String!]! +} + +type Language { + _id: ID! + createdAt: String! + en: String! + translation: [LanguageModel] +} + +input LanguageInput { + en_value: String! + translation_lang_code: String! + translation_value: String! +} + +type LanguageModel { + _id: ID! + createdAt: DateTime! + lang_code: String! + value: String! + verified: Boolean! +} + +scalar Latitude + +input LoginInput { + email: EmailAddress! + password: String! +} + +scalar Longitude + +type MaximumLengthError implements FieldError { + message: String! + path: [String!]! +} + +type MaximumValueError implements FieldError { + limit: Int! + message: String! + path: [String!]! +} + +type MembershipRequest { + _id: ID! + organization: Organization! + user: User! +} + +type Message { + _id: ID! + createdAt: DateTime + creator: User + imageUrl: URL + text: String + videoUrl: URL +} + +type MessageChat { + _id: ID! + createdAt: DateTime! + languageBarrier: Boolean + message: String! + receiver: User! + sender: User! +} + +input MessageChatInput { + message: String! + receiver: ID! +} + +type MinimumLengthError implements FieldError { + limit: Int! + message: String! + path: [String!]! +} + +type MinimumValueError implements FieldError { + message: String! + path: [String!]! +} + +type Mutation { + acceptAdmin(id: ID!): Boolean! + acceptMembershipRequest(membershipRequestId: ID!): MembershipRequest! + addEventAttendee(data: EventAttendeeInput!): User! + addFeedback(data: FeedbackInput!): Feedback! + addLanguageTranslation(data: LanguageInput!): Language! + addOrganizationImage(file: String!, organizationId: String!): Organization! + addUserImage(file: String!): User! + addUserToGroupChat(chatId: ID!, userId: ID!): GroupChat! + adminRemoveEvent(eventId: ID!): Event! + adminRemoveGroup(groupId: ID!): GroupChat! + assignUserTag(input: ToggleUserTagAssignInput!): User + blockPluginCreationBySuperadmin(blockUser: Boolean!, userId: ID!): User! + blockUser(organizationId: ID!, userId: ID!): User! + cancelMembershipRequest(membershipRequestId: ID!): MembershipRequest! + checkIn(data: CheckInInput!): CheckIn! + createAdmin(data: UserAndOrganizationInput!): User! + createAdvertisement( + endDate: Date! + link: String! + name: String! + orgId: ID! + startDate: Date! + type: String! + ): Advertisement! + createComment(data: CommentInput!, postId: ID!): Comment + createDirectChat(data: createChatInput!): DirectChat! + createDonation( + amount: Float! + nameOfOrg: String! + nameOfUser: String! + orgId: ID! + payPalId: ID! + userId: ID! + ): Donation! + createEvent(data: EventInput): Event! + createEventProject(data: EventProjectInput!): EventProject! + createGroupChat(data: createGroupChatInput!): GroupChat! + createMember(input: UserAndOrganizationInput!): Organization! + createMessageChat(data: MessageChatInput!): MessageChat! + createOrganization(data: OrganizationInput, file: String): Organization! + createPlugin( + pluginCreatedBy: String! + pluginDesc: String! + pluginName: String! + uninstalledOrgs: [ID!] + ): Plugin! + createPost(data: PostInput!, file: String): Post + createTask(data: TaskInput!, eventProjectId: ID!): Task! + createUserTag(input: CreateUserTagInput!): UserTag + deleteAdvertisementById(id: ID!): DeletePayload! + deleteDonationById(id: ID!): DeletePayload! + forgotPassword(data: ForgotPasswordData!): Boolean! + joinPublicOrganization(organizationId: ID!): User! + leaveOrganization(organizationId: ID!): User! + likeComment(id: ID!): Comment + likePost(id: ID!): Post + login(data: LoginInput!): AuthData! + logout: Boolean! + otp(data: OTPInput!): OtpData! + recaptcha(data: RecaptchaVerification!): Boolean! + refreshToken(refreshToken: String!): ExtendSession! + registerForEvent(id: ID!): Event! + rejectAdmin(id: ID!): Boolean! + rejectMembershipRequest(membershipRequestId: ID!): MembershipRequest! + removeAdmin(data: UserAndOrganizationInput!): User! + removeAdvertisement(id: ID!): Advertisement + removeComment(id: ID!): Comment + removeDirectChat(chatId: ID!, organizationId: ID!): DirectChat! + removeEvent(id: ID!): Event! + removeEventAttendee(data: EventAttendeeInput!): User! + removeEventProject(id: ID!): EventProject! + removeGroupChat(chatId: ID!): GroupChat! + removeMember(data: UserAndOrganizationInput!): Organization! + removeOrganization(id: ID!): User! + removeOrganizationImage(organizationId: String!): Organization! + removePost(id: ID!): Post + removeTask(id: ID!): Task + removeUserFromGroupChat(chatId: ID!, userId: ID!): GroupChat! + removeUserImage: User! + removeUserTag(id: ID!): UserTag + revokeRefreshTokenForUser(userId: String!): Boolean! + saveFcmToken(token: String): Boolean! + sendMembershipRequest(organizationId: ID!): MembershipRequest! + sendMessageToDirectChat( + chatId: ID! + messageContent: String! + ): DirectChatMessage! + sendMessageToGroupChat( + chatId: ID! + messageContent: String! + ): GroupChatMessage! + setTaskVolunteers(id: ID!, volunteers: [ID]!): Task + signUp(data: UserInput!, file: String): AuthData! + togglePostPin(id: ID!): Post! + unassignUserTag(input: ToggleUserTagAssignInput!): User + unblockUser(organizationId: ID!, userId: ID!): User! + unlikeComment(id: ID!): Comment + unlikePost(id: ID!): Post + unregisterForEventByUser(id: ID!): Event! + updateEvent(data: UpdateEventInput, id: ID!): Event! + updateEventProject(data: UpdateEventProjectInput!, id: ID!): EventProject! + updateLanguage(languageCode: String!): User! + updateOrganization( + data: UpdateOrganizationInput + file: String + id: ID! + ): Organization! + updatePluginStatus(id: ID!, orgId: ID!): Plugin! + updatePost(data: PostUpdateInput, id: ID!): Post! + updateTask(data: UpdateTaskInput!, id: ID!): Task + updateUserPassword(data: UpdateUserPasswordInput!): User! + updateUserProfile(data: UpdateUserInput, file: String): User! + updateUserTag(input: UpdateUserTagInput!): UserTag + updateUserType(data: UpdateUserTypeInput!): Boolean! +} + +input OTPInput { + email: EmailAddress! +} + +type Organization { + _id: ID! + admins(adminId: ID): [User] + apiUrl: URL! + blockedUsers: [User] + createdAt: DateTime + creator: User! + description: String! + image: String + isPublic: Boolean! + location: String + members: [User] + membershipRequests: [MembershipRequest] + name: String! + pinnedPosts: [Post] + userTags( + after: String + before: String + first: PositiveInt + last: PositiveInt + ): UserTagsConnection + visibleInSearch: Boolean! +} + +type OrganizationInfoNode { + _id: ID! + apiUrl: URL! + creator: User! + description: String! + image: String + isPublic: Boolean! + name: String! + visibleInSearch: Boolean! +} + +input OrganizationInput { + apiUrl: URL + attendees: String + description: String! + image: String + isPublic: Boolean! + location: String + name: String! + visibleInSearch: Boolean! +} + +enum OrganizationOrderByInput { + apiUrl_ASC + apiUrl_DESC + description_ASC + description_DESC + id_ASC + id_DESC + name_ASC + name_DESC +} + +input OrganizationWhereInput { + apiUrl: URL + apiUrl_contains: URL + apiUrl_in: [URL!] + apiUrl_not: URL + apiUrl_not_in: [URL!] + apiUrl_starts_with: URL + description: String + description_contains: String + description_in: [String!] + description_not: String + description_not_in: [String!] + description_starts_with: String + id: ID + id_contains: ID + id_in: [ID!] + id_not: ID + id_not_in: [ID!] + id_starts_with: ID + isPublic: Boolean + name: String + name_contains: String + name_in: [String!] + name_not: String + name_not_in: [String!] + name_starts_with: String + visibleInSearch: Boolean +} + +type OtpData { + otpToken: String! +} + +""" +Information about pagination in a connection. +""" +type PageInfo { + currPageNo: Int + + """ + When paginating forwards, are there more items? + """ + hasNextPage: Boolean! + + """ + When paginating backwards, are there more items? + """ + hasPreviousPage: Boolean! + nextPageNo: Int + prevPageNo: Int + totalPages: Int +} + +enum PaginationDirection { + BACKWARD + FORWARD +} + +scalar PhoneNumber + +type Plugin { + _id: ID! + pluginCreatedBy: String! + pluginDesc: String! + pluginName: String! + uninstalledOrgs: [ID!]! +} + +type PluginField { + createdAt: DateTime + key: String! + status: Status! + value: String! +} + +input PluginFieldInput { + key: String! + value: String! +} + +input PluginInput { + fields: [PluginFieldInput] + orgId: ID! + pluginKey: String + pluginName: String! + pluginType: Type +} + +scalar PositiveInt + +type Post { + _id: ID + commentCount: Int + comments: [Comment] + createdAt: DateTime + creator: User! + imageUrl: URL + likeCount: Int + likedBy: [User] + organization: Organization! + pinned: Boolean + text: String! + title: String + videoUrl: URL +} + +""" +A connection to a list of items. +""" +type PostConnection { + aggregate: AggregatePost! + + """ + A list of edges. + """ + edges: [Post]! + + """ + Information to aid in pagination. + """ + pageInfo: PageInfo! +} + +input PostInput { + _id: ID + imageUrl: URL + organizationId: ID! + pinned: Boolean + text: String! + title: String + videoUrl: URL +} + +enum PostOrderByInput { + commentCount_ASC + commentCount_DESC + createdAt_ASC + createdAt_DESC + id_ASC + id_DESC + imageUrl_ASC + imageUrl_DESC + likeCount_ASC + likeCount_DESC + text_ASC + text_DESC + title_ASC + title_DESC + videoUrl_ASC + videoUrl_DESC +} + +input PostUpdateInput { + imageUrl: String + text: String + title: String + videoUrl: String +} + +input PostWhereInput { + id: ID + id_contains: ID + id_in: [ID!] + id_not: ID + id_not_in: [ID!] + id_starts_with: ID + text: String + text_contains: String + text_in: [String!] + text_not: String + text_not_in: [String!] + text_starts_with: String + title: String + title_contains: String + title_in: [String!] + title_not: String + title_not_in: [String!] + title_starts_with: String +} + +type Query { + adminPlugin(orgId: ID!): [Plugin] + checkAuth: User! + directChatsByUserID(id: ID!): [DirectChat] + directChatsMessagesByChatID(id: ID!): [DirectChatMessage] + event(id: ID!): Event + eventsByOrganization(id: ID, orderBy: EventOrderByInput): [Event] + eventsByOrganizationConnection( + first: Int + orderBy: EventOrderByInput + skip: Int + where: EventWhereInput + ): [Event!]! + getAdvertisements: [Advertisement] + getDonationById(id: ID!): Donation! + getDonationByOrgId(orgId: ID!): [Donation] + getDonationByOrgIdConnection( + first: Int + orgId: ID! + skip: Int + where: DonationWhereInput + ): [Donation!]! + getPlugins: [Plugin] + getlanguage(lang_code: String!): [Translation] + hasSubmittedFeedback(eventId: ID!, userId: ID!): Boolean + joinedOrganizations(id: ID): [Organization] + me: User! + myLanguage: String + organizations(id: ID, orderBy: OrganizationOrderByInput): [Organization] + organizationsConnection( + first: Int + orderBy: OrganizationOrderByInput + skip: Int + where: OrganizationWhereInput + ): [Organization]! + organizationsMemberConnection( + first: Int + orderBy: UserOrderByInput + orgId: ID! + skip: Int + where: UserWhereInput + ): UserConnection! + plugin(orgId: ID!): [Plugin] + post(id: ID!): Post + postsByOrganization(id: ID!, orderBy: PostOrderByInput): [Post] + postsByOrganizationConnection( + first: Int + id: ID! + orderBy: PostOrderByInput + skip: Int + where: PostWhereInput + ): PostConnection + registeredEventsByUser(id: ID, orderBy: EventOrderByInput): [Event] + registrantsByEvent(id: ID!): [User] + user(id: ID!): User! + userLanguage(userId: ID!): String + users(orderBy: UserOrderByInput, where: UserWhereInput): [User] + usersConnection( + first: Int + orderBy: UserOrderByInput + skip: Int + where: UserWhereInput + ): [User]! +} + +input RecaptchaVerification { + recaptchaToken: String! +} + +enum Recurrance { + DAILY + MONTHLY + ONCE + WEEKLY + YEARLY +} + +enum Status { + ACTIVE + BLOCKED + DELETED +} + +type Subscription { + directMessageChat: MessageChat + messageSentToDirectChat: DirectChatMessage + messageSentToGroupChat: GroupChatMessage + onPluginUpdate: Plugin +} + +type Task { + _id: ID! + completed: Boolean + createdAt: DateTime! + creator: User! + deadline: DateTime + description: String + event: Event! + title: String! + volunteers: [User] +} + +input TaskInput { + deadline: DateTime! + description: String! + title: String! +} + +enum TaskOrderByInput { + createdAt_ASC + createdAt_DESC + deadline_ASC + deadline_DESC + description_ASC + description_DESC + id_ASC + id_DESC + title_ASC + title_DESC +} + +scalar Time + +input ToggleUserTagAssignInput { + tagId: ID! + userId: ID! +} + +type Translation { + en_value: String + lang_code: String + translation: String + verified: Boolean +} + +enum Type { + PRIVATE + UNIVERSAL +} + +scalar URL + +type UnauthenticatedError implements Error { + message: String! +} + +type UnauthorizedError implements Error { + message: String! +} + +input UpdateEventInput { + allDay: Boolean + description: String + endDate: Date + endTime: Time + isPublic: Boolean + isRegisterable: Boolean + latitude: Latitude + location: String + longitude: Longitude + recurrance: Recurrance + recurring: Boolean + startDate: Date + startTime: Time + title: String +} + +input UpdateEventProjectInput { + description: String + title: String +} + +input UpdateOrganizationInput { + description: String + isPublic: Boolean + location: String + name: String + visibleInSearch: Boolean +} + +input UpdateTaskInput { + completed: Boolean + deadline: DateTime + description: String + title: String +} + +input UpdateUserInput { + email: EmailAddress + firstName: String + lastName: String +} + +input UpdateUserPasswordInput { + confirmNewPassword: String! + newPassword: String! + previousPassword: String! +} + +input UpdateUserTagInput { + _id: ID! + name: String! +} + +input UpdateUserTypeInput { + id: ID + userType: String +} + +scalar Upload + +type User { + _id: ID! + adminApproved: Boolean + adminFor: [Organization] + appLanguageCode: String! + assignedTasks: [Task] + createdAt: DateTime + createdEvents: [Event] + createdOrganizations: [Organization] + email: EmailAddress! + eventAdmin: [Event] + firstName: String! + image: String + joinedOrganizations: [Organization] + lastName: String! + membershipRequests: [MembershipRequest] + organizationUserBelongsTo: Organization + organizationsBlockedBy: [Organization] + pluginCreationAllowed: Boolean + registeredEvents: [Event] + tagsAssignedWith( + after: String + before: String + first: PositiveInt + last: PositiveInt + organizationId: ID + ): UserTagsConnection + tokenVersion: Int! + userType: String +} + +input UserAndOrganizationInput { + organizationId: ID! + userId: ID! +} + +type UserConnection { + aggregate: AggregateUser! + edges: [User]! + pageInfo: PageInfo! +} + +type UserEdge { + cursor: String! + node: User! +} + +input UserInput { + appLanguageCode: String + email: EmailAddress! + firstName: String! + lastName: String! + organizationUserBelongsToId: ID + password: String! +} + +enum UserOrderByInput { + appLanguageCode_ASC + appLanguageCode_DESC + email_ASC + email_DESC + firstName_ASC + firstName_DESC + id_ASC + id_DESC + lastName_ASC + lastName_DESC +} + +type UserTag { + _id: ID! + childTags(input: UserTagsConnectionInput!): UserTagsConnectionResult! + name: String! + organization: Organization + parentTag: UserTag + usersAssignedTo(input: UsersConnectionInput!): UsersConnectionResult! +} + +type UserTagEdge { + cursor: String! + node: UserTag! +} + +type UserTagsConnection { + edges: [UserTagEdge!]! + pageInfo: ConnectionPageInfo! +} + +input UserTagsConnectionInput { + cursor: String + direction: PaginationDirection! + limit: PositiveInt! +} + +type UserTagsConnectionResult { + data: UserTagsConnection + errors: [ConnectionError!]! +} + +enum UserType { + ADMIN + SUPERADMIN + USER +} + +input UserWhereInput { + admin_for: ID + appLanguageCode: String + appLanguageCode_contains: String + appLanguageCode_in: [String!] + appLanguageCode_not: String + appLanguageCode_not_in: [String!] + appLanguageCode_starts_with: String + email: EmailAddress + email_contains: EmailAddress + email_in: [EmailAddress!] + email_not: EmailAddress + email_not_in: [EmailAddress!] + email_starts_with: EmailAddress + event_title_contains: String + firstName: String + firstName_contains: String + firstName_in: [String!] + firstName_not: String + firstName_not_in: [String!] + firstName_starts_with: String + id: ID + id_contains: ID + id_in: [ID!] + id_not: ID + id_not_in: [ID!] + id_starts_with: ID + lastName: String + lastName_contains: String + lastName_in: [String!] + lastName_not: String + lastName_not_in: [String!] + lastName_starts_with: String +} + +type UsersConnection { + edges: [UserEdge!]! + pageInfo: ConnectionPageInfo! +} + +input UsersConnectionInput { + cursor: String + direction: PaginationDirection! + limit: PositiveInt! +} + +type UsersConnectionResult { + data: UsersConnection + errors: [ConnectionError!]! +} + +input createChatInput { + organizationId: ID! + userIds: [ID!]! +} + +input createGroupChatInput { + organizationId: ID! + title: String! + userIds: [ID!]! +} diff --git a/src/App.tsx b/src/App.tsx index cf20c722e5..1c69256c32 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -32,6 +32,7 @@ import Settings from 'screens/UserPortal/Settings/Settings'; import Donate from 'screens/UserPortal/Donate/Donate'; import Events from 'screens/UserPortal/Events/Events'; import Tasks from 'screens/UserPortal/Tasks/Tasks'; +import Advertisements from 'components/Advertisements/Advertisements'; import Chat from 'screens/UserPortal/Chat/Chat'; function app(): JSX.Element { @@ -109,6 +110,7 @@ function app(): JSX.Element { + diff --git a/src/GraphQl/Mutations/mutations.ts b/src/GraphQl/Mutations/mutations.ts index dca9e713c3..34abda2b26 100644 --- a/src/GraphQl/Mutations/mutations.ts +++ b/src/GraphQl/Mutations/mutations.ts @@ -400,7 +400,34 @@ export const ADD_PLUGIN_MUTATION = gql` } } `; - +export const ADD_ADVERTISEMENT_MUTATION = gql` + mutation ( + $orgId: ID! + $name: String! + $link: String! + $type: String! + $startDate: Date! + $endDate: Date! + ) { + createAdvertisement( + orgId: $orgId + name: $name + link: $link + type: $type + startDate: $startDate + endDate: $endDate + ) { + _id + } + } +`; +export const DELETE_ADVERTISEMENT_BY_ID = gql` + mutation ($id: ID!) { + deleteAdvertisementById(id: $id) { + success + } + } +`; export const UPDATE_POST_MUTATION = gql` mutation UpdatePost( $id: ID! diff --git a/src/GraphQl/Queries/Queries.ts b/src/GraphQl/Queries/Queries.ts index 500dde6192..7bf99d48ee 100644 --- a/src/GraphQl/Queries/Queries.ts +++ b/src/GraphQl/Queries/Queries.ts @@ -704,7 +704,19 @@ export const PLUGIN_GET = gql` } } `; - +export const ADVERTISEMENTS_GET = gql` + query getAdvertisement { + getAdvertisements { + _id + name + type + orgId + link + endDate + startDate + } + } +`; export const ORGANIZATION_EVENTS_CONNECTION = gql` query EventsByOrganizationConnection( $organization_id: ID! diff --git a/src/components/AddOn/core/AddOnStore/AddOnStore.tsx b/src/components/AddOn/core/AddOnStore/AddOnStore.tsx index 2e3e511149..d81ce6fa4c 100644 --- a/src/components/AddOn/core/AddOnStore/AddOnStore.tsx +++ b/src/components/AddOn/core/AddOnStore/AddOnStore.tsx @@ -160,7 +160,7 @@ function addOnStore(): JSX.Element { Search results for {searchText}

) : null} - + = new ApolloClient({ + cache: new InMemoryCache(), + link: ApolloLink.from([httpLink]), +}); +describe('Testing Advertisement Component', () => { + test('Temporary test for Advertisement', () => { + expect(true).toBe(true); + const { getByTestId } = render( + + + + + {} + + + + + ); + expect(getByTestId('AdEntryStore')).toBeInTheDocument(); + }); + + test('renders advertisement data', async () => { + const mocks = [ + { + request: { + query: ADVERTISEMENTS_GET, + variables: { + name: 'Test', + }, + }, + result: { + data: { + getAdvertisements: [ + { + _id: '1', + name: 'Advertisement', + type: 'POPUP', + orgId: 'org1', + link: 'http://example.com', + endDate: new Date(), + startDate: new Date(), + }, + // Add more mock data if needed + ], + }, + loading: false, + }, + }, + ]; + + const { getByTestId } = render( + + + + + + + + + + + + ); + + expect(getByTestId('AdEntryStore')).toBeInTheDocument(); + }); +}); diff --git a/src/components/Advertisements/Advertisements.tsx b/src/components/Advertisements/Advertisements.tsx new file mode 100644 index 0000000000..2d9d936d1e --- /dev/null +++ b/src/components/Advertisements/Advertisements.tsx @@ -0,0 +1,197 @@ +/* eslint-disable @typescript-eslint/no-unused-vars */ +import React, { useEffect, useState } from 'react'; +import PropTypes from 'react'; +import styles from './Advertisement.module.css'; +import { useQuery } from '@apollo/client'; +import { ADVERTISEMENTS_GET, PLUGIN_GET } from 'GraphQl/Queries/Queries'; // PLUGIN_LIST +import { useSelector } from 'react-redux'; +import type { RootState } from '../../state/reducers'; +import { Col, Form, Row, Tab, Tabs } from 'react-bootstrap'; +import PluginHelper from 'components/AddOn/support/services/Plugin.helper'; +import { store } from 'state/store'; +import { useTranslation } from 'react-i18next'; +import Loader from 'components/Loader/Loader'; +import OrganizationScreen from 'components/OrganizationScreen/OrganizationScreen'; +import AdvertisementEntry from './core/AdvertisementEntry/AdvertisementEntry'; +import AdvertisementRegister from './core/AdvertisementRegister/AdvertisementRegister'; +import AddOnRegister from 'components/AddOn/core/AddOnRegister/AddOnRegister'; +export default function advertisements(): JSX.Element { + const { + data: data2, + loading: loading2, + error: error2, + } = useQuery(ADVERTISEMENTS_GET); + const currentOrgId = window.location.href.split('/id=')[1] + ''; + const { t } = useTranslation('translation', { keyPrefix: 'advertisement' }); + document.title = t('title'); + + const [isStore, setIsStore] = useState(true); + const [showEnabled, setShowEnabled] = useState(true); + const [searchText, setSearchText] = useState(''); + const [dataList, setDataList] = useState([]); + + const [render, setRender] = useState(true); + const appRoutes = useSelector((state: RootState) => state.appRoutes); + const { targets, configUrl } = appRoutes; + + const plugins = useSelector((state: RootState) => state.plugins); + const { installed, addonStore } = plugins; + const { data, loading, error } = useQuery(PLUGIN_GET); + /* istanbul ignore next */ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + const getStorePlugins = async () => { + let plugins = await new PluginHelper().fetchStore(); + const installIds = (await new PluginHelper().fetchInstalled()).map( + (plugin: any) => plugin.id + ); + plugins = plugins.map((plugin: any) => { + plugin.installed = installIds.includes(plugin.id); + return plugin; + }); + store.dispatch({ type: 'UPDATE_STORE', payload: plugins }); + }; + + /* istanbul ignore next */ + const getInstalledPlugins: () => any = () => { + setDataList(data); + }; + // const getAdvertisements: () => any = ()=> { + // return + // } + + /* istanbul ignore next */ + const updateLinks = async (links: any[]): Promise => { + store.dispatch({ type: 'UPDATE_P_TARGETS', payload: links }); + }; + // /* istanbul ignore next */ + const pluginModified = (): void => { + return getInstalledPlugins(); + // .then((installedPlugins) => { + // getStorePlugins(); + // return installedPlugins; + // }); + }; + + const updateSelectedTab = (tab: any): void => { + setIsStore(tab === 'activeAds'); + isStore ? getStorePlugins() : getInstalledPlugins(); + }; + + const filterChange = (ev: any): void => { + setShowEnabled(ev.target.value === 'enabled'); + }; + + /* istanbul ignore next */ + if (loading) { + return ( + <> +
+ + ); + } + + return ( + <> + + + +
+

{t('pHeading')}

+ + + + + {data2?.getAdvertisements + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) > new Date()) + .length == 0 ? ( +

{t('pMessage')}

// eslint-disable-line + ) : ( + data2?.getAdvertisements + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) > new Date()) + .map( + ( + ad: { + _id: string; + name: string | undefined; + type: string | undefined; + orgId: string; + link: string; + endDate: Date; + startDate: Date; + }, + i: React.Key | null | undefined + ): JSX.Element => ( + + ) + ) + )} +
+ + {data2?.getAdvertisements + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) < new Date()) + .length == 0 ? ( +

{t('pMessage')}

// eslint-disable-line + ) : ( + data2?.getAdvertisements + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) < new Date()) + .map( + ( + ad: { + _id: string; + name: string | undefined; + type: string | undefined; + orgId: string; + link: string; + endDate: Date; + startDate: Date; + }, + i: React.Key | null | undefined + ): JSX.Element => ( + + ) + ) + )} +
+
+
+ +
+
+ + ); +} + +advertisements.defaultProps = {}; + +advertisements.propTypes = {}; diff --git a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.module.css b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.module.css new file mode 100644 index 0000000000..1f1ea89996 --- /dev/null +++ b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.module.css @@ -0,0 +1,20 @@ +.entrytoggle { + margin: 24px 24px 0 auto; + width: fit-content; +} + +.entryaction { + margin-left: auto; + display: flex !important; + align-items: center; +} + +.entryaction i { + margin-right: 8px; +} + +.entryaction .spinner-grow { + height: 1rem; + width: 1rem; + margin-right: 8px; +} diff --git a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx new file mode 100644 index 0000000000..896c61e5e1 --- /dev/null +++ b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx @@ -0,0 +1,60 @@ +import React from 'react'; +import { render } from '@testing-library/react'; + +import { + ApolloClient, + ApolloProvider, + InMemoryCache, + ApolloLink, + HttpLink, +} from '@apollo/client'; + +import type { NormalizedCacheObject } from '@apollo/client'; +import { BrowserRouter } from 'react-router-dom'; +import AdvertisementEntry from './AdvertisementEntry'; +import { Provider } from 'react-redux'; +import { store } from 'state/store'; +import { BACKEND_URL } from 'Constant/constant'; +import i18nForTest from 'utils/i18nForTest'; +import { I18nextProvider } from 'react-i18next'; + +const httpLink = new HttpLink({ + uri: BACKEND_URL, + headers: { + authorization: 'Bearer ' + localStorage.getItem('token') || '', + }, +}); + +const client: ApolloClient = new ApolloClient({ + cache: new InMemoryCache(), + link: ApolloLink.from([httpLink]), +}); +describe('Testing Advertisement Entry Component', () => { + test('Temporary test for Advertisement Entry', () => { + const { getByTestId, getAllByText } = render( + + + + + { + + } + + + + + ); + expect(getByTestId('AdEntry')).toBeInTheDocument(); + expect(getAllByText('POPUP')[0]).toBeInTheDocument(); + expect(getAllByText('Advert1')[0]).toBeInTheDocument(); + }); +}); diff --git a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx new file mode 100644 index 0000000000..36d99004fc --- /dev/null +++ b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx @@ -0,0 +1,103 @@ +import React, { useState } from 'react'; +import PropTypes from 'prop-types'; +import styles from './AdvertisementEntry.module.css'; +import { Button, Card, Col, Row, Spinner } from 'react-bootstrap'; +import { DELETE_ADVERTISEMENT_BY_ID } from 'GraphQl/Mutations/mutations'; +import { useMutation } from '@apollo/client'; +import { useTranslation } from 'react-i18next'; +interface InterfaceAddOnEntryProps { + id: string; + name: string; + link: string; + type: string; + orgId: string; + startDate: Date; + endDate: Date; +} +function advertisementEntry({ + id, + name, + type, + // eslint-disable-next-line @typescript-eslint/no-unused-vars + orgId, + link, + endDate, + // eslint-disable-next-line @typescript-eslint/no-unused-vars + startDate, +}: InterfaceAddOnEntryProps): JSX.Element { + const { t } = useTranslation('translation', { keyPrefix: 'advertisement' }); + const [buttonLoading, setButtonLoading] = useState(false); + const [deleteAdById] = useMutation(DELETE_ADVERTISEMENT_BY_ID); + + const onDelete = async (): Promise => { + setButtonLoading(true); + await deleteAdById({ + variables: { + id: id.toString(), + }, + }); + setButtonLoading(false); + }; + return ( + <> + + {Array.from({ length: 4 }).map((_, idx) => ( + + + + + {name} + Ends on {endDate?.toDateString()} + + {type} + + {link} + + + + + ))} + +
+ + ); +} + +advertisementEntry.propTypes = { + name: PropTypes.string, + type: PropTypes.string, + orgId: PropTypes.string, + link: PropTypes.string, + endDate: PropTypes.instanceOf(Date), + startDate: PropTypes.instanceOf(Date), +}; + +advertisementEntry.defaultProps = { + name: '', + type: '', + orgId: '', + link: '', + endDate: new Date(), + startDate: new Date(), +}; +export default advertisementEntry; diff --git a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.module.css b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.module.css new file mode 100644 index 0000000000..c122d386fa --- /dev/null +++ b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.module.css @@ -0,0 +1,9 @@ +.modalbtn { + display: flex !important; + margin-left: auto; + align-items: center; +} + +.modalbtn i { + margin-right: 8px; +} diff --git a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx new file mode 100644 index 0000000000..e205d50598 --- /dev/null +++ b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx @@ -0,0 +1,207 @@ +import React, { useState } from 'react'; +import PropTypes from 'prop-types'; +import styles from './AdvertisementRegister.module.css'; +import { Button, Form, Modal } from 'react-bootstrap'; +import { useMutation } from '@apollo/client'; +import { ADD_ADVERTISEMENT_MUTATION } from 'GraphQl/Mutations/mutations'; +import { useTranslation } from 'react-i18next'; +import { toast } from 'react-toastify'; +import dayjs from 'dayjs'; + +interface InterfaceAddOnRegisterProps { + id?: string; // OrgId + createdBy?: string; // User +} +interface InterfaceFormStateTypes { + name: string; + link: string; + type: string; + startDate: Date; + endDate: Date; + orgId: string; +} +// eslint-disable-next-line @typescript-eslint/no-unused-vars +function advertisementRegister({ + /* eslint-disable-next-line @typescript-eslint/no-unused-vars */ + createdBy, +}: InterfaceAddOnRegisterProps): JSX.Element { + const { t } = useTranslation('translation', { keyPrefix: 'advertisement' }); + + const [show, setShow] = useState(false); + + const handleClose = (): void => setShow(false); + const handleShow = (): void => setShow(true); + const [create] = useMutation(ADD_ADVERTISEMENT_MUTATION); + + //getting orgId from URL + const currentOrg = window.location.href.split('/id=')[1] + ''; + const [formState, setFormState] = useState({ + name: '', + link: '', + type: 'BANNER', + startDate: new Date(), + endDate: new Date(), + orgId: currentOrg, + }); + const handleRegister = async (): Promise => { + try { + console.log('At handle register', formState); + const { data } = await create({ + variables: { + orgId: currentOrg, + name: formState.name as string, + link: formState.link as string, + type: formState.type as string, + startDate: dayjs(formState.startDate).format('YYYY-MM-DD'), + endDate: dayjs(formState.endDate).format('YYYY-MM-DD'), + }, + }); + + if (data) { + toast.success('Advertisement created successfully'); + setTimeout(() => { + window.location.reload(); + }, 2000); + } + } catch (error) { + console.log('error occured', error); + } + }; + return ( + <> + + + + + {t('RClose')} + + +
+ + {t('Rname')} + { + setFormState({ + ...formState, + name: e.target.value, + }); + }} + /> + + + {t('Rlink')} + { + setFormState({ + ...formState, + link: e.target.value, + }); + }} + /> + + + {t('Rtype')} + { + setFormState({ + ...formState, + type: e.target.value, + }); + console.log(e.target, e.target.value, typeof e.target.value); + }} + > + + + + + + + {t('RstartDate')} + { + setFormState({ + ...formState, + startDate: new Date(e.target.value), + }); + }} + /> + + + + {t('RendDate')} + { + setFormState({ + ...formState, + endDate: new Date(e.target.value), + }); + }} + /> + +
+
+ + + + +
+ + ); +} + +advertisementRegister.defaultProps = { + name: '', + link: '', + type: 'BANNER', + startDate: new Date(), + endDate: new Date(), + orgId: '', +}; + +advertisementRegister.propTypes = { + name: PropTypes.string, + link: PropTypes.string, + type: PropTypes.string, + startDate: PropTypes.instanceOf(Date), + endDate: PropTypes.instanceOf(Date), + orgId: PropTypes.string, +}; + +export default advertisementRegister; diff --git a/src/components/IconComponent/IconComponent.tsx b/src/components/IconComponent/IconComponent.tsx index 8504b13d2a..4c490c9c94 100644 --- a/src/components/IconComponent/IconComponent.tsx +++ b/src/components/IconComponent/IconComponent.tsx @@ -88,6 +88,8 @@ const iconComponent = (props: InterfaceIconComponent): JSX.Element => { stroke={props.fill} /> ); + case 'Advertisement': + return ; default: return ( state.appRoutes); const { targets, configUrl } = appRoutes; - return ( <> + + +
+ + {'Promoted Content'} +
+
+ + {props.title} + {props.title} + {props.image && ( + + )} + +
+ + ); +} diff --git a/src/screens/UserPortal/Home/Home.tsx b/src/screens/UserPortal/Home/Home.tsx index b657fc0c02..ad02e1b26c 100644 --- a/src/screens/UserPortal/Home/Home.tsx +++ b/src/screens/UserPortal/Home/Home.tsx @@ -11,13 +11,17 @@ import getOrganizationId from 'utils/getOrganizationId'; import SendIcon from '@mui/icons-material/Send'; import PostCard from 'components/UserPortal/PostCard/PostCard'; import { useMutation, useQuery } from '@apollo/client'; -import { ORGANIZATION_POST_CONNECTION_LIST } from 'GraphQl/Queries/Queries'; +import { + ADVERTISEMENTS_GET, + ORGANIZATION_POST_CONNECTION_LIST, +} from 'GraphQl/Queries/Queries'; import { CREATE_POST_MUTATION } from 'GraphQl/Mutations/mutations'; import { errorHandler } from 'utils/errorHandler'; import { useTranslation } from 'react-i18next'; import convertToBase64 from 'utils/convertToBase64'; import { toast } from 'react-toastify'; import HourglassBottomIcon from '@mui/icons-material/HourglassBottom'; +import PromotedPost from 'components/UserPortal/PromotedPost/PromotedPost'; interface InterfacePostCardProps { id: string; @@ -60,11 +64,19 @@ export default function home(): JSX.Element { const [posts, setPosts] = React.useState([]); const [postContent, setPostContent] = React.useState(''); const [postImage, setPostImage] = React.useState(''); + const currentOrgId = window.location.href.split('/id=')[1] + ''; + const [adContent, setAdContent] = React.useState([]); const navbarProps = { currentPage: 'home', }; - + const { + data: promotedPostsData, + // eslint-disable-next-line @typescript-eslint/no-unused-vars + refetch: _promotedPostsRefetch, + // eslint-disable-next-line @typescript-eslint/no-unused-vars + loading: promotedPostsLoading, + } = useQuery(ADVERTISEMENTS_GET); const { data, refetch, @@ -116,6 +128,12 @@ export default function home(): JSX.Element { } }, [data]); + React.useEffect(() => { + if (promotedPostsData) { + setAdContent(promotedPostsData.getAdvertisements); + } + }, [data]); + return ( <> @@ -184,6 +202,21 @@ export default function home(): JSX.Element { + {adContent + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) > new Date()).length == 0 + ? '' + : adContent + .filter((ad: any) => ad.orgId == currentOrgId) + .filter((ad: any) => new Date(ad.endDate) > new Date()) + .map((post: any) => ( + + ))} {loadingPosts ? (
Loading... diff --git a/src/state/reducers/routesReducer.test.ts b/src/state/reducers/routesReducer.test.ts index 89743a8508..765920e6f3 100644 --- a/src/state/reducers/routesReducer.test.ts +++ b/src/state/reducers/routesReducer.test.ts @@ -18,6 +18,7 @@ describe('Testing Routes reducer', () => { name: 'Block/Unblock', url: '/blockuser/id=undefined', }, + { name: 'Advertisement', url: '/orgads/id=undefined' }, { name: 'Plugins', subTargets: [ @@ -50,6 +51,11 @@ describe('Testing Routes reducer', () => { }, { name: 'Posts', comp_id: 'orgpost', component: 'OrgPost' }, { name: 'Block/Unblock', comp_id: 'blockuser', component: 'BlockUser' }, + { + name: 'Advertisement', + comp_id: 'orgads', + component: 'Advertisements', + }, { name: 'Plugins', comp_id: null, @@ -83,6 +89,7 @@ describe('Testing Routes reducer', () => { { name: 'Events', url: '/orgevents/id=undefined' }, { name: 'Posts', url: '/orgpost/id=undefined' }, { name: 'Block/Unblock', url: '/blockuser/id=undefined' }, + { name: 'Advertisement', url: '/orgads/id=undefined' }, { name: 'Plugins', subTargets: [ @@ -116,6 +123,11 @@ describe('Testing Routes reducer', () => { }, { name: 'Posts', comp_id: 'orgpost', component: 'OrgPost' }, { name: 'Block/Unblock', comp_id: 'blockuser', component: 'BlockUser' }, + { + name: 'Advertisement', + comp_id: 'orgads', + component: 'Advertisements', + }, { name: 'Plugins', comp_id: null, @@ -152,6 +164,7 @@ describe('Testing Routes reducer', () => { name: 'Block/Unblock', url: '/blockuser/id=undefined', }, + { name: 'Advertisement', url: '/orgads/id=undefined' }, { name: 'Settings', url: '/orgsetting/id=undefined' }, { name: 'All Organizations', url: '/orglist/id=undefined' }, { @@ -185,8 +198,14 @@ describe('Testing Routes reducer', () => { comp_id: 'orgevents', component: 'OrganizationEvents', }, + { name: 'Posts', comp_id: 'orgpost', component: 'OrgPost' }, { name: 'Block/Unblock', comp_id: 'blockuser', component: 'BlockUser' }, + { + name: 'Advertisement', + comp_id: 'orgads', + component: 'Advertisements', + }, { name: 'Plugins', comp_id: null, diff --git a/src/state/reducers/routesReducer.ts b/src/state/reducers/routesReducer.ts index d0d5a7ab26..4fafbd1ebb 100644 --- a/src/state/reducers/routesReducer.ts +++ b/src/state/reducers/routesReducer.ts @@ -68,6 +68,7 @@ const components: ComponentType[] = [ { name: 'Events', comp_id: 'orgevents', component: 'OrganizationEvents' }, { name: 'Posts', comp_id: 'orgpost', component: 'OrgPost' }, { name: 'Block/Unblock', comp_id: 'blockuser', component: 'BlockUser' }, + { name: 'Advertisement', comp_id: 'orgads', component: 'Advertisements' }, { name: 'Plugins', comp_id: null, @@ -81,6 +82,7 @@ const components: ComponentType[] = [ }, ], }, + { name: 'Settings', comp_id: 'orgsetting', component: 'OrgSettings' }, { name: 'All Organizations', comp_id: 'orglist', component: 'OrgList' }, { name: '', comp_id: 'member', component: 'MemberDetail' }, From 29b87a9c3e9a611c811eda82478222cd00b6e784 Mon Sep 17 00:00:00 2001 From: Akhilender Bongirwar <112749383+akhilender-bongirwar@users.noreply.github.com> Date: Fri, 24 Nov 2023 18:11:55 +0530 Subject: [PATCH 4/4] feat: Implemented Sorting Functionality for Users Screen (#1081) * feat: Implemented Sorting Functionality for Users Screen Changes Made: - Implemented sorting functionality for the users screen. - Added options for sorting users by latest first and oldest first. - Wrote corresponding tests to ensure the sorting behavior is accurate. - Tested the sorting feature by logging into the Talawa admin dashboard, navigating to the users option, and checking the sort button. Signed-off-by: Akhilender * fix: altering the words - Made Latest to Newest Signed-off-by: Akhilender --------- Signed-off-by: Akhilender --- public/locales/en.json | 2 + public/locales/fr.json | 2 + public/locales/hi.json | 2 + public/locales/sp.json | 2 + public/locales/zh.json | 2 + src/screens/Users/Users.test.tsx | 38 +++++++++++++++++- src/screens/Users/Users.tsx | 68 +++++++++++++++++++++++++++----- 7 files changed, 104 insertions(+), 12 deletions(-) diff --git a/public/locales/en.json b/public/locales/en.json index 50945e8dde..bfefeec253 100644 --- a/public/locales/en.json +++ b/public/locales/en.json @@ -141,6 +141,8 @@ "loadingUsers": "Loading Users...", "noUserFound": "No User Found", "sort": "Sort", + "Newest": "Newest First", + "Oldest": "Oldest First", "filter": "Filter", "noOrgError": "Organizations not found, please create an organization through dashboard", "roleUpdated": "Role Updated.", diff --git a/public/locales/fr.json b/public/locales/fr.json index 75522a0fed..818ffd6fcd 100644 --- a/public/locales/fr.json +++ b/public/locales/fr.json @@ -132,6 +132,8 @@ "loadingUsers": "Chargement des utilisateurs...", "noUserFound": "Aucun utilisateur trouvé", "sort": "Trier", + "Oldest": "Les plus anciennes d'abord", + "Newest": "Les plus récentes d'abord", "filter": "Filtre", "roleUpdated": "Rôle mis à jour.", "noResultsFoundFor": "Aucun résultat trouvé pour ", diff --git a/public/locales/hi.json b/public/locales/hi.json index d952ff7f30..fb9ae1ab23 100644 --- a/public/locales/hi.json +++ b/public/locales/hi.json @@ -131,6 +131,8 @@ "loadingUsers": "उपयोगकर्ता लोड हो रहा है ...", "noUserFound": "कोई उपयोगकर्ता नहीं मिला।", "sort": "छांटें", + "Oldest": "सबसे पुराना पहले", + "Newest": "सबसे नवीनतम पहले", "filter": "फ़िल्टर", "roleUpdated": "भूमिका अपडेट की गई।", "noResultsFoundFor": "के लिए कोई परिणाम नहीं मिला ", diff --git a/public/locales/sp.json b/public/locales/sp.json index 9b107afd44..09d6472753 100644 --- a/public/locales/sp.json +++ b/public/locales/sp.json @@ -131,6 +131,8 @@ "loadingUsers": "Cargando usuarios ...", "noUserFound": "No se encontró ningún usuario.", "sort": "Ordenar", + "Oldest": "Más Antiguas Primero", + "Newest": "Más Recientes Primero", "filter": "Filtrar", "roleUpdated": "Rol actualizado.", "noResultsFoundFor": "No se encontraron resultados para ", diff --git a/public/locales/zh.json b/public/locales/zh.json index 92e8c14c83..138a345206 100644 --- a/public/locales/zh.json +++ b/public/locales/zh.json @@ -131,6 +131,8 @@ "loadingUsers": "正在加載用戶...", "noUserFound": "找不到用戶。", "sort": "排序", + "Oldest": "最旧的优先", + "Newest": "最新的优先", "filter": "過濾", "roleUpdated": "角色已更新。", "noResultsFoundFor": "未找到结果 ", diff --git a/src/screens/Users/Users.test.tsx b/src/screens/Users/Users.test.tsx index 03a65c49fd..4484518529 100644 --- a/src/screens/Users/Users.test.tsx +++ b/src/screens/Users/Users.test.tsx @@ -1,13 +1,12 @@ import React from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { act, fireEvent, render, screen } from '@testing-library/react'; import 'jest-localstorage-mock'; import 'jest-location-mock'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import { ToastContainer } from 'react-toastify'; - import userEvent from '@testing-library/user-event'; import { store } from 'state/store'; import { StaticMockLink } from 'utils/StaticMockLink'; @@ -179,4 +178,39 @@ describe('Testing Users screen', () => { 'Organizations not found, please create an organization through dashboard' ); }); + + test('Testing sort Newest and oldest toggle', async () => { + await act(async () => { + render( + + + + + + + + + + + ); + + await wait(); + + const searchInput = screen.getByTestId('sort'); + expect(searchInput).toBeInTheDocument(); + + const inputText = screen.getByTestId('sortUsers'); + + fireEvent.click(inputText); + const toggleText = screen.getByTestId('newest'); + + fireEvent.click(toggleText); + + expect(searchInput).toBeInTheDocument(); + fireEvent.click(inputText); + const toggleTite = screen.getByTestId('oldest'); + fireEvent.click(toggleTite); + expect(searchInput).toBeInTheDocument(); + }); + }); }); diff --git a/src/screens/Users/Users.tsx b/src/screens/Users/Users.tsx index e7b1e0c1af..64a9be1735 100644 --- a/src/screens/Users/Users.tsx +++ b/src/screens/Users/Users.tsx @@ -31,6 +31,7 @@ const Users = (): JSX.Element => { const [hasMore, setHasMore] = useState(true); const [isLoadingMore, setIsLoadingMore] = useState(false); const [searchByName, setSearchByName] = useState(''); + const [sortingOption, setSortingOption] = useState('newest'); const userType = localStorage.getItem('UserType'); const loggedInUserId = localStorage.getItem('id'); @@ -57,6 +58,7 @@ const Users = (): JSX.Element => { }); const { data: dataOrgs } = useQuery(ORGANIZATION_CONNECTION_LIST); + const [displayedUsers, setDisplayedUsers] = useState(usersData?.users || []); // Manage loading more state useEffect(() => { @@ -66,7 +68,11 @@ const Users = (): JSX.Element => { if (usersData.users.length < perPageResult) { setHasMore(false); } - }, [usersData]); + if (usersData && usersData.users) { + const newDisplayedUsers = sortUsers(usersData.users, sortingOption); + setDisplayedUsers(newDisplayedUsers); + } + }, [usersData, sortingOption]); // To clear the search when the component is unmounted useEffect(() => { @@ -155,6 +161,32 @@ const Users = (): JSX.Element => { }); }; const debouncedHandleSearchByName = debounce(handleSearchByName); + // console.log(usersData); + + const handleSorting = (option: string): void => { + setSortingOption(option); + }; + + const sortUsers = ( + allUsers: InterfaceQueryUserListItem[], + sortingOption: string + ): InterfaceQueryUserListItem[] => { + const sortedUsers = [...allUsers]; + + if (sortingOption === 'newest') { + sortedUsers.sort( + (a, b) => + new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime() + ); + } else if (sortingOption === 'oldest') { + sortedUsers.sort( + (a, b) => + new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime() + ); + } + + return sortedUsers; + }; const headerTitles: string[] = [ '#', @@ -196,15 +228,31 @@ const Users = (): JSX.Element => {
-
{isLoading == false && usersData && - usersData.users.length === 0 && + displayedUsers.length === 0 && searchByName.length > 0 ? (

{t('noResultsFoundFor')} "{searchByName}"

- ) : isLoading == false && usersData && usersData.users.length === 0 ? ( + ) : isLoading == false && usersData && displayedUsers.length === 0 ? ( // eslint-disable-next-line react/jsx-indent

{t('noUserFound')}

@@ -244,7 +292,7 @@ const Users = (): JSX.Element => { /> ) : ( { {usersData && - usersData?.users.map((user, index) => { + displayedUsers.map((user, index) => { return (