From 8bf41d5a3809ac8a42e808af874b10d0f28fcdde Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 27 Dec 2023 16:40:57 +0000 Subject: [PATCH 01/12] chore(deps): bump word-wrap from 1.2.3 to 1.2.5 Bumps [word-wrap](https://github.com/jonschlinkert/word-wrap) from 1.2.3 to 1.2.5. - [Release notes](https://github.com/jonschlinkert/word-wrap/releases) - [Commits](https://github.com/jonschlinkert/word-wrap/compare/1.2.3...1.2.5) --- updated-dependencies: - dependency-name: word-wrap dependency-type: indirect ... Signed-off-by: dependabot[bot] --- yarn.lock | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/yarn.lock b/yarn.lock index be54e7c6..3412bd70 100644 --- a/yarn.lock +++ b/yarn.lock @@ -14574,9 +14574,9 @@ wide-align@^1.1.5: string-width "^1.0.2 || 2 || 3 || 4" word-wrap@~1.2.3: - version "1.2.3" - resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.3.tgz#610636f6b1f703891bd34771ccb17fb93b47079c" - integrity sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ== + version "1.2.5" + resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34" + integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA== wordwrap@^1.0.0: version "1.0.0" From c0e5aa7e6a003dc4791a28ce69a2b8b76c69fe74 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 27 Dec 2023 16:57:35 +0000 Subject: [PATCH 02/12] chore(deps): bump decode-uri-component from 0.2.0 to 0.2.2 Bumps [decode-uri-component](https://github.com/SamVerschueren/decode-uri-component) from 0.2.0 to 0.2.2. - [Release notes](https://github.com/SamVerschueren/decode-uri-component/releases) - [Commits](https://github.com/SamVerschueren/decode-uri-component/compare/v0.2.0...v0.2.2) --- updated-dependencies: - dependency-name: decode-uri-component dependency-type: indirect ... Signed-off-by: dependabot[bot] --- yarn.lock | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/yarn.lock b/yarn.lock index 2e467346..e722ec43 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6755,9 +6755,9 @@ decimal.js@^10.3.1: integrity sha512-F29o+vci4DodHYT9UrR5IEbfBw9pE5eSapIJdTqXK5+6hq+t8VRxwQyKlW2i+KDKFkkJQRvFyI/QXD83h8LyQw== decode-uri-component@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz#eb3913333458775cb84cd1a1fae062106bb87545" - integrity sha512-hjf+xovcEn31w/EUYdTXQh/8smFL/dzYjohQGEIgjyNavaJfBY2p5F527Bo1VPATxv0VYTUC2bOcXvqFwk78Og== + version "0.2.2" + resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.2.tgz#e69dbe25d37941171dd540e024c444cd5188e1e9" + integrity sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ== deep-is@^0.1.3, deep-is@~0.1.3: version "0.1.4" From 1f1169bf3e9bc7ec5dd7307ed55ecb53e6b6747f Mon Sep 17 00:00:00 2001 From: Satyam Bajpai Date: Sat, 6 Jan 2024 23:08:27 +0530 Subject: [PATCH 03/12] feat: add query params restriction to go to another steps (#816) * feat: add query params restriction to go to another steps * refactor: change double quote to single quote --- app/components/stepper-signup.js | 20 +++++++++++++++----- 1 file changed, 15 insertions(+), 5 deletions(-) diff --git a/app/components/stepper-signup.js b/app/components/stepper-signup.js index 6aef71d1..fadd8102 100644 --- a/app/components/stepper-signup.js +++ b/app/components/stepper-signup.js @@ -8,26 +8,35 @@ const MIN_STEP = 0; export default class StepperSignupComponent extends Component { @service login; @service router; + @service fastboot; + @service featureFlag; @tracked preValid = false; @tracked isValid = JSON.parse(localStorage.getItem('isValid')) ?? false; - @tracked currentStep = - Number(localStorage.getItem('currentStep')) ?? Number(this.args.step) ?? 0; + @tracked currentStep = Number(localStorage.getItem('currentStep')) ?? 0; @tracked stepOneData = JSON.parse(localStorage.getItem('stepOneData')); @tracked stepTwoData = JSON.parse(localStorage.getItem('stepTwoData')); @tracked stepThreeData = JSON.parse(localStorage.getItem('stepThreeData')); setIsValid = (newVal) => (this.isValid = newVal); setIsPreValid = (newVal) => (this.preValid = newVal); + constructor() { super(...arguments); - this.currentStep = Number( - new URLSearchParams(window.location.search).get('step'), - ); + if (!this.fastboot.isFastBoot && this.featureFlag.isDevMode) { + if ( + localStorage.getItem('currentStep') !== + new URLSearchParams(window.location.search).get('step') + ) { + const queryParams = { dev: true, step: this.currentStep }; + this.router.transitionTo('join', { queryParams }); + } + } } @action decrementStep() { if (this.currentStep > MIN_STEP) { this.currentStep -= 1; + localStorage.setItem('currentStep', this.currentStep); const queryParams = { dev: true, step: this.currentStep }; this.router.transitionTo('join', { queryParams }); } @@ -36,6 +45,7 @@ export default class StepperSignupComponent extends Component { @action incrementStep() { if (this.currentStep < MAX_STEP) { this.currentStep += 1; + localStorage.setItem('currentStep', this.currentStep); const queryParams = { dev: true, step: this.currentStep }; this.router.transitionTo('join', { queryParams }); } From cf65e470d4e9bddcc87fabc5f1a1963a1246363b Mon Sep 17 00:00:00 2001 From: Vinit khandal <111434418+vinit717@users.noreply.github.com> Date: Sun, 7 Jan 2024 02:59:50 +0530 Subject: [PATCH 04/12] Integrate applications api in onboarding (#817) * integrate applications api * move application in service and refactor test * refactor onboarding service for application api --------- Co-authored-by: satyam73 --- app/components/join-steps/step-four.js | 35 +++++---------- app/components/signup-steps/step-one.js | 34 --------------- app/components/stepper-signup.hbs | 2 +- app/components/stepper-signup.js | 32 ++++++++++++++ app/services/onboarding.js | 16 +++++++ .../integration/components/step-four-test.js | 43 ++++++++++--------- 6 files changed, 81 insertions(+), 81 deletions(-) diff --git a/app/components/join-steps/step-four.js b/app/components/join-steps/step-four.js index 2cf20db4..4218b1ca 100644 --- a/app/components/join-steps/step-four.js +++ b/app/components/join-steps/step-four.js @@ -9,78 +9,66 @@ export default class StepFourComponent extends Component { JOIN_DATA = [ { id: 'one', - key: 'first_name', - label: 'First Name', - data: '', - }, - { - id: 'two', - key: 'last_name', - label: 'Last Name', - data: '', - }, - { - id: 'three', key: 'city', label: 'Your City', data: '', }, { - id: 'four', + id: 'two', key: 'state', label: 'Your State', data: '', }, { - id: 'five', + id: 'three', key: 'country', label: 'Your Country', data: '', }, { - id: 'six', + id: 'four', key: 'introduction', label: 'Your Introduction', data: '', }, { - id: 'seven', + id: 'five', key: 'skills', label: 'Your Skills', data: '', }, { - id: 'eight', + id: 'six', key: 'college', label: 'Your Institution', data: '', }, { - id: 'nine', + id: 'seven', key: 'forFun', label: 'What do you do for fun?', data: '', }, { - id: 'ten', + id: 'eight', key: 'funFact', label: 'Fun facts about you', data: '', }, { - id: 'eleven', + id: 'nine', key: 'whyRds', label: 'Why do you want to join Real Dev Squad?', data: '', }, { - id: 'twelve', + id: 'ten', key: 'numberOfHours', label: 'How many hours per week, are you willing to contribute?', data: '', }, { - id: 'thirteen', + id: 'eleven', key: 'foundFrom', label: 'How did you hear about us?', data: '', @@ -99,8 +87,5 @@ export default class StepFourComponent extends Component { const key = data.key; data.data = this.allStepsData[key]; }); - - this.JOIN_DATA[0].data = localStorage.getItem('first_name'); - this.JOIN_DATA[1].data = localStorage.getItem('last_name'); } } diff --git a/app/components/signup-steps/step-one.js b/app/components/signup-steps/step-one.js index a072cc48..73ce9c91 100644 --- a/app/components/signup-steps/step-one.js +++ b/app/components/signup-steps/step-one.js @@ -5,8 +5,6 @@ import { debounce } from '@ember/runloop'; import { inject as service } from '@ember/service'; import { ROLE } from '../../constants/stepper-signup-data'; import { JOIN_DEBOUNCE_TIME } from '../../constants/join'; -import { APPS } from '../../constants/urls'; -import { toastNotificationTimeoutOptions } from '../../constants/toast-notification'; export default class SignupStepsStepOneComponent extends Component { @service toast; @service onboarding; @@ -81,38 +79,6 @@ export default class SignupStepsStepOneComponent extends Component { debounce(this.data, passVal, JOIN_DEBOUNCE_TIME); } - @action async getUsername() { - try { - const firstname = this.data.firstname.toLowerCase(); - const lastname = this.data.lastname.toLowerCase(); - const response = await fetch( - `${APPS.API_BACKEND}/users/username?firstname=${firstname}&lastname=${lastname}&dev=true`, - { - method: 'GET', - headers: { - 'Content-Type': 'application/json', - }, - credentials: 'include', - }, - ); - const data = await response.json(); - if (response.status === 200) { - this.data = { - ...this.data, - username: data.username, - }; - } else if (response.status === 401) { - this.toast.error( - 'Please login to continue.', - '', - toastNotificationTimeoutOptions, - ); - } - } catch (err) { - console.log('Error: ', 'Something went wrong'); - } - } - @action async signup() { const { username } = await this.onboarding.generateUsername( this.data.firstname, diff --git a/app/components/stepper-signup.hbs b/app/components/stepper-signup.hbs index 195e9d8b..6536b57b 100644 --- a/app/components/stepper-signup.hbs +++ b/app/components/stepper-signup.hbs @@ -90,7 +90,7 @@ diff --git a/app/components/stepper-signup.js b/app/components/stepper-signup.js index fadd8102..f903bd83 100644 --- a/app/components/stepper-signup.js +++ b/app/components/stepper-signup.js @@ -2,6 +2,7 @@ import Component from '@glimmer/component'; import { tracked } from '@glimmer/tracking'; import { action } from '@ember/object'; import { inject as service } from '@ember/service'; +import { TOAST_OPTIONS } from '../constants/toast-options'; const MAX_STEP = 15; const MIN_STEP = 0; @@ -10,6 +11,8 @@ export default class StepperSignupComponent extends Component { @service router; @service fastboot; @service featureFlag; + @service toast; + @service onboarding; @tracked preValid = false; @tracked isValid = JSON.parse(localStorage.getItem('isValid')) ?? false; @tracked currentStep = Number(localStorage.getItem('currentStep')) ?? 0; @@ -73,4 +76,33 @@ export default class StepperSignupComponent extends Component { const queryParams = { dev: true, step: this.currentStep }; this.router.transitionTo('join', { queryParams }); } + + @action async applicationHandler() { + const firstName = this.login.userData.first_name; + const lastName = this.login.userData.last_name; + const data = JSON.stringify({ + firstName, + lastName, + ...this.stepOneData, + ...this.stepTwoData, + ...this.stepThreeData, + }); + + const response = await this.onboarding.addApplication(data); + + if (response.status === 201) { + this.toast.success( + 'Successfully submitted the form', + 'Success!', + TOAST_OPTIONS, + ); + this.incrementStep(); + } else if (response.status === 409) { + this.toast.error( + 'You have already filled the form', + 'User Exist!', + TOAST_OPTIONS, + ); + } + } } diff --git a/app/services/onboarding.js b/app/services/onboarding.js index 7b025522..b8f5acea 100644 --- a/app/services/onboarding.js +++ b/app/services/onboarding.js @@ -2,6 +2,7 @@ import Service from '@ember/service'; import { inject as service } from '@ember/service'; import { TOAST_OPTIONS } from '../constants/toast-options'; import { ERROR_MESSAGES } from '../constants/error-messages'; +import { APPS } from '../constants/urls'; export default class OnboardingService extends Service { @service store; @@ -57,4 +58,19 @@ export default class OnboardingService extends Service { ); } } + + async addApplication(data) { + try { + const response = await fetch(`${APPS.API_BACKEND}/applications`, { + method: 'POST', + headers: { 'Content-Type': 'application/json' }, + credentials: 'include', + body: data, + }); + return response; + } catch (err) { + console.log('Error: ', err); + this.toast.error('Some error occured', 'Error ocurred!', TOAST_OPTIONS); + } + } } diff --git a/tests/integration/components/step-four-test.js b/tests/integration/components/step-four-test.js index 25982c02..aba4e287 100644 --- a/tests/integration/components/step-four-test.js +++ b/tests/integration/components/step-four-test.js @@ -7,73 +7,74 @@ module('Integration | Component | step-four', function (hooks) { setupRenderingTest(hooks); test('step four renders', async function (assert) { - assert.expect(27); + assert.expect(23); await render(hbs``); const joinData = [ { id: 'one', - label: 'First Name', - data: '', - }, - { - id: 'two', - label: 'Last Name', - data: '', - }, - { - id: 'three', + key: 'city', label: 'Your City', data: '', }, { - id: 'four', + id: 'two', + key: 'state', label: 'Your State', data: '', }, { - id: 'five', + id: 'three', + key: 'country', label: 'Your Country', data: '', }, { - id: 'six', + id: 'four', + key: 'introduction', label: 'Your Introduction', data: '', }, { - id: 'seven', + id: 'five', + key: 'skills', label: 'Your Skills', data: '', }, { - id: 'eight', + id: 'six', + key: 'college', label: 'Your Institution', data: '', }, { - id: 'nine', + id: 'seven', + key: 'forFun', label: 'What do you do for fun?', data: '', }, { - id: 'ten', + id: 'eight', + key: 'funFact', label: 'Fun facts about you', data: '', }, { - id: 'eleven', + id: 'nine', + key: 'whyRds', label: 'Why do you want to join Real Dev Squad?', data: '', }, { - id: 'twelve', + id: 'ten', + key: 'numberOfHours', label: 'How many hours per week, are you willing to contribute?', data: '', }, { - id: 'thirteen', + id: 'eleven', + key: 'foundFrom', label: 'How did you hear about us?', data: '', }, From c2aa85d0d4908a3998b3458b881c7f4e043a0b95 Mon Sep 17 00:00:00 2001 From: Prakash Choudhary <34452139+prakashchoudhary07@users.noreply.github.com> Date: Sun, 7 Jan 2024 04:54:20 +0530 Subject: [PATCH 05/12] Update firstname, lastname, usrname on username generation (#829) * feat: update firstname, lastname, usrname on username generation * refactor: remove unused login service --- app/components/signup-steps/step-one.js | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/app/components/signup-steps/step-one.js b/app/components/signup-steps/step-one.js index 73ce9c91..af497b99 100644 --- a/app/components/signup-steps/step-one.js +++ b/app/components/signup-steps/step-one.js @@ -8,6 +8,7 @@ import { JOIN_DEBOUNCE_TIME } from '../../constants/join'; export default class SignupStepsStepOneComponent extends Component { @service toast; @service onboarding; + @service login; @tracked data = { firstname: '', lastname: '', username: '', role: '' }; @tracked isSignupButtonDisabled = true; @tracked isValid = true; @@ -100,6 +101,14 @@ export default class SignupStepsStepOneComponent extends Component { } await this.onboarding.signup(dataToUpdate); + // Update user records firstname and lastname + const user = this.login.userData; + if (!user.first_name || !user.last_name) { + this.login.userData.first_name = this.data.firstname; + this.login.userData.last_name = this.data.lastname; + this.login.userData.username = username; + } + // To get user details after signup localStorage.setItem('role', this.data.role); this.args.incrementStep(); } From a4a3b753fd23b15f5516ae039c0b17c6d6af0c53 Mon Sep 17 00:00:00 2001 From: Vinit khandal <111434418+vinit717@users.noreply.github.com> Date: Sun, 7 Jan 2024 12:44:09 +0530 Subject: [PATCH 06/12] Integrate discord API (#814) * integrate discord api * integrate discord api * feat: integrate discord invite api * fix: fix the product_manager key * feat: redirect to discord invite link * write test for discord invite api --------- Co-authored-by: Prakash --- app/components/signup-steps/step-one.js | 2 +- app/components/signup-steps/step-two.hbs | 41 +++++++++++----------- app/components/signup-steps/step-two.js | 11 ++++++ app/services/onboarding.js | 29 ++++++++++++++-- tests/unit/services/onboarding-test.js | 43 ++++++++++++++++++++++++ 5 files changed, 102 insertions(+), 24 deletions(-) diff --git a/app/components/signup-steps/step-one.js b/app/components/signup-steps/step-one.js index af497b99..67504cf4 100644 --- a/app/components/signup-steps/step-one.js +++ b/app/components/signup-steps/step-one.js @@ -96,7 +96,7 @@ export default class SignupStepsStepOneComponent extends Component { dataToUpdate.roles = { maven: this.data.role === 'Maven', designer: this.data.role === 'Designer', - productmanager: this.data.role === 'Product Manager', + product_manager: this.data.role === 'Product Manager', }; } diff --git a/app/components/signup-steps/step-two.hbs b/app/components/signup-steps/step-two.hbs index 03ad6840..1b7d4f51 100644 --- a/app/components/signup-steps/step-two.hbs +++ b/app/components/signup-steps/step-two.hbs @@ -1,21 +1,20 @@ - -

Congratulations

-

- {{#if (eq this.role 'Developer')}} - Let's get your journey started with Real Dev Squad - {{else}} - You can now join our Discord server where we do our discussions, learning - and mentoring stuff - {{/if}} -

- +

Congratulations

+

+ {{#if (eq this.role 'Developer')}} + Let's get your journey started with Real Dev Squad + {{else}} + You can now join our Discord server where we do our discussions, learning + and mentoring stuff + {{/if}} +

+ diff --git a/app/components/signup-steps/step-two.js b/app/components/signup-steps/step-two.js index 000da006..9e15d6fd 100644 --- a/app/components/signup-steps/step-two.js +++ b/app/components/signup-steps/step-two.js @@ -1,6 +1,17 @@ import Component from '@glimmer/component'; import { tracked } from '@glimmer/tracking'; +import { inject as service } from '@ember/service'; +import { action } from '@ember/object'; export default class SignupStepsStepTwoComponent extends Component { + @service onboarding; @tracked role = localStorage.getItem('role'); + + @action + async generateDiscordLink() { + const inviteLink = await this.onboarding.discordInvite(); + if (inviteLink) { + window.open(`https://${inviteLink}`, '_blank'); + } + } } diff --git a/app/services/onboarding.js b/app/services/onboarding.js index b8f5acea..1c2ad2d6 100644 --- a/app/services/onboarding.js +++ b/app/services/onboarding.js @@ -1,7 +1,7 @@ -import Service from '@ember/service'; -import { inject as service } from '@ember/service'; +import Service, { service } from '@ember/service'; import { TOAST_OPTIONS } from '../constants/toast-options'; import { ERROR_MESSAGES } from '../constants/error-messages'; +import { GET_API_CONFIGS, POST_API_CONFIGS } from '../constants/live'; import { APPS } from '../constants/urls'; export default class OnboardingService extends Service { @@ -73,4 +73,29 @@ export default class OnboardingService extends Service { this.toast.error('Some error occured', 'Error ocurred!', TOAST_OPTIONS); } } + + async discordInvite() { + const discordInviteUrl = `${APPS.API_BACKEND}/discord-actions/invite`; + try { + let response = await fetch(discordInviteUrl, POST_API_CONFIGS); + + if (response.status === 409) { + response = await fetch(discordInviteUrl, { + ...GET_API_CONFIGS, + credentials: 'include', + }); + } + + const { inviteLink, message } = await response.json(); + + if (response.status === 403) { + this.toast.error(message, 'error!', TOAST_OPTIONS); + } + + return inviteLink; + } catch (error) { + console.error(error); + this.toast.error('Something went wrong!', 'error!', TOAST_OPTIONS); + } + } } diff --git a/tests/unit/services/onboarding-test.js b/tests/unit/services/onboarding-test.js index 5656b3c5..2dbefd04 100644 --- a/tests/unit/services/onboarding-test.js +++ b/tests/unit/services/onboarding-test.js @@ -103,4 +103,47 @@ module('Unit | Service | onboarding', function (hooks) { assert.verifySteps(['store.createRecord called']); }); + + test('discordInvite method', async function (assert) { + let service = this.owner.lookup('service:onboarding'); + let fetch = window.fetch; + let toast = this.owner.lookup('service:toast'); + + toast.error = (message) => assert.step(`toast.error: ${message}`); + + window.fetch = (url, configs) => { + assert.step(`fetch called with url: ${url}`); + + if (configs.method === 'POST') { + return Promise.resolve({ + status: 409, + json: () => Promise.resolve({}), + }); + } else { + return Promise.resolve({ + status: 403, + json: () => Promise.resolve({ message: 'Forbidden' }), + }); + } + }; + + const inviteLink = await service.discordInvite(); + + assert.verifySteps( + [ + `fetch called with url: https://staging-api.realdevsquad.com/discord-actions/invite`, + 'fetch called with url: https://staging-api.realdevsquad.com/discord-actions/invite', + 'toast.error: Forbidden', + ], + 'Correct methods were called in the service', + ); + + assert.strictEqual( + inviteLink, + undefined, + 'Invite link should be undefined due to error status', + ); + + window.fetch = fetch; + }); }); From f4df75049dc35ccc564eaf90f18d239d749366aa Mon Sep 17 00:00:00 2001 From: Satyam Bajpai Date: Mon, 8 Jan 2024 09:21:46 +0530 Subject: [PATCH 07/12] Feat/status card (#830) * refactor condition to show signup form * feat: initial draft pr * chore: remove unnecessary code * integrate discord api in card status * write test for status card * fix failing test --------- Co-authored-by: vinit717 --- app/components/join-steps/status-card.hbs | 70 +++++++++++++++ app/components/join-steps/status-card.js | 28 ++++++ app/components/stepper-signup.hbs | 62 ++++++++----- app/components/stepper-signup.js | 15 ++++ app/constants/join.js | 6 ++ app/services/onboarding.js | 30 ++++++- app/styles/app.css | 1 + app/styles/status-card.module.css | 15 ++++ .../components/status-card-test.js | 86 +++++++++++++++++++ 9 files changed, 290 insertions(+), 23 deletions(-) create mode 100644 app/components/join-steps/status-card.hbs create mode 100644 app/components/join-steps/status-card.js create mode 100644 app/styles/status-card.module.css create mode 100644 tests/integration/components/status-card-test.js diff --git a/app/components/join-steps/status-card.hbs b/app/components/join-steps/status-card.hbs new file mode 100644 index 00000000..665ec4cf --- /dev/null +++ b/app/components/join-steps/status-card.hbs @@ -0,0 +1,70 @@ +
+
+ {{#each this.APPLICATION_STATUSES as |statusDetails|}} + {{#if (eq @status statusDetails.status)}} +

+ {{statusDetails.heading}} +

+ + {{/if}} + {{/each}} +
+
+ {{#if (eq @status this.APPLICATION_STATUS_TYPES.pending)}} +

+ Your application is currently in pending state, please regularly check + this page for invite link. +

+ {{else if (eq @status this.APPLICATION_STATUS_TYPES.rejected)}} +

+ Your application is rejected +

+

+ Here's the feedback for your application +

+

+ {{@feedback}} +

+ {{else if (eq @status this.APPLICATION_STATUS_TYPES.accepted)}} +

+ Congratulations! + Your application is accepted by us +

+

+ Here's the feedback for your application +

+

+ {{@feedback}} +

+

+ Take the Next Step and Join Our Discord Server. +

+ {{/if}} +
+
+ +
+
\ No newline at end of file diff --git a/app/components/join-steps/status-card.js b/app/components/join-steps/status-card.js new file mode 100644 index 00000000..18bc3b11 --- /dev/null +++ b/app/components/join-steps/status-card.js @@ -0,0 +1,28 @@ +import Component from '@glimmer/component'; +import { inject as service } from '@ember/service'; +import { APPLICATION_STATUS_TYPES } from '../../constants/join'; + +export default class StatusCardComponent extends Component { + @service login; + @service router; + + APPLICATION_STATUS_TYPES = APPLICATION_STATUS_TYPES; + + APPLICATION_STATUSES = [ + { + status: APPLICATION_STATUS_TYPES.pending, + heading: 'Pending', + icon: 'mdi:timer-sand', + }, + { + status: APPLICATION_STATUS_TYPES.rejected, + heading: 'Rejected', + icon: 'mdi:close-circle', + }, + { + status: APPLICATION_STATUS_TYPES.accepted, + heading: 'Accepted', + icon: 'mdi:check-circle', + }, + ]; +} diff --git a/app/components/stepper-signup.hbs b/app/components/stepper-signup.hbs index 6536b57b..82b97038 100644 --- a/app/components/stepper-signup.hbs +++ b/app/components/stepper-signup.hbs @@ -1,11 +1,11 @@
-
+
{{#if (not-eq this.currentStep 0)}} {{/if}} @@ -14,11 +14,23 @@ @currentStep={{this.currentStep}} @startHandler={{this.startHandler}} /> - {{else if (eq this.currentStep 1)}} + {{else if + (and (eq this.currentStep 1) this.login.userData.incompleteUserDetails) + }} + {{else if (eq this.currentStep 1)}} +

You have already completed the sign-up process. + Please proceed to the following steps.

+ {{else if (eq this.currentStep 2)}} {{else if (eq this.currentStep 3)}} @@ -73,42 +85,48 @@ @goToGenerateChaincodePage={{this.goToGenerateChaincodePage}} @startHandler={{this.startHandler}} /> + {{else if (eq this.currentStep 14)}} + {{/if}}
{{#if (and (gte this.currentStep 3) (lte this.currentStep 6))}} -
+
{{#if (eq this.currentStep 6)}} {{else}} {{/if}}
diff --git a/app/components/stepper-signup.js b/app/components/stepper-signup.js index f903bd83..6a5db272 100644 --- a/app/components/stepper-signup.js +++ b/app/components/stepper-signup.js @@ -36,6 +36,14 @@ export default class StepperSignupComponent extends Component { } } + get applicationStatus() { + return this.onboarding.applicationData?.status; + } + + get applicationFeedback() { + return this.onboarding.applicationData?.feedback; + } + @action decrementStep() { if (this.currentStep > MIN_STEP) { this.currentStep -= 1; @@ -105,4 +113,11 @@ export default class StepperSignupComponent extends Component { ); } } + + @action async joinDiscordHandler() { + const inviteLink = await this.onboarding.discordInvite(); + if (inviteLink) { + window.open(`https://${inviteLink}`, '_blank'); + } + } } diff --git a/app/constants/join.js b/app/constants/join.js index 91e1b024..e573c74d 100644 --- a/app/constants/join.js +++ b/app/constants/join.js @@ -20,3 +20,9 @@ export const STEP_THREE_LIMITS = { numberOfHours: { min: 1, max: 100 }, }, }; + +export const APPLICATION_STATUS_TYPES = { + accepted: 'accepted', + rejected: 'rejected', + pending: 'pending', +}; diff --git a/app/services/onboarding.js b/app/services/onboarding.js index 1c2ad2d6..21857ae8 100644 --- a/app/services/onboarding.js +++ b/app/services/onboarding.js @@ -1,12 +1,22 @@ +import { tracked } from '@glimmer/tracking'; import Service, { service } from '@ember/service'; import { TOAST_OPTIONS } from '../constants/toast-options'; import { ERROR_MESSAGES } from '../constants/error-messages'; import { GET_API_CONFIGS, POST_API_CONFIGS } from '../constants/live'; import { APPS } from '../constants/urls'; - export default class OnboardingService extends Service { + @service login; @service store; @service toast; + @tracked applicationData; + + constructor() { + super(...arguments); + + (async () => { + await this.getApplicationDetails(); + })(); + } async signup(dataToUpdate) { try { @@ -98,4 +108,22 @@ export default class OnboardingService extends Service { this.toast.error('Something went wrong!', 'error!', TOAST_OPTIONS); } } + + async getApplicationDetails() { + try { + const userId = this.login.userData.id; + const applicationResponse = await fetch( + `${APPS.API_BACKEND}/applications?userId=${userId}`, + { + credentials: 'include', + }, + ); + const applicationData = await applicationResponse.json(); + + this.applicationData = applicationData?.applications?.[0]; + } catch (err) { + console.error('Error: ', err); + this.toast.error('Some error occured', 'Error ocurred!', TOAST_OPTIONS); + } + } } diff --git a/app/styles/app.css b/app/styles/app.css index bea768d0..06c9ad64 100644 --- a/app/styles/app.css +++ b/app/styles/app.css @@ -32,6 +32,7 @@ @import url("wheel-animations.css"); @import url("goto.module.css"); @import url("event-card.module.css"); +@import url("status-card.module.css"); * { margin: 0; diff --git a/app/styles/status-card.module.css b/app/styles/status-card.module.css new file mode 100644 index 00000000..a824c3cb --- /dev/null +++ b/app/styles/status-card.module.css @@ -0,0 +1,15 @@ +.status-card { + display: flex; + flex-direction: column; + gap: 1.5rem; + align-items: center; +} + +.status-card__heading-Icon { + display: flex; +} + +.status-card__description { + text-align: center; + line-height: 2rem; +} diff --git a/tests/integration/components/status-card-test.js b/tests/integration/components/status-card-test.js new file mode 100644 index 00000000..edf4dc7a --- /dev/null +++ b/tests/integration/components/status-card-test.js @@ -0,0 +1,86 @@ +import { module, test } from 'qunit'; +import { setupRenderingTest } from 'ember-qunit'; +import { render } from '@ember/test-helpers'; +import { hbs } from 'ember-cli-htmlbars'; + +module('Integration | Component | status-card', function (hooks) { + setupRenderingTest(hooks); + + hooks.beforeEach(function () { + this.set('joinDiscordHandler', () => { + window.open = this.spy(); + }); + }); + + test('it renders pending status', async function (assert) { + assert.expect(3); + + this.set('status', 'pending'); + this.set('feedback', 'Feedback for pending status'); + + await render(hbs` + + `); + + assert.dom('[data-test-status-card-heading]').hasText('Pending'); + assert.dom('[data-test-icon="pending"]').exists(); + assert + .dom('[data-test-status-card-description-1]') + .hasText( + 'Your application is currently in pending state, please regularly check this page for invite link.', + ); + }); + + test('it renders rejected status', async function (assert) { + assert.expect(4); + this.set('status', 'rejected'); + this.set('feedback', 'Feedback for rejected status'); + await render(hbs` + + `); + + assert.dom('[data-test-status-card-heading]').hasText('Rejected'); + assert.dom('[data-test-icon="rejected"]').exists(); + assert + .dom('[data-test-status-card-description-1]') + .hasText('Your application is rejected'); + assert + .dom('[data-test-status-card-description-2]') + .hasText("Here's the feedback for your application"); + }); + + test('it renders accepted status', async function (assert) { + assert.expect(5); + + this.set('status', 'accepted'); + this.set('feedback', 'Feedback for accepted status'); + + await render(hbs` + + `); + + assert.dom('[data-test-status-card-heading]').hasText('Accepted'); + assert.dom('[data-test-icon="accepted"]').exists(); + assert + .dom('[data-test-status-card-description-1]') + .hasText('Congratulations! Your application is accepted by us'); + assert + .dom('[data-test-status-card-description-2]') + .hasText("Here's the feedback for your application"); + assert + .dom('[data-test-status-card-description-3]') + .hasText('Feedback for accepted status'); + }); +}); From 9d8fc7d252b93d9ab779be384500dbdda3734a0c Mon Sep 17 00:00:00 2001 From: Satyam Bajpai Date: Tue, 9 Jan 2024 22:03:40 +0530 Subject: [PATCH 08/12] feat: add tooltip in indentity code copy step (#825) * feat: add tooltip in indentity code copy step * refactor: modified tooltip styles * fix:url and failing test * add success toast in signup and fix css for preview * remove double quotes --------- Co-authored-by: vinit717 --- app/components/identity-steps/step-five.hbs | 39 +++++++----- app/components/identity-steps/step-four.hbs | 4 ++ app/components/identity-steps/step-four.js | 6 +- app/components/tooltip.hbs | 6 ++ app/services/onboarding.js | 1 + app/styles/app.css | 1 + app/styles/onboarding-card.module.css | 62 ++++++++++++------- app/styles/stepper.module.css | 1 + app/styles/tooltip.module.css | 59 ++++++++++++++++++ app/templates/join.hbs | 61 +++++++++++------- .../identity-steps/step-five-test.js | 2 +- 11 files changed, 176 insertions(+), 66 deletions(-) create mode 100644 app/components/tooltip.hbs create mode 100644 app/styles/tooltip.module.css diff --git a/app/components/identity-steps/step-five.hbs b/app/components/identity-steps/step-five.hbs index 9dbfc09c..76e4f764 100644 --- a/app/components/identity-steps/step-five.hbs +++ b/app/components/identity-steps/step-five.hbs @@ -28,23 +28,30 @@ {{on 'mouseout' this.closeTooltipInfo}} > + +

What is Profile Service URL?

+
+ Details in the template: + {{! TODO fix this eslint error }} + {{! template-lint-disable no-nested-interactive}} + + https://github.com/RahulGoyal-tech/profile-service-rds + +
+
-
-

What is Profile Service URL?

- -
diff --git a/app/components/identity-steps/step-four.hbs b/app/components/identity-steps/step-four.hbs index 68775569..03e601d9 100644 --- a/app/components/identity-steps/step-four.hbs +++ b/app/components/identity-steps/step-four.hbs @@ -44,6 +44,10 @@ {{on 'click' this.handleCopy}} > +
diff --git a/app/components/identity-steps/step-four.js b/app/components/identity-steps/step-four.js index 9b98be77..7ba8f7f7 100644 --- a/app/components/identity-steps/step-four.js +++ b/app/components/identity-steps/step-four.js @@ -2,13 +2,14 @@ import Component from '@glimmer/component'; import { tracked } from '@glimmer/tracking'; import { action } from '@ember/object'; import { inject as service } from '@ember/service'; -import { toastNotificationTimeoutOptions } from '../../constants/toast-notification'; +import { TOAST_OPTIONS } from '../../constants/toast-options'; export default class IdentityStepsStepFourComponent extends Component { @service toast; @tracked hideChaincode = true; @tracked isCopyClicked = false; @tracked isChaincodePageButtonDisabled = true; + @tracked isTooltipVisible = true; @action toggleEye() { this.hideChaincode = !this.hideChaincode; @@ -17,9 +18,10 @@ export default class IdentityStepsStepFourComponent extends Component { @action handleCopy() { navigator.clipboard.writeText(this.Chaincode); this.isCopyClicked = true; + this.isTooltipVisible = false; this.isChaincodePageButtonDisabled = false; if (this.isCopyClicked === true) { - this.toast.info('Copied', '', toastNotificationTimeoutOptions); + this.toast.info('Copied', '', TOAST_OPTIONS); } } } diff --git a/app/components/tooltip.hbs b/app/components/tooltip.hbs new file mode 100644 index 00000000..18a86136 --- /dev/null +++ b/app/components/tooltip.hbs @@ -0,0 +1,6 @@ + + {{@text}} + \ No newline at end of file diff --git a/app/services/onboarding.js b/app/services/onboarding.js index 21857ae8..0df85fa6 100644 --- a/app/services/onboarding.js +++ b/app/services/onboarding.js @@ -39,6 +39,7 @@ export default class OnboardingService extends Service { }); await user.save(); + this.toast.success('Signup successfully', 'Success!', TOAST_OPTIONS); } catch (error) { this.toast.error( ERROR_MESSAGES.somethingWentWrong, diff --git a/app/styles/app.css b/app/styles/app.css index 06c9ad64..090c4937 100644 --- a/app/styles/app.css +++ b/app/styles/app.css @@ -33,6 +33,7 @@ @import url("goto.module.css"); @import url("event-card.module.css"); @import url("status-card.module.css"); +@import url("tooltip.module.css"); * { margin: 0; diff --git a/app/styles/onboarding-card.module.css b/app/styles/onboarding-card.module.css index 933c8016..37967a66 100644 --- a/app/styles/onboarding-card.module.css +++ b/app/styles/onboarding-card.module.css @@ -117,6 +117,7 @@ border: none; padding: 0 0.4rem; cursor: pointer; + position: relative; } .profile-service-page { @@ -151,6 +152,8 @@ profile-service-page__inputcontainer__input-label { cursor: pointer; border: none; background: none; + padding: 1rem; + position: relative; } .tooltip-info { @@ -162,19 +165,48 @@ profile-service-page__inputcontainer__input-label { var(--color-blackshadow) 0 1px 2px 0, var(--color-blackshadow2) 0 2px 6px 2px; border-radius: 10px; - border-top-left-radius: 0; + border-bottom-right-radius: 0; width: 10rem; padding: 1rem; position: absolute; - left: 20rem; - top: 2rem; + left: -10rem; + top: -5.5rem; display: none; } -.active-tooltip-info { +.tooltip-info--show { + animation: fade-in 0.3s linear 0s 1 normal forwards; display: flex; } +.tooltip-info--hidden { + animation: fade-out 0.3s linear 0s 1 normal forwards; +} + +@keyframes fade-in { + 0% { + opacity: 0.2; + transform: translateY(5px); + } + + 100% { + opacity: 1; + transform: translateY(0); + } +} + +@keyframes fade-out { + 0% { + opacity: 1; + display: block; + } + + 100% { + opacity: 0; + display: none; + } +} + .tooltip-info__container { margin-top: 0.5rem; display: flex; @@ -212,7 +244,7 @@ profile-service-page__inputcontainer__input-label { text-align: center; } -@media (width <= 1024px) { +@media (width <=1024px) { .btn-generateUsername { width: 5rem; font-size: 12px; @@ -223,7 +255,7 @@ profile-service-page__inputcontainer__input-label { } } -@media (width <= 768px) { +@media (width <=768px) { .onboarding-card { width: 30rem; padding: 3.5rem 1rem; @@ -252,19 +284,9 @@ profile-service-page__inputcontainer__input-label { width: 2.8rem; height: 2.5rem; } - - .tooltip-info { - font-size: 0.5rem; - top: 2rem; - left: 11rem; - width: 8rem; - padding: 15px 5px; - border-top-left-radius: 10px; - border-top-right-radius: 0; - } } -@media (width <= 480px) { +@media (width <=480px) { .onboarding-card { width: 82%; } @@ -281,12 +303,6 @@ profile-service-page__inputcontainer__input-label { letter-spacing: 0; } - .tooltip-info { - top: -5rem; - left: 9rem; - border-top-left-radius: 10px; - } - .heading__h3 { line-height: 2rem; } diff --git a/app/styles/stepper.module.css b/app/styles/stepper.module.css index 858fcac7..ad0435d2 100644 --- a/app/styles/stepper.module.css +++ b/app/styles/stepper.module.css @@ -55,6 +55,7 @@ border-radius: 10px; padding: 15px; text-align: justify; + word-wrap: break-word; } /* Step Five: Copy Link */ diff --git a/app/styles/tooltip.module.css b/app/styles/tooltip.module.css new file mode 100644 index 00000000..47baf06d --- /dev/null +++ b/app/styles/tooltip.module.css @@ -0,0 +1,59 @@ +.tooltip { + position: absolute; + font-size: 12px; + width: 12rem; + top: -2.4rem; + left: -11.6rem; + right: 0; + background-color: var(--color-navyblue); + padding: 6px 10px; + border-radius: 5px; + text-align: center; + color: var(--color-white); + z-index: 1; + font-weight: 600; +} + +.tooltip::before { + content: ""; + position: absolute; + transform: rotate(45deg); + height: 12px; + z-index: -1; + inset: 1.2rem 6rem -5px 12.3rem; + width: 13px; + background: var(--color-navyblue); +} + +.tooltip--fade-in { + animation: tooltip-fade-in 0.3s linear 0s 1 normal forwards; + display: block; +} + +.tooltip--fade-out { + animation: tooltip-fade-out 0.3s linear 0s 1 normal forwards; +} + +@keyframes tooltip-fade-in { + 0% { + opacity: 0.2; + transform: translateY(5px); + } + + 100% { + opacity: 1; + transform: translateY(0); + } +} + +@keyframes tooltip-fade-out { + 0% { + opacity: 1; + display: block; + } + + 100% { + opacity: 0; + display: none; + } +} diff --git a/app/templates/join.hbs b/app/templates/join.hbs index d4df938a..6472f931 100644 --- a/app/templates/join.hbs +++ b/app/templates/join.hbs @@ -1,30 +1,43 @@ {{page-title 'Join'}}
{{#if this.login.isLoading}} -
- -
+
+ +
{{else}} - {{#if this.login.userData.roles.archived}} - -

Archived User

-

- This user account has been archived. If you have any questions or need assistance, - please contact Ankush via Twitter. -

- -
- {{else}} - {{#if this.isDevMode}} - - {{else}} - - {{/if}} - {{/if}} + {{#if this.login.userData.roles.archived}} + +

Archived User

+

+ This user account has been archived. If you have any questions or need + assistance, please contact Ankush via Twitter. +

+ +
+ {{else}} + {{#if this.isDevMode}} + + {{else}} + + {{/if}} + {{/if}} {{/if}}
\ No newline at end of file diff --git a/tests/integration/components/identity-steps/step-five-test.js b/tests/integration/components/identity-steps/step-five-test.js index ca661c9a..7d3afec2 100644 --- a/tests/integration/components/identity-steps/step-five-test.js +++ b/tests/integration/components/identity-steps/step-five-test.js @@ -80,7 +80,7 @@ module('Integration | Component | identity-steps/step-five', function (hooks) { await triggerEvent('[data-test=tooltip]', 'mouseover'); - assert.dom('[data-test=tooltip-info]').hasClass('active-tooltip-info'); + assert.dom('[data-test=tooltip-info]').hasClass('tooltip-info'); }); test('Not Display Tooltip Information on Mouse Out', async function (assert) { From e1527f16768d64816c0586b39d77b16f20b843d5 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 12 Jan 2024 13:59:34 +0530 Subject: [PATCH 09/12] chore(deps): bump follow-redirects from 1.15.2 to 1.15.4 (#834) Bumps [follow-redirects](https://github.com/follow-redirects/follow-redirects) from 1.15.2 to 1.15.4. - [Release notes](https://github.com/follow-redirects/follow-redirects/releases) - [Commits](https://github.com/follow-redirects/follow-redirects/compare/v1.15.2...v1.15.4) --- updated-dependencies: - dependency-name: follow-redirects dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- yarn.lock | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/yarn.lock b/yarn.lock index 2e467346..889a4772 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9005,9 +9005,9 @@ flatted@^3.2.9: integrity sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ== follow-redirects@^1.0.0: - version "1.15.2" - resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.2.tgz#b460864144ba63f2681096f274c4e57026da2c13" - integrity sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA== + version "1.15.4" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.4.tgz#cdc7d308bf6493126b17ea2191ea0ccf3e535adf" + integrity sha512-Cr4D/5wlrb0z9dgERpUL3LrmPKVDsETIJhaCMeDfuFYcqa5bldGV6wBsAN6X/vxlXQtFBMrXdXxdL8CbDTGniw== for-in@^1.0.2: version "1.0.2" From 77dcf0669c23a6d769d7a339dda74f5d0f04f3aa Mon Sep 17 00:00:00 2001 From: Satyam Bajpai Date: Tue, 16 Jan 2024 13:57:09 +0530 Subject: [PATCH 10/12] Add question answers word cloud feature (#771) * feat: add initial tab section for survey page * feat: add ask-question-modal component * feat: create answer reply modal * refactor: make question modal responsive * refactor: answer and question modal code * feat: added question and answers api * feat: add word cloud with api * styles: improve styles of survey page * feat: add filter to answers * feat: add min length for answer and handle max_characters null case * feat: add wordCloud feature flag * chore: remove unnecessary code * feat: add toast messages for error cases in approving rejecting answers * feat: add word cloud feature flag to word-cloud component * tests: fix failing tests * refactor: remove unused code * styles: add colors to variables * fix: eslint no-duplicate-id error --- .eslintrc.js | 1 + app/components/answer-reply-modal.hbs | 46 +++ app/components/answer-view-card.hbs | 36 +++ app/components/answer-view-card.js | 36 +++ app/components/ask-question-modal.hbs | 62 ++++ app/components/events/survey-page.hbs | 68 ++++ app/components/events/survey-page.js | 161 ++++++++++ app/components/live-header.hbs | 32 +- app/components/live-header.js | 6 + app/components/reusables/button.hbs | 9 +- app/components/reusables/input-box.hbs | 18 +- app/components/word-cloud.hbs | 6 + app/components/word-cloud.js | 24 ++ app/constants/live.js | 9 + app/controllers/live.js | 199 +++++++++++- app/d3/word-cloud.js | 122 +++++++ app/services/feature-flag.js | 5 + app/services/live.js | 2 + app/services/survey.js | 173 ++++++++++ app/styles/answer-reply-modal.module.css | 76 +++++ app/styles/answer-view-card.module.css | 48 +++ app/styles/app.css | 5 + app/styles/ask-question-modal.module.css | 147 +++++++++ app/styles/button.module.css | 5 + app/styles/icon-button.module.css | 31 +- app/styles/input.module.css | 38 +++ app/styles/survey-page.module.css | 43 +++ app/styles/variables.css | 2 + app/styles/word-cloud.module.css | 4 + app/templates/live.hbs | 68 ++-- app/utils/common-utils.js | 11 + package.json | 3 + .../components/answer-reply-modal-test.js | 17 + .../components/ask-question-modal-test.js | 17 + .../components/events/survey-page-test.js | 14 + .../integration/components/word-cloud-test.js | 17 + yarn.lock | 300 +++++++++++++++++- 37 files changed, 1803 insertions(+), 58 deletions(-) create mode 100644 app/components/answer-reply-modal.hbs create mode 100644 app/components/answer-view-card.hbs create mode 100644 app/components/answer-view-card.js create mode 100644 app/components/ask-question-modal.hbs create mode 100644 app/components/events/survey-page.hbs create mode 100644 app/components/events/survey-page.js create mode 100644 app/components/word-cloud.hbs create mode 100644 app/components/word-cloud.js create mode 100644 app/d3/word-cloud.js create mode 100644 app/services/survey.js create mode 100644 app/styles/answer-reply-modal.module.css create mode 100644 app/styles/answer-view-card.module.css create mode 100644 app/styles/ask-question-modal.module.css create mode 100644 app/styles/survey-page.module.css create mode 100644 app/styles/word-cloud.module.css create mode 100644 app/utils/common-utils.js create mode 100644 tests/integration/components/answer-reply-modal-test.js create mode 100644 tests/integration/components/ask-question-modal-test.js create mode 100644 tests/integration/components/events/survey-page-test.js create mode 100644 tests/integration/components/word-cloud-test.js diff --git a/.eslintrc.js b/.eslintrc.js index 2e92f79b..8b48a31a 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -24,6 +24,7 @@ module.exports = { }, rules: { 'qunit/require-expect': [1, 'except-simple'], + 'no-self-assign': ['warn'], }, overrides: [ // node files diff --git a/app/components/answer-reply-modal.hbs b/app/components/answer-reply-modal.hbs new file mode 100644 index 00000000..0c2c71ed --- /dev/null +++ b/app/components/answer-reply-modal.hbs @@ -0,0 +1,46 @@ + +
+

Host asked you a question😀

+
+ +

+ ¡ + Do not use abusive words, this event is moderated!

+
+ + +
+ +
+ +
\ No newline at end of file diff --git a/app/components/answer-view-card.hbs b/app/components/answer-view-card.hbs new file mode 100644 index 00000000..f5759eb8 --- /dev/null +++ b/app/components/answer-view-card.hbs @@ -0,0 +1,36 @@ +
+

+ {{this.answerText}} + + {{#if this.isTextMoreThanMaxCharacters}} + + {{/if}} +

+ +
+ + +
+
\ No newline at end of file diff --git a/app/components/answer-view-card.js b/app/components/answer-view-card.js new file mode 100644 index 00000000..b69493b4 --- /dev/null +++ b/app/components/answer-view-card.js @@ -0,0 +1,36 @@ +import Component from '@glimmer/component'; +import { tracked } from '@glimmer/tracking'; +import { readMoreFormatter } from '../utils/common-utils'; +import { action } from '@ember/object'; +import { ANSWER_STATUS } from '../constants/live'; + +const maxCharactersToShow = 70; +export default class AnswerViewCardComponent extends Component { + @tracked answerText = readMoreFormatter( + this.args.answerObject.answer, + maxCharactersToShow, + ); + + @tracked isTextMoreThanMaxCharacters = + this.args.answerObject.answer?.length > maxCharactersToShow; + @tracked isReadMoreEnabled = false; + @tracked readMoreOrLessText = this.isReadMoreEnabled ? 'Less' : 'More'; + @tracked isApproved = + this.args.answerObject.status === ANSWER_STATUS.APPROVED; + @tracked isPending = this.args.answerObject.status === ANSWER_STATUS.PENDING; + @tracked isRejected = + this.args.answerObject.status === ANSWER_STATUS.REJECTED; + + @action toggleReadMore() { + this.isReadMoreEnabled = !this.isReadMoreEnabled; + this.readMoreOrLessText = this.isReadMoreEnabled ? 'Less' : 'More'; + if (this.isReadMoreEnabled) { + this.answerText = this.args.answerObject.answer; + } else { + this.answerText = readMoreFormatter( + this.args.answerObject.answer, + maxCharactersToShow, + ); + } + } +} diff --git a/app/components/ask-question-modal.hbs b/app/components/ask-question-modal.hbs new file mode 100644 index 00000000..9fe79e34 --- /dev/null +++ b/app/components/ask-question-modal.hbs @@ -0,0 +1,62 @@ + +
+

Ask Question

+ +
+ + +
+ +
+ + +
+
+
\ No newline at end of file diff --git a/app/components/events/survey-page.hbs b/app/components/events/survey-page.hbs new file mode 100644 index 00000000..9d09a710 --- /dev/null +++ b/app/components/events/survey-page.hbs @@ -0,0 +1,68 @@ +
+ +
+ +
+

Recent Question

+

{{(or + @questionAsked.question "No recent question" + )}}

+
+
+
+

+ Answers +

+
+ + +
+
+ {{#if this.isAnswersPresent}} + {{#each this.answers as |answer|}} + + {{/each}} + {{else}} +
No answers present currently!
+ {{/if}} + +
+
+
\ No newline at end of file diff --git a/app/components/events/survey-page.js b/app/components/events/survey-page.js new file mode 100644 index 00000000..b56d32d1 --- /dev/null +++ b/app/components/events/survey-page.js @@ -0,0 +1,161 @@ +import Component from '@glimmer/component'; +import { action } from '@ember/object'; +import { tracked } from '@glimmer/tracking'; +import { debounce } from '@ember/runloop'; +import { inject as service } from '@ember/service'; +import { + QUESTION_DEBOUNCE_TIME, + QUESTION_MIN_LENGTH, + POST_API_CONFIGS, +} from '../../constants/live'; +import { APPS } from '../../constants/urls'; +import { TOAST_OPTIONS } from '../../constants/toast-options'; + +export default class SurveyPageComponent extends Component { + @service live; + @service login; + @service toast; + @tracked isMaxCharactersChecked = false; + @tracked isAskQuestionModalOpen = false; + @tracked maxCharacters; + @tracked question = ''; + @tracked isQuestionValid = false; + @tracked isQuestionSubmitButtonDisabled = true; + @tracked isQuestionApiLoading = false; + @tracked ANSWER_STATUS_FILTERS = ['ALL', 'PENDING', 'APPROVED', 'REJECTED']; + @tracked activeAnswerFilterValue = 'ALL'; + @tracked userData = this.login?.userData; + get isAnswersPresent() { + return Boolean(this.answers?.length); + } + + get answers() { + const allAnswers = this.args.answers; + + if (this.activeAnswerFilterValue === 'ALL') return allAnswers; + + const answersToShow = allAnswers.filter( + (answer) => answer.status === this.activeAnswerFilterValue, + ); + + return answersToShow; + } + + get isAskQuestionButtonDisabled() { + return !this.userData?.roles?.super_user; + } + + @action onAnswerFilterChange(event) { + this.activeAnswerFilterValue = event.target.value; + } + + @action openAskQuestionModal() { + this.isAskQuestionModalOpen = true; + } + + @action closeAskQuestionModal() { + this.isAskQuestionModalOpen = false; + this.onQuestionModalUnmount(); + } + + @action async onQuestionSubmit() { + this.isQuestionApiLoading = true; + const questionBody = { + question: this.question.trim(), + createdBy: this.login.userData.id, + eventId: this.live.activeRoomId, + maxCharacters: this.maxCharacters || null, + }; + + try { + const questionResponse = await fetch(`${APPS.API_BACKEND}/questions`, { + ...POST_API_CONFIGS, + body: JSON.stringify(questionBody), + }); + const question = await questionResponse.json(); + + if (!questionResponse.ok) + return this.toast.error( + question.message, + question.error, + TOAST_OPTIONS, + ); + + this.toast.success(question.message, question.error, TOAST_OPTIONS); + } catch (error) { + console.error(error); + } finally { + this.isQuestionApiLoading = false; + this.onQuestionModalUnmount(); + this.isAskQuestionModalOpen = false; + } + } + + @action toggleMaxCharacterChecked() { + this.isMaxCharactersChecked = !this.isMaxCharactersChecked; + + if (!this.isMaxCharactersChecked) { + this.maxCharacters = null; + } + + if (!this.isMaxCharactersChecked && this.isQuestionValid) { + this.isQuestionSubmitButtonDisabled = false; + } else { + this.isQuestionSubmitButtonDisabled = true; + } + } + + @action onCharacterLimitInput(event) { + this.maxCharacters = event.target.value && Number(event.target.value); + + if (!this.isMaxCharactersChecked && this.isQuestionValid) { + this.isQuestionSubmitButtonDisabled = false; + return; + } + + if (this.maxCharacters && this.isQuestionValid) { + this.isQuestionSubmitButtonDisabled = false; + return; + } + + this.isQuestionSubmitButtonDisabled = true; + } + + @action onQuestionInput(event) { + const setQuestion = () => { + this.question = event.target.value; + + if (this.question.length > QUESTION_MIN_LENGTH) { + this.isQuestionValid = true; + } else { + this.isQuestionValid = false; + } + + if (!this.isMaxCharactersChecked && this.isQuestionValid) { + this.isQuestionSubmitButtonDisabled = false; + return; + } + + if ( + this.isMaxCharactersChecked && + this.maxCharacters && + this.isQuestionValid + ) { + this.isQuestionSubmitButtonDisabled = false; + return; + } + + this.isQuestionSubmitButtonDisabled = true; + }; + + debounce(setQuestion, QUESTION_DEBOUNCE_TIME, event.target.value); + } + + onQuestionModalUnmount() { + this.isMaxCharactersChecked = false; + this.isQuestionValid = false; + this.isQuestionSubmitButtonDisabled = true; + this.question = ''; + this.maxCharacters = null; + } +} diff --git a/app/components/live-header.hbs b/app/components/live-header.hbs index 40d8e018..d6ae77ea 100644 --- a/app/components/live-header.hbs +++ b/app/components/live-header.hbs @@ -1,27 +1,37 @@ - -
+ +
{{@activeTab}}
-
+
{{#each @tabs as |tab|}} - {{#if (eq tab.label 'Logs')}} + {{#if (eq tab.label "Logs")}} {{#if this.live.userData.roles.super_user}} + {{/if}} + {{else if (eq tab.label "Survey")}} + {{#if this.isWordCloudFeatureOn}} + {{/if}} @@ -30,7 +40,7 @@ @tabId={{tab.id}} @test={{tab.label}} @label={{tab.label}} - @variant={{if tab.active 'active' ''}} + @variant={{if tab.active "active" ""}} @onClick={{@tabHandler}} /> {{/if}} diff --git a/app/components/live-header.js b/app/components/live-header.js index e359cd19..b8b446be 100644 --- a/app/components/live-header.js +++ b/app/components/live-header.js @@ -5,7 +5,13 @@ import { inject as service } from '@ember/service'; export default class LiveHeaderComponent extends Component { @service live; + @service featureFlag; @tracked isTabOpen = false; + + get isWordCloudFeatureOn() { + return this.featureFlag.isWordCloud; + } + @action toggleTabs() { this.isTabOpen = !this.isTabOpen; } diff --git a/app/components/reusables/button.hbs b/app/components/reusables/button.hbs index 21d07845..60c9caf1 100644 --- a/app/components/reusables/button.hbs +++ b/app/components/reusables/button.hbs @@ -1,14 +1,15 @@ \ No newline at end of file diff --git a/app/components/reusables/input-box.hbs b/app/components/reusables/input-box.hbs index 031269ef..4d842bb4 100644 --- a/app/components/reusables/input-box.hbs +++ b/app/components/reusables/input-box.hbs @@ -1,17 +1,25 @@ -
- +
+ {{#if @required}} - * + * {{/if}} + {{#if @shouldShowHelperText}} +

{{@helperText}}

+ {{/if}}
\ No newline at end of file diff --git a/app/components/word-cloud.hbs b/app/components/word-cloud.hbs new file mode 100644 index 00000000..c7d9b634 --- /dev/null +++ b/app/components/word-cloud.hbs @@ -0,0 +1,6 @@ +{{! word cloud forms in this component with the help of d3.js }} +{{#if @isWordCloud}} +
+ +
+{{/if}} \ No newline at end of file diff --git a/app/components/word-cloud.js b/app/components/word-cloud.js new file mode 100644 index 00000000..e0824ed5 --- /dev/null +++ b/app/components/word-cloud.js @@ -0,0 +1,24 @@ +import Component from '@glimmer/component'; +import { inject as service } from '@ember/service'; +import { registerDestructor } from '@ember/destroyable'; + +const SHOW_WORD_CLOUD_AFTER_TIME = 1000; //time in milliseconds +export default class WordCloudComponent extends Component { + @service survey; + @service fastboot; + + constructor() { + super(...arguments); + let timeout; + + if (!this.fastboot.isFastBoot) { + timeout = setTimeout(() => { + this.survey.showWordCloud(); + }, SHOW_WORD_CLOUD_AFTER_TIME); + } + + registerDestructor(this, () => { + clearTimeout(timeout); + }); + } +} diff --git a/app/constants/live.js b/app/constants/live.js index 501cfcc5..02dc92ea 100644 --- a/app/constants/live.js +++ b/app/constants/live.js @@ -39,3 +39,12 @@ export const EVENTS_LOGS_POLL_TIME = 40000; export const EVENTS_LOGS_TYPE = { EVENTS_REMOVE_PEER: 'EVENTS_REMOVE_PEER', }; + +export const QUESTION_DEBOUNCE_TIME = 200; +export const QUESTION_MIN_LENGTH = 2; +export const ANSWER_MIN_LENGTH = 2; +export const ANSWER_STATUS = { + PENDING: 'PENDING', + APPROVED: 'APPROVED', + REJECTED: 'REJECTED', +}; diff --git a/app/controllers/live.js b/app/controllers/live.js index e432b0a8..c39c5db4 100644 --- a/app/controllers/live.js +++ b/app/controllers/live.js @@ -4,18 +4,30 @@ import { tracked } from '@glimmer/tracking'; import { inject as service } from '@ember/service'; import { getOwner } from '@ember/application'; import { globalRef } from 'ember-ref-bucket'; -import { ROLES, BUTTONS_TYPE } from '../constants/live'; +import { registerDestructor } from '@ember/destroyable'; +import { + ROLES, + BUTTONS_TYPE, + ANSWER_STATUS, + ANSWER_MIN_LENGTH, +} from '../constants/live'; import { TOAST_OPTIONS } from '../constants/toast-options'; +import { APPS } from '../constants/urls'; export default class LiveController extends Controller { queryParams = ['dev']; ROLES = ROLES; + @service featureFlag; @service login; @service toast; + @service fastboot; + @service survey; + answerEventSource; + questionEventSource; @tracked TABS = [ { id: 1, label: 'Screenshare', active: true }, - { id: 2, label: 'Previous Events', active: false }, - { id: 3, label: 'Real Dev Squad', active: false }, + { id: 2, label: 'Survey', active: false }, { id: 4, label: 'Logs', active: false }, + { id: 3, label: 'More', active: false }, ]; @tracked activeTab = 'Screenshare'; @tracked isLoading = false; @@ -30,6 +42,17 @@ export default class LiveController extends Controller { @tracked newRoomCode = ''; @tracked isActiveEventFound; @tracked buttonText = ''; + @tracked isAnswerReplyModalOpen = false; + @tracked answerValue = ''; + @tracked answerValidationDetails = { + isError: false, + isHelperTextVisible: true, + helperText: `Minimum character limit is ${ANSWER_MIN_LENGTH} characters`, + }; + @tracked answerSubmitButtonState = { + isDisabled: true, + isLoading: false, + }; @globalRef('videoEl') videoEl; get liveService() { return getOwner(this).lookup('service:live'); @@ -37,9 +60,24 @@ export default class LiveController extends Controller { constructor() { super(...arguments); + + if (!this.fastboot.isFastBoot) { + const queryParams = new URLSearchParams(window.location.search); + const isWordCloudFeatureOn = queryParams.get('wordCloud') === 'true'; + + if (isWordCloudFeatureOn) { + this.questionSSEListener(); + this.answerSSEListener(); + } + } setTimeout(() => { this.isLoading = false; }, 4000); + + registerDestructor(this, () => { + this.questionEventSource?.close(); + this.answerEventSource?.close(); + }); } @action inputHandler(type, event) { @@ -136,6 +174,82 @@ export default class LiveController extends Controller { this.isWarningModalOpen = !this.isWarningModalOpen; } + @action openAnswerReplyModal() { + this.isAnswerReplyModalOpen = true; + } + + @action closeAnswerReplyModal() { + this.isAnswerReplyModalOpen = false; + } + + @action onAnswerInput(event) { + const maxCharacters = this.survey.recentQuestion.max_characters; + + this.answerValue = event.target.value; + const answerLength = this.answerValue.trim().length; + const isAnswerEqualToMinLength = answerLength >= ANSWER_MIN_LENGTH; + + if (!isAnswerEqualToMinLength) { + this.answerValidationDetails.helperText = `Minimum character limit is ${ANSWER_MIN_LENGTH} characters`; + this.answerValidationDetails.isHelperTextVisible = true; + + this.answerValidationDetails = this.answerValidationDetails; + + this.answerSubmitButtonState.isDisabled = true; + this.answerSubmitButtonState = this.answerSubmitButtonState; + + return; + } + + if (maxCharacters === null) { + this.resetAnswerValidators(); + return; + } + + if (this.answerValue.trim().length > maxCharacters) { + this.answerValidationDetails.isError = true; + this.answerValidationDetails.helperText = `Maximum character limit is ${maxCharacters} characters`; + this.answerValidationDetails.isHelperTextVisible = true; + this.answerValidationDetails = this.answerValidationDetails; + + this.answerSubmitButtonState.isDisabled = true; + this.answerSubmitButtonState = this.answerSubmitButtonState; + } else { + this.resetAnswerValidators(); + } + } + + @action async submitAnswer() { + this.answerSubmitButtonState.isLoading = true; + this.answerSubmitButtonState.isDisabled = true; + + this.answerSubmitButtonState = this.answerSubmitButtonState; + + const answerBody = { + answer: this.answerValue.trim(), + answeredBy: this.liveService.localPeer?.id, + eventId: this.liveService?.activeRoomId, + questionId: this.survey.recentQuestion?.id, + }; + + const { error } = await this.survey.answerSubmitHandler(answerBody); + + if (!error) { + this.isAnswerReplyModalOpen = false; + this.answerSubmitButtonState.isLoading = false; + this.answerSubmitButtonState.isDisabled = false; + this.answerSubmitButtonState = this.answerSubmitButtonState; + } + } + + @action async onAnswerReject(id) { + this.survey.answerRejectHandler(id); + } + + @action async onAnswerApprove(id) { + this.survey.answerApproveHandler(id); + } + @action buttonClickHandler(buttonId) { switch (buttonId) { case BUTTONS_TYPE.SCREEN_SHARE: @@ -172,4 +286,83 @@ export default class LiveController extends Controller { this.newRoomCode = ''; } } + + resetAnswerValidators() { + this.answerValidationDetails.isError = false; + this.answerValidationDetails.helperText = ''; + this.answerValidationDetails.isHelperTextVisible = false; + this.answerValidationDetails = this.answerValidationDetails; + + this.answerSubmitButtonState.isDisabled = false; + this.answerSubmitButtonState = this.answerSubmitButtonState; + } + questionSSEListener() { + const event = new EventSource(`${APPS.API_BACKEND}/questions`); + this.questionEventSource = event; + + event.onmessage = async (event) => { + const parsedQuestion = JSON.parse(event.data); + const question = parsedQuestion || {}; + + const isQuestionChanged = question?.id !== this.survey.recentQuestion?.id; + + this.survey.setRecentQuestion(question); + + this.answerValue = ''; + this.answerValidationDetails.isError = false; + this.answerValidationDetails.helperText = `Minimum character limit is ${ANSWER_MIN_LENGTH} characters`; + this.answerValidationDetails.isHelperTextVisible = true; + this.answerValidationDetails = this.answerValidationDetails; + + if (isQuestionChanged) { + this.answerEventSource?.close(); + this.answerSSEListener(); + } + + if ( + question && + this.liveService.isJoined && + this.liveService.localPeer.roleName !== this.ROLES.host + ) { + this.isAnswerReplyModalOpen = true; + } + }; + + event.onerror = (event) => { + console.error(event); + }; + } + + answerSSEListener() { + const localPeerRole = this.liveService.localPeer?.roleName; + const isHost = localPeerRole === this.ROLES.host; + const isModerator = localPeerRole === this.ROLES.moderator; + const activeEventId = this.liveService?.activeRoomId; + let answersEventStreamURL = ''; + + if (isHost || isModerator) { + answersEventStreamURL = `${APPS.API_BACKEND}/answers?eventId=${activeEventId}&questionId=${this.survey.recentQuestion?.id}`; + } else { + answersEventStreamURL = `${APPS.API_BACKEND}/answers?eventId=${activeEventId}&questionId=${this.survey.recentQuestion?.id}&status=${ANSWER_STATUS.APPROVED}`; + } + + const event = new EventSource(answersEventStreamURL); + this.answerEventSource = event; + + event.onmessage = async (event) => { + const parsedAnswers = JSON.parse(event.data); + const answers = parsedAnswers || []; + + if (isHost || isModerator) { + this.survey.setAnswers(answers); + } else { + this.survey.setApprovedAnswers(answers); + } + this.survey.showWordCloud(); + }; + + event.onerror = (event) => { + console.error(event); + }; + } } diff --git a/app/d3/word-cloud.js b/app/d3/word-cloud.js new file mode 100644 index 00000000..e1dcdd8b --- /dev/null +++ b/app/d3/word-cloud.js @@ -0,0 +1,122 @@ +import cloud from 'd3-cloud'; +import { select } from 'd3-selection'; + +const colors = [ + '#FF6633', + '#FFB399', + '#FF33FF', + '#FFFF99', + '#00B3E6', + '#E6B333', + '#3366E6', + '#999966', + '#99FF99', + '#B34D4D', + '#80B300', + '#809900', + '#E6B3B3', + '#6680B3', + '#66991A', + '#FF99E6', + '#CCFF1A', + '#FF1A66', + '#E6331A', + '#33FFCC', + '#66994D', + '#B366CC', + '#4D8000', + '#B33300', + '#CC80CC', + '#66664D', + '#991AFF', + '#E666FF', + '#4DB3FF', + '#1AB399', + '#E666B3', + '#33991A', + '#CC9999', + '#B3B31A', + '#00E680', + '#4D8066', + '#809980', + '#E6FF80', + '#1AFF33', + '#999933', + '#FF3380', + '#CCCC00', + '#66E64D', + '#4D80CC', + '#9900B3', + '#E64D66', + '#4DB380', + '#FF4D4D', + '#99E6E6', + '#6666FF', +]; + +const defaultSize = { + x: 500, + y: 400, +}; +function generateWordCloud(words, elementSelector, size = defaultSize) { + const container = select(elementSelector); + container.selectAll('*').remove(); + + var layout = cloud() + .size([size.x, size.y]) + .words( + words?.map(function (d) { + return { + text: d, + size: 10 + Math.random() * 37, + color: colors[Math.floor(Math.random() * colors.length)], + }; + }), + ) + .padding(5) + .rotate(function () { + return ~~(Math.random() * 2) * 90; + }) + .font('raleway') + .fontSize(function (d) { + return d.size; + }) + .timeInterval(100) + .on('end', draw); + + function draw(words) { + const container = select(elementSelector); + + container + .append('svg') + .attr('width', layout.size()[0]) + .attr('height', layout.size()[1]) + .append('g') + .attr( + 'transform', + 'translate(' + layout.size()[0] / 2 + ',' + layout.size()[1] / 2 + ')', + ) + .selectAll('text') + .data(words) + .enter() + .append('text') + .style('font-size', function (d) { + return d.size + 'px'; + }) + .style('fill', function (d) { + return d.color; + }) + .style('font-family', 'Impact') + .attr('text-anchor', 'middle') + .attr('transform', function (d) { + return 'translate(' + [d.x, d.y] + ')rotate(' + d.rotate + ')'; + }) + .text(function (d) { + return d.text; + }); + } + + //layout started + layout.start(); +} +export { generateWordCloud }; diff --git a/app/services/feature-flag.js b/app/services/feature-flag.js index 87fa2ec7..6ee6188e 100644 --- a/app/services/feature-flag.js +++ b/app/services/feature-flag.js @@ -7,4 +7,9 @@ export default class FeatureFlagService extends Service { const queryParams = this.router?.currentRoute?.queryParams; return queryParams?.dev === 'true'; } + + get isWordCloud() { + const queryParams = this.router?.currentRoute?.queryParams; + return queryParams?.wordCloud === 'true'; + } } diff --git a/app/services/live.js b/app/services/live.js index 92c9fd8b..d232e848 100644 --- a/app/services/live.js +++ b/app/services/live.js @@ -327,6 +327,7 @@ export default class LiveService extends Service { }); const peer = this.hmsStore.getState(selectLocalPeer); this.localPeer = peer; + this.activeRoomId = roomId; const addedPeerData = await this.addPeer(roomId, peer); if (addedPeerData) { this.toast.success( @@ -346,6 +347,7 @@ export default class LiveService extends Service { }); const peer = this.hmsStore.getState(selectLocalPeer); this.localPeer = peer; + this.activeRoomId = roomId; const addedPeerData = await this.addPeer(roomId, peer); if (addedPeerData) { this.toast.success( diff --git a/app/services/survey.js b/app/services/survey.js new file mode 100644 index 00000000..28bbe68d --- /dev/null +++ b/app/services/survey.js @@ -0,0 +1,173 @@ +import Service, { inject as service } from '@ember/service'; +import { registerDestructor } from '@ember/destroyable'; +import { tracked } from '@glimmer/tracking'; +import { TOAST_OPTIONS } from '../constants/toast-options'; +import { APPS } from '../constants/urls'; +import { + ANSWER_STATUS, + API_METHOD, + PATCH_API_CONFIGS, + ROLES, +} from '../constants/live'; +import { generateWordCloud } from '../d3/word-cloud'; + +export default class SurveyService extends Service { + @service router; + @service toast; + @service live; + @service fastboot; + @tracked answers = []; + @tracked approvedAnswers = []; + @tracked recentQuestion; + @tracked screenWidth; + + constructor() { + super(...arguments); + + const onResize = () => { + this.screenWidth = window.innerWidth; + this.showWordCloud(); + }; + + if (!this.fastboot.isFastBoot) { + this.screenWidth = window.innerWidth; + window.addEventListener('resize', onResize); + registerDestructor(this, () => { + window.removeEventListener('resize', onResize); + }); + } + } + setApprovedAnswers(approvedAnswers) { + this.approvedAnswers = approvedAnswers; + } + + setAnswers(answers) { + this.answers = answers; + } + + setRecentQuestion(question) { + this.recentQuestion = question; + } + + async answerSubmitHandler(payload) { + let error = null, + answer = null; + + try { + const answerResponse = await fetch(`${APPS.API_BACKEND}/answers`, { + method: API_METHOD.POST, + headers: { + 'Content-Type': 'application/json', + }, + + body: JSON.stringify(payload), + }); + answer = await answerResponse.json(); + + if (!answerResponse.ok) + return this.toast.error(answer.message, answer.error, TOAST_OPTIONS); + + this.toast.success(answer.message, answer.error, TOAST_OPTIONS); + + return { error: error, result: answer }; + } catch (error) { + console.error('Error while submitting answer: ', error); + return { error: error, result: null }; + } + } + async answerApproveHandler(id) { + const actualId = id.split('approve-button-')[1]; + const approvalPayload = { + status: ANSWER_STATUS.APPROVED, + }; + + try { + const approveResponse = await fetch( + `${APPS.API_BACKEND}/answers/${actualId}`, + { + ...PATCH_API_CONFIGS, + body: JSON.stringify(approvalPayload), + }, + ); + + if (!approveResponse.ok) throw new Error(); + + this.toast.success( + 'Answer approved successfully', + 'Success', + TOAST_OPTIONS, + ); + } catch (error) { + console.error('Error while approving answer: ', error); + this.toast.error('Error while approving answer', 'Error', TOAST_OPTIONS); + } + } + + async answerRejectHandler(id) { + const actualId = id.split('reject-button-')[1]; + const rejectionPayload = { + status: ANSWER_STATUS.REJECTED, + }; + + try { + const rejectResponse = await fetch( + `${APPS.API_BACKEND}/answers/${actualId}`, + { + ...PATCH_API_CONFIGS, + body: JSON.stringify(rejectionPayload), + }, + ); + + if (!rejectResponse.ok) throw new Error(); + + this.toast.success( + 'Answer rejected successfully', + 'Success', + TOAST_OPTIONS, + ); + } catch (error) { + console.error('Error while rejecting answer: ', error); + this.toast.error('Error while rejecting answer', 'Error', TOAST_OPTIONS); + } + } + + getFilteredApprovedAnswersArray() { + const isHost = this.live.localPeer?.roleName === ROLES.host; + const isModerator = this.live.localPeer?.roleName === ROLES.moderator; + const filteredApprovedAnswersArray = []; + + if (isHost || isModerator) { + const filteredApprovedAnswers = this.answers?.filter( + (answer) => answer.status === ANSWER_STATUS.APPROVED, + ); + filteredApprovedAnswers?.forEach((answer) => { + filteredApprovedAnswersArray.push(answer.answer); + }); + return filteredApprovedAnswersArray; + } + + this.approvedAnswers?.forEach((answer) => { + filteredApprovedAnswersArray.push(answer.answer); + }); + return filteredApprovedAnswersArray; + } + + showWordCloud() { + const element = '.word-cloud'; + const words = this.getFilteredApprovedAnswersArray(); + + let wordCloudSize = { + x: this.screenWidth, + y: this.screenWidth, + }; + + if (!words?.length) return; + + //for mobile/small screens + if (this.screenWidth < 500) + return generateWordCloud(words, element, wordCloudSize); + + // for screen >=500px + generateWordCloud(words, element); + } +} diff --git a/app/styles/answer-reply-modal.module.css b/app/styles/answer-reply-modal.module.css new file mode 100644 index 00000000..107bd4c7 --- /dev/null +++ b/app/styles/answer-reply-modal.module.css @@ -0,0 +1,76 @@ +.answer-reply-modal { + width: 31.25rem; + box-sizing: border-box; + background-color: var(--color-white); + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + max-width: 96svw; + border-radius: 25px; + padding: 1.25rem; + padding-right: 40px; +} + +.answer-reply-modal__heading { + text-align: center; + color: var(--color-pink); + font-size: 1.5rem; + text-decoration: underline; + font-style: normal; + font-weight: 800; + line-height: normal; + margin-bottom: 40px; +} + +.answer-reply-modal__answer-label { + font-size: 1.2rem; + font-weight: 600; + color: var(--color-navyblue); +} + +.answer-reply-modal__answer-input { + width: 100%; + padding: 10px; + box-sizing: border-box; + font-size: 1.2rem; + border: 2px solid var(--color-navyblue); + border-radius: 10px; + margin-top: 10px; +} + +.answer-reply-modal__answer-input:focus { + outline: none; + box-shadow: 0 0 0 4px var(--color-light-navyblue); +} + +.answer-reply-modal__info-text { + font-size: 0.8rem; + color: var(--color-lightgrey); +} + +.answer-reply-modal__info-icon { + border: 1px solid var(--color-darkgrey); + padding: 1px 7px; + border-radius: 50%; + font-size: 0.7rem; + color: var(--color-darkgrey); +} + +.answer-reply-modal__actions { + text-align: center; +} + +/* media queries */ +@media only screen and (width <= 425px) { + .answer-reply-modal__submit-button, + .answer-reply-modal__cancel-button { + margin-right: 0; + width: fit-content; + padding: 0 20px; + } + + .answer-reply-modal__submit-button { + padding: 0 34px; + } +} diff --git a/app/styles/answer-view-card.module.css b/app/styles/answer-view-card.module.css new file mode 100644 index 00000000..b3383df1 --- /dev/null +++ b/app/styles/answer-view-card.module.css @@ -0,0 +1,48 @@ +.answer-view-card { + box-shadow: 0 0 4px 0 var(--color-lightgrey); + min-height: 5rem; + height: fit-content; + display: flex; + justify-content: space-between; + align-items: center; + flex: 3 1; + border-radius: 10px; + box-sizing: border-box; + padding: 1rem; +} + +.answer-view-card__text { + width: 80%; +} + +.answer-view-card__read-more-button { + all: unset; + color: var(--color-pink); + font-weight: 600; + cursor: pointer; +} + +.answer-view-card--pending { + background-color: var(--color-yellow-low-opacity); +} + +.answer-view-card--approved { + background-color: var(--color-green-low-opacity); +} + +.answer-view-card--rejected { + background-color: var(--color-pink-low-opacity); +} + +@media (width <=1024px) { + .answer-view-card { + flex-direction: column; + gap: 10px; + } + + .answer-view-card__text { + width: 100%; + padding-bottom: 10px; + border-bottom: 1px solid var(--color-lightgrey); + } +} diff --git a/app/styles/app.css b/app/styles/app.css index 090c4937..459dc41c 100644 --- a/app/styles/app.css +++ b/app/styles/app.css @@ -31,6 +31,11 @@ @import url("identity-card.module.css"); @import url("wheel-animations.css"); @import url("goto.module.css"); +@import url("survey-page.module.css"); +@import url("ask-question-modal.module.css"); +@import url("answer-reply-modal.module.css"); +@import url("answer-view-card.module.css"); +@import url("word-cloud.module.css"); @import url("event-card.module.css"); @import url("status-card.module.css"); @import url("tooltip.module.css"); diff --git a/app/styles/ask-question-modal.module.css b/app/styles/ask-question-modal.module.css new file mode 100644 index 00000000..5ca01a57 --- /dev/null +++ b/app/styles/ask-question-modal.module.css @@ -0,0 +1,147 @@ +.ask-question-modal { + height: 21rem; + width: 31.25rem; + background-color: var(--color-white); + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + border-radius: 10px; + padding: 1.25rem; + max-width: 80svw; +} + +.ask-question-modal__heading { + text-align: center; + color: var(--color-pink); + font-size: 1.5rem; + text-decoration: underline; + font-style: normal; + font-weight: 800; + line-height: normal; +} + +.ask-question-modal__textarea { + width: 100%; + height: 47%; + outline: none; + border: 0; + resize: none; + margin-top: 1.25rem; + font-family: raleway, sans-serif; + font-size: 1.1rem; + font-style: normal; + font-weight: 500; + line-height: normal; + overflow-y: auto; +} + +.ask-question-modal__textarea::-webkit-scrollbar { + width: 5px; +} + +.ask-question-modal__textarea::-webkit-scrollbar-thumb { + background: var(--color-darkgrey); + border-radius: 10px; +} + +.ask-question-modal__checkbox-container { + display: flex; + gap: 0.5rem; +} + +.ask-question-modal__checkbox { + appearance: none; + background-color: var(--color-white); + margin: 0; + font: inherit; + color: currentcolor; + width: 1.15rem; + height: 1.15rem; + border: 0.15em solid var(--color-navyblue); + border-radius: 0.15em; + display: grid; + place-content: center; +} + +.ask-question-modal__checkbox::before { + content: ""; + width: 0.65rem; + height: 0.65rem; + transform: scale(0); + transition: 120ms transform ease-in-out; + box-shadow: inset 1em 1em var(--color-pink); + background-color: CanvasText; + transform-origin: bottom left; + clip-path: polygon(14% 44%, 0 65%, 50% 100%, 100% 16%, 80% 0%, 43% 62%); +} + +.ask-question-modal__checkbox:checked { + border: 2px solid var(--color-pink); +} + +.ask-question-modal__checkbox:checked::before { + transform: scale(1); +} + +.ask-question-modal__checkbox-label { + font-size: 1rem; + font-weight: 600; +} + +.ask-question-modal__max-characters-input { + width: 100%; + padding: 10px; + box-sizing: border-box; + font-size: 1.2rem; + border: 2px solid var(--color-navyblue); + border-radius: 10px; + margin-top: 10px; +} + +.ask-question-modal__max-characters-input:focus { + outline: none; + box-shadow: 0 0 0 4px var(--color-light-navyblue); +} + +.ask-question-modal__actions { + text-align: center; +} + +.ask-question-modal__cancel-button { + margin-right: 1rem; +} + +/* media queries */ +@media only screen and (width <=625px) { + .ask-question-modal { + height: 23rem; + padding: 1.25rem 1rem; + } + + .ask-question-modal__checkbox { + width: 1.2rem; + } +} + +@media only screen and (width <=425px) { + .ask-question-modal__checkbox { + width: 1.6rem; + } + + .ask-question-modal__submit-button, + .ask-question-modal__cancel-button { + margin-right: 0; + width: fit-content; + padding: 0 20px; + } +} + +/* utils css */ +.visibility--hidden { + visibility: hidden; +} + +.visibility--visible { + visibility: visible; +} diff --git a/app/styles/button.module.css b/app/styles/button.module.css index 520dafcb..f3a73117 100644 --- a/app/styles/button.module.css +++ b/app/styles/button.module.css @@ -32,6 +32,11 @@ background-color: var(--color-pink); } +.btn-pink:disabled { + opacity: 0.4; + color: var(--color-white); +} + .btn--sm { width: 8rem; } diff --git a/app/styles/icon-button.module.css b/app/styles/icon-button.module.css index 15d81d2b..f0697b8e 100644 --- a/app/styles/icon-button.module.css +++ b/app/styles/icon-button.module.css @@ -27,7 +27,36 @@ transform: translate(50%, 50%); } -@media (width <= 425px) { +.icon-button--sm { + border-radius: 50%; + height: 37px; + width: 38px; + margin-right: 2rem; + position: relative; +} + +.icon-button--sm .iconify.iconify--material-symbols { + height: 50%; + width: 50%; + position: absolute; + bottom: 50%; + right: 50%; + transform: translate(50%, 50%); +} + +.icon-button--green { + background-color: var(--color-green); + color: var(--color-white); + border: none; +} + +.icon-button--red-outlined { + border: 2px solid var(--text-red); + background-color: var(--color-white); + color: var(--text-red); +} + +@media (width <=425px) { .icon-button--md { margin-right: 1rem; } diff --git a/app/styles/input.module.css b/app/styles/input.module.css index 0982e719..6173a8b6 100644 --- a/app/styles/input.module.css +++ b/app/styles/input.module.css @@ -47,3 +47,41 @@ justify-content: space-between; overflow-y: scroll; } + +.user-input--error { + animation: error-animation 0.1s ease-in-out 3; + border: 1px solid red; +} + +.user-input--error.user-input:focus { + box-shadow: 0 0 0 4px var(--color-soft-magenta); +} + +.input-box .input--full-width { + width: 100%; +} + +.input-box__helper-text { + font-size: 0.8rem; + margin-left: 5px; + margin-top: 5px; + color: var(--color-lightgrey); +} + +.input-box__helper-text--error { + color: var(--text-red); +} + +@keyframes error-animation { + 0% { + transform: translateX(-4px); + } + + 50% { + transform: translateX(4px); + } + + 100% { + transform: translateX(-4px); + } +} diff --git a/app/styles/survey-page.module.css b/app/styles/survey-page.module.css new file mode 100644 index 00000000..08e9edfc --- /dev/null +++ b/app/styles/survey-page.module.css @@ -0,0 +1,43 @@ +.survey-page { + width: 100%; + margin-top: 3.1rem; + min-height: 80vh; + box-sizing: border-box; + padding: 0.6rem 1rem; + border-radius: 1rem; +} + +.survey-page__question-container { + display: flex; + align-items: center; + gap: 2rem; +} + +.survey-page__filter { + margin-bottom: 1rem; +} + +.survey-page__answers { + display: flex; + flex-direction: column; + gap: 10px; +} + +.survey-page__answers-heading { + border-radius: 10px; + background: var(--color-light-gray); + color: var(--color-pink); + font-size: 2.5rem; + padding: 0 10px; + line-height: 150%; + margin-bottom: 20px; + box-sizing: border-box; + height: 65px; +} + +@media (width <=768px) { + .survey-page__answers-heading { + font-size: 2rem; + line-height: 191%; + } +} diff --git a/app/styles/variables.css b/app/styles/variables.css index f1ef6444..93e1472e 100644 --- a/app/styles/variables.css +++ b/app/styles/variables.css @@ -23,6 +23,7 @@ --color-blue: #0d6efd; --color-navyblue: #1d1283; --color-green: #49a82e; + --color-green-low-opacity: #98fb98; --color-light-navyblue: #1d128340; --color-cultured: #f6f6f6; --color-sidebar: #dadada; @@ -33,4 +34,5 @@ --color-red: #ae03ba; --skeleton-bg: #483bc0; --skeleton-shine-color: #edeef1; + --color-yellow-low-opacity: #ffffe0; } diff --git a/app/styles/word-cloud.module.css b/app/styles/word-cloud.module.css new file mode 100644 index 00000000..9f4ec580 --- /dev/null +++ b/app/styles/word-cloud.module.css @@ -0,0 +1,4 @@ +.word-cloud { + min-height: 400px; + margin-top: 10px; +} diff --git a/app/templates/live.hbs b/app/templates/live.hbs index 38e94fe7..e961eb87 100644 --- a/app/templates/live.hbs +++ b/app/templates/live.hbs @@ -1,6 +1,6 @@ -{{page-title 'Live'}} +{{page-title "Live"}} -
+
{{#if this.liveService.isJoined}} {{/if}} - + @@ -33,7 +43,7 @@ @roomCodes={{this.liveService.roomCodesForMaven}} @closeModal={{modal.closeModal}} @newCode={{this.newRoomCode}} - @onInput={{fn this.inputHandler 'newRoomCode'}} + @onInput={{fn this.inputHandler "newRoomCode"}} @createRoomCode={{this.createRoomCodeHandler}} @toast={{this.liveService.toast}} @isLoading={{this.liveService.roomCodeLoading}} @@ -47,34 +57,33 @@ > - {{! TODO - add more else if statement instead of only else }} - {{#if (eq this.activeTab 'Screenshare')}} + {{#if (eq this.activeTab "Screenshare")}} {{#if this.liveService.isLoading}} -
- +
+
{{else}} {{#if this.liveService.isJoined}} -
+
{{! TODO - add skeleton to the whole structure }} {{#if this.isLoading}} {{else}} - + {{/if}}
-
+
{{#if this.isLoading}} {{else}} @@ -86,7 +95,7 @@ {{/if}}
-
+
{{#if this.isLoading}} {{else}} @@ -100,8 +109,9 @@ /> {{/if}}
+ {{else}} - {{#if (eq this.role '')}} + {{#if (eq this.role "")}} + {{else if (eq this.activeTab "Logs")}} +
+ {{else if (eq this.activeTab "Survey")}} + {{else}} - {{! TODO - add the respective component here }} -

Coming Soon!

+

Coming Soon!

{{/if}}
\ No newline at end of file diff --git a/app/utils/common-utils.js b/app/utils/common-utils.js new file mode 100644 index 00000000..44d4b1be --- /dev/null +++ b/app/utils/common-utils.js @@ -0,0 +1,11 @@ +export const readMoreFormatter = (string, lengthToDisplay) => { + if (!string) { + return; + } + + if (string.length > lengthToDisplay) { + return string.slice(0, lengthToDisplay) + '...'; + } else { + return string; + } +}; diff --git a/package.json b/package.json index 0ce7d021..f3330486 100644 --- a/package.json +++ b/package.json @@ -30,7 +30,10 @@ }, "dependencies": { "@100mslive/hms-video-store": "^0.10.6", + "d3-cloud": "^1.2.7", "dotenv": "^16.0.2", + "ember-d3": "^0.5.1", + "exists-sync": "^0.1.0", "fastboot-app-server": "^3.3.2" }, "devDependencies": { diff --git a/tests/integration/components/answer-reply-modal-test.js b/tests/integration/components/answer-reply-modal-test.js new file mode 100644 index 00000000..6e527fb3 --- /dev/null +++ b/tests/integration/components/answer-reply-modal-test.js @@ -0,0 +1,17 @@ +import { module, test } from 'qunit'; +import { setupRenderingTest } from 'website-www/tests/helpers'; + +module('Integration | Component | answer-reply-modal', function (hooks) { + setupRenderingTest(hooks); + + test('it renders', async function (assert) { + assert.ok(true); + // TODO - add tests + // Set any properties with this.set('myProperty', 'value'); + // Handle any actions with this.set('myAction', function(val) { ... }); + + // await render(hbs``); + + // assert.dom(this.element).hasText(''); + }); +}); diff --git a/tests/integration/components/ask-question-modal-test.js b/tests/integration/components/ask-question-modal-test.js new file mode 100644 index 00000000..d3d482c1 --- /dev/null +++ b/tests/integration/components/ask-question-modal-test.js @@ -0,0 +1,17 @@ +import { module, test } from 'qunit'; +import { setupRenderingTest } from 'website-www/tests/helpers'; + +module('Integration | Component | ask-question-modal', function (hooks) { + setupRenderingTest(hooks); + + test('it renders', async function (assert) { + assert.ok(true); + // TODO - add tests + // Set any properties with this.set('myProperty', 'value'); + // Handle any actions with this.set('myAction', function(val) { ... }); + + // await render(hbs``); + + // assert.dom(this.element).hasText(''); + }); +}); diff --git a/tests/integration/components/events/survey-page-test.js b/tests/integration/components/events/survey-page-test.js new file mode 100644 index 00000000..3697cd14 --- /dev/null +++ b/tests/integration/components/events/survey-page-test.js @@ -0,0 +1,14 @@ +import { module, test } from 'qunit'; +import { setupRenderingTest } from 'website-www/tests/helpers'; +import { render } from '@ember/test-helpers'; +import { hbs } from 'ember-cli-htmlbars'; + +module('Integration | Component | survey-page', function (hooks) { + setupRenderingTest(hooks); + + test('Events::SurveyPage renders', async function (assert) { + await render(hbs``); + + assert.ok(true, 'survey page tests'); + }); +}); diff --git a/tests/integration/components/word-cloud-test.js b/tests/integration/components/word-cloud-test.js new file mode 100644 index 00000000..8f21c2f1 --- /dev/null +++ b/tests/integration/components/word-cloud-test.js @@ -0,0 +1,17 @@ +import { module, test } from 'qunit'; +import { setupRenderingTest } from 'website-www/tests/helpers'; + +module('Integration | Component | word-cloud', function (hooks) { + setupRenderingTest(hooks); + + test('it renders', async function (assert) { + assert.ok(true); + // TODO - add tests + // Set any properties with this.set('myProperty', 'value'); + // Handle any actions with this.set('myAction', function(val) { ... }); + + // await render(hbs``); + + // assert.dom().hasText(''); + }); +}); diff --git a/yarn.lock b/yarn.lock index 0b1f6d8f..b4e3ddfc 100644 --- a/yarn.lock +++ b/yarn.lock @@ -5571,7 +5571,7 @@ broccoli-merge-trees@^2.0.0: broccoli-plugin "^1.3.0" merge-trees "^1.0.1" -broccoli-merge-trees@^3.0.1, broccoli-merge-trees@^3.0.2: +broccoli-merge-trees@^3.0.0, broccoli-merge-trees@^3.0.1, broccoli-merge-trees@^3.0.2: version "3.0.2" resolved "https://registry.yarnpkg.com/broccoli-merge-trees/-/broccoli-merge-trees-3.0.2.tgz#f33b451994225522b5c9bcf27d59decfd8ba537d" integrity sha512-ZyPAwrOdlCddduFbsMyyFzJUrvW6b04pMvDiAQZrCwghlvgowJDY+EfoXn+eR1RRA5nmGHJ+B68T63VnpRiT1A== @@ -6303,6 +6303,11 @@ combined-stream@^1.0.8: dependencies: delayed-stream "~1.0.0" +commander@2, commander@^2.20.0, commander@^2.6.0: + version "2.20.3" + resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" + integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== + commander@2.8.x: version "2.8.1" resolved "https://registry.yarnpkg.com/commander/-/commander-2.8.1.tgz#06be367febfda0c330aa1e2a072d3dc9762425d4" @@ -6320,11 +6325,6 @@ commander@^10.0.0: resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06" integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug== -commander@^2.20.0, commander@^2.6.0: - version "2.20.3" - resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" - integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== - commander@^4.1.1: version "4.1.1" resolved "https://registry.yarnpkg.com/commander/-/commander-4.1.1.tgz#9fd602bd936294e9e9ef46a3f4d6964044b18068" @@ -6668,6 +6668,269 @@ cssstyle@^2.3.0: dependencies: cssom "~0.3.6" +d3-array@1, d3-array@^1.1.1, d3-array@^1.2.0: + version "1.2.4" + resolved "https://registry.yarnpkg.com/d3-array/-/d3-array-1.2.4.tgz#635ce4d5eea759f6f605863dbcfc30edc737f71f" + integrity sha512-KHW6M86R+FUPYGb3R5XiYjXPq7VzwxZ22buHhAEVG5ztoEcZZMLov530mmccaqA1GghZArjQV46fuc8kUqhhHw== + +d3-axis@1: + version "1.0.12" + resolved "https://registry.yarnpkg.com/d3-axis/-/d3-axis-1.0.12.tgz#cdf20ba210cfbb43795af33756886fb3638daac9" + integrity sha512-ejINPfPSNdGFKEOAtnBtdkpr24c4d4jsei6Lg98mxf424ivoDP2956/5HDpIAtmHo85lqT4pruy+zEgvRUBqaQ== + +d3-brush@1: + version "1.1.6" + resolved "https://registry.yarnpkg.com/d3-brush/-/d3-brush-1.1.6.tgz#b0a22c7372cabec128bdddf9bddc058592f89e9b" + integrity sha512-7RW+w7HfMCPyZLifTz/UnJmI5kdkXtpCbombUSs8xniAyo0vIbrDzDwUJB6eJOgl9u5DQOt2TQlYumxzD1SvYA== + dependencies: + d3-dispatch "1" + d3-drag "1" + d3-interpolate "1" + d3-selection "1" + d3-transition "1" + +d3-chord@1: + version "1.0.6" + resolved "https://registry.yarnpkg.com/d3-chord/-/d3-chord-1.0.6.tgz#309157e3f2db2c752f0280fedd35f2067ccbb15f" + integrity sha512-JXA2Dro1Fxw9rJe33Uv+Ckr5IrAa74TlfDEhE/jfLOaXegMQFQTAgAw9WnZL8+HxVBRXaRGCkrNU7pJeylRIuA== + dependencies: + d3-array "1" + d3-path "1" + +d3-cloud@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/d3-cloud/-/d3-cloud-1.2.7.tgz#5a733c4bae43238cbb4760bb8f2d15912a8ad7a5" + integrity sha512-8TrgcgwRIpoZYQp7s3fGB7tATWfhckRb8KcVd1bOgqkNdkJRDGWfdSf4HkHHzZxSczwQJdSxvfPudwir5IAJ3w== + dependencies: + d3-dispatch "^1.0.3" + +d3-collection@1: + version "1.0.7" + resolved "https://registry.yarnpkg.com/d3-collection/-/d3-collection-1.0.7.tgz#349bd2aa9977db071091c13144d5e4f16b5b310e" + integrity sha512-ii0/r5f4sjKNTfh84Di+DpztYwqKhEyUlKoPrzUFfeSkWxjW49xU2QzO9qrPrNkpdI0XJkfzvmTu8V2Zylln6A== + +d3-color@1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/d3-color/-/d3-color-1.4.1.tgz#c52002bf8846ada4424d55d97982fef26eb3bc8a" + integrity sha512-p2sTHSLCJI2QKunbGb7ocOh7DgTAn8IrLx21QRc/BSnodXM4sv6aLQlnfpvehFMLZEfBc6g9pH9SWQccFYfJ9Q== + +d3-contour@1: + version "1.3.2" + resolved "https://registry.yarnpkg.com/d3-contour/-/d3-contour-1.3.2.tgz#652aacd500d2264cb3423cee10db69f6f59bead3" + integrity sha512-hoPp4K/rJCu0ladiH6zmJUEz6+u3lgR+GSm/QdM2BBvDraU39Vr7YdDCicJcxP1z8i9B/2dJLgDC1NcvlF8WCg== + dependencies: + d3-array "^1.1.1" + +d3-dispatch@1, d3-dispatch@^1.0.3: + version "1.0.6" + resolved "https://registry.yarnpkg.com/d3-dispatch/-/d3-dispatch-1.0.6.tgz#00d37bcee4dd8cd97729dd893a0ac29caaba5d58" + integrity sha512-fVjoElzjhCEy+Hbn8KygnmMS7Or0a9sI2UzGwoB7cCtvI1XpVN9GpoYlnb3xt2YV66oXYb1fLJ8GMvP4hdU1RA== + +d3-drag@1: + version "1.2.5" + resolved "https://registry.yarnpkg.com/d3-drag/-/d3-drag-1.2.5.tgz#2537f451acd39d31406677b7dc77c82f7d988f70" + integrity sha512-rD1ohlkKQwMZYkQlYVCrSFxsWPzI97+W+PaEIBNTMxRuxz9RF0Hi5nJWHGVJ3Om9d2fRTe1yOBINJyy/ahV95w== + dependencies: + d3-dispatch "1" + d3-selection "1" + +d3-dsv@1: + version "1.2.0" + resolved "https://registry.yarnpkg.com/d3-dsv/-/d3-dsv-1.2.0.tgz#9d5f75c3a5f8abd611f74d3f5847b0d4338b885c" + integrity sha512-9yVlqvZcSOMhCYzniHE7EVUws7Fa1zgw+/EAV2BxJoG3ME19V6BQFBwI855XQDsxyOuG7NibqRMTtiF/Qup46g== + dependencies: + commander "2" + iconv-lite "0.4" + rw "1" + +d3-ease@1: + version "1.0.7" + resolved "https://registry.yarnpkg.com/d3-ease/-/d3-ease-1.0.7.tgz#9a834890ef8b8ae8c558b2fe55bd57f5993b85e2" + integrity sha512-lx14ZPYkhNx0s/2HX5sLFUI3mbasHjSSpwO/KaaNACweVwxUruKyWVcb293wMv1RqTPZyZ8kSZ2NogUZNcLOFQ== + +d3-fetch@1: + version "1.2.0" + resolved "https://registry.yarnpkg.com/d3-fetch/-/d3-fetch-1.2.0.tgz#15ce2ecfc41b092b1db50abd2c552c2316cf7fc7" + integrity sha512-yC78NBVcd2zFAyR/HnUiBS7Lf6inSCoWcSxFfw8FYL7ydiqe80SazNwoffcqOfs95XaLo7yebsmQqDKSsXUtvA== + dependencies: + d3-dsv "1" + +d3-force@1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/d3-force/-/d3-force-1.2.1.tgz#fd29a5d1ff181c9e7f0669e4bd72bdb0e914ec0b" + integrity sha512-HHvehyaiUlVo5CxBJ0yF/xny4xoaxFxDnBXNvNcfW9adORGZfyNF1dj6DGLKyk4Yh3brP/1h3rnDzdIAwL08zg== + dependencies: + d3-collection "1" + d3-dispatch "1" + d3-quadtree "1" + d3-timer "1" + +d3-format@1: + version "1.4.5" + resolved "https://registry.yarnpkg.com/d3-format/-/d3-format-1.4.5.tgz#374f2ba1320e3717eb74a9356c67daee17a7edb4" + integrity sha512-J0piedu6Z8iB6TbIGfZgDzfXxUFN3qQRMofy2oPdXzQibYGqPB/9iMcxr/TGalU+2RsyDO+U4f33id8tbnSRMQ== + +d3-geo@1: + version "1.12.1" + resolved "https://registry.yarnpkg.com/d3-geo/-/d3-geo-1.12.1.tgz#7fc2ab7414b72e59fbcbd603e80d9adc029b035f" + integrity sha512-XG4d1c/UJSEX9NfU02KwBL6BYPj8YKHxgBEw5om2ZnTRSbIcego6dhHwcxuSR3clxh0EpE38os1DVPOmnYtTPg== + dependencies: + d3-array "1" + +d3-hierarchy@1: + version "1.1.9" + resolved "https://registry.yarnpkg.com/d3-hierarchy/-/d3-hierarchy-1.1.9.tgz#2f6bee24caaea43f8dc37545fa01628559647a83" + integrity sha512-j8tPxlqh1srJHAtxfvOUwKNYJkQuBFdM1+JAUfq6xqH5eAqf93L7oG1NVqDa4CpFZNvnNKtCYEUC8KY9yEn9lQ== + +d3-interpolate@1: + version "1.4.0" + resolved "https://registry.yarnpkg.com/d3-interpolate/-/d3-interpolate-1.4.0.tgz#526e79e2d80daa383f9e0c1c1c7dcc0f0583e987" + integrity sha512-V9znK0zc3jOPV4VD2zZn0sDhZU3WAE2bmlxdIwwQPPzPjvyLkd8B3JUVdS1IDUFDkWZ72c9qnv1GK2ZagTZ8EA== + dependencies: + d3-color "1" + +d3-path@1: + version "1.0.9" + resolved "https://registry.yarnpkg.com/d3-path/-/d3-path-1.0.9.tgz#48c050bb1fe8c262493a8caf5524e3e9591701cf" + integrity sha512-VLaYcn81dtHVTjEHd8B+pbe9yHWpXKZUC87PzoFmsFrJqgFwDe/qxfp5MlfsfM1V5E/iVt0MmEbWQ7FVIXh/bg== + +d3-polygon@1: + version "1.0.6" + resolved "https://registry.yarnpkg.com/d3-polygon/-/d3-polygon-1.0.6.tgz#0bf8cb8180a6dc107f518ddf7975e12abbfbd38e" + integrity sha512-k+RF7WvI08PC8reEoXa/w2nSg5AUMTi+peBD9cmFc+0ixHfbs4QmxxkarVal1IkVkgxVuk9JSHhJURHiyHKAuQ== + +d3-quadtree@1: + version "1.0.7" + resolved "https://registry.yarnpkg.com/d3-quadtree/-/d3-quadtree-1.0.7.tgz#ca8b84df7bb53763fe3c2f24bd435137f4e53135" + integrity sha512-RKPAeXnkC59IDGD0Wu5mANy0Q2V28L+fNe65pOCXVdVuTJS3WPKaJlFHer32Rbh9gIo9qMuJXio8ra4+YmIymA== + +d3-random@1: + version "1.1.2" + resolved "https://registry.yarnpkg.com/d3-random/-/d3-random-1.1.2.tgz#2833be7c124360bf9e2d3fd4f33847cfe6cab291" + integrity sha512-6AK5BNpIFqP+cx/sreKzNjWbwZQCSUatxq+pPRmFIQaWuoD+NrbVWw7YWpHiXpCQ/NanKdtGDuB+VQcZDaEmYQ== + +d3-scale-chromatic@1: + version "1.5.0" + resolved "https://registry.yarnpkg.com/d3-scale-chromatic/-/d3-scale-chromatic-1.5.0.tgz#54e333fc78212f439b14641fb55801dd81135a98" + integrity sha512-ACcL46DYImpRFMBcpk9HhtIyC7bTBR4fNOPxwVSl0LfulDAwyiHyPOTqcDG1+t5d4P9W7t/2NAuWu59aKko/cg== + dependencies: + d3-color "1" + d3-interpolate "1" + +d3-scale@2: + version "2.2.2" + resolved "https://registry.yarnpkg.com/d3-scale/-/d3-scale-2.2.2.tgz#4e880e0b2745acaaddd3ede26a9e908a9e17b81f" + integrity sha512-LbeEvGgIb8UMcAa0EATLNX0lelKWGYDQiPdHj+gLblGVhGLyNbaCn3EvrJf0A3Y/uOOU5aD6MTh5ZFCdEwGiCw== + dependencies: + d3-array "^1.2.0" + d3-collection "1" + d3-format "1" + d3-interpolate "1" + d3-time "1" + d3-time-format "2" + +d3-selection-multi@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/d3-selection-multi/-/d3-selection-multi-1.0.1.tgz#cd6c25413d04a2cb97470e786f2cd877f3e34f58" + integrity sha512-mEnRkJ6A+Otd1LuRPV3az+s/RLDmIEhuz/MnT21O2nPcWNH7wZotdIlRjMA4Wpr+n7AESQ5fD8v1J3nL2Mnw9g== + dependencies: + d3-selection "1" + d3-transition "1" + +d3-selection@1, d3-selection@^1.1.0: + version "1.4.2" + resolved "https://registry.yarnpkg.com/d3-selection/-/d3-selection-1.4.2.tgz#dcaa49522c0dbf32d6c1858afc26b6094555bc5c" + integrity sha512-SJ0BqYihzOjDnnlfyeHT0e30k0K1+5sR3d5fNueCNeuhZTnGw4M4o8mqJchSwgKMXCNFo+e2VTChiSJ0vYtXkg== + +d3-shape@1: + version "1.3.7" + resolved "https://registry.yarnpkg.com/d3-shape/-/d3-shape-1.3.7.tgz#df63801be07bc986bc54f63789b4fe502992b5d7" + integrity sha512-EUkvKjqPFUAZyOlhY5gzCxCeI0Aep04LwIRpsZ/mLFelJiUfnK56jo5JMDSE7yyP2kLSb6LtF+S5chMk7uqPqw== + dependencies: + d3-path "1" + +d3-time-format@2: + version "2.3.0" + resolved "https://registry.yarnpkg.com/d3-time-format/-/d3-time-format-2.3.0.tgz#107bdc028667788a8924ba040faf1fbccd5a7850" + integrity sha512-guv6b2H37s2Uq/GefleCDtbe0XZAuy7Wa49VGkPVPMfLL9qObgBST3lEHJBMUp8S7NdLQAGIvr2KXk8Hc98iKQ== + dependencies: + d3-time "1" + +d3-time@1: + version "1.1.0" + resolved "https://registry.yarnpkg.com/d3-time/-/d3-time-1.1.0.tgz#b1e19d307dae9c900b7e5b25ffc5dcc249a8a0f1" + integrity sha512-Xh0isrZ5rPYYdqhAVk8VLnMEidhz5aP7htAADH6MfzgmmicPkTo8LhkLxci61/lCB7n7UmE3bN0leRt+qvkLxA== + +d3-timer@1: + version "1.0.10" + resolved "https://registry.yarnpkg.com/d3-timer/-/d3-timer-1.0.10.tgz#dfe76b8a91748831b13b6d9c793ffbd508dd9de5" + integrity sha512-B1JDm0XDaQC+uvo4DT79H0XmBskgS3l6Ve+1SBCfxgmtIb1AVrPIoqd+nPSv+loMX8szQ0sVUhGngL7D5QPiXw== + +d3-transition@1: + version "1.3.2" + resolved "https://registry.yarnpkg.com/d3-transition/-/d3-transition-1.3.2.tgz#a98ef2151be8d8600543434c1ca80140ae23b398" + integrity sha512-sc0gRU4PFqZ47lPVHloMn9tlPcv8jxgOQg+0zjhfZXMQuvppjG6YuwdMBE0TuqCZjeJkLecku/l9R0JPcRhaDA== + dependencies: + d3-color "1" + d3-dispatch "1" + d3-ease "1" + d3-interpolate "1" + d3-selection "^1.1.0" + d3-timer "1" + +d3-voronoi@1: + version "1.1.4" + resolved "https://registry.yarnpkg.com/d3-voronoi/-/d3-voronoi-1.1.4.tgz#dd3c78d7653d2bb359284ae478645d95944c8297" + integrity sha512-dArJ32hchFsrQ8uMiTBLq256MpnZjeuBtdHpaDlYuQyjU0CVzCJl/BVW+SkszaAeH95D/8gxqAhgx0ouAWAfRg== + +d3-zoom@1: + version "1.8.3" + resolved "https://registry.yarnpkg.com/d3-zoom/-/d3-zoom-1.8.3.tgz#b6a3dbe738c7763121cd05b8a7795ffe17f4fc0a" + integrity sha512-VoLXTK4wvy1a0JpH2Il+F2CiOhVu7VRXWF5M/LroMIh3/zBAC3WAt7QoIvPibOavVo20hN6/37vwAsdBejLyKQ== + dependencies: + d3-dispatch "1" + d3-drag "1" + d3-interpolate "1" + d3-selection "1" + d3-transition "1" + +d3@^5.0.0: + version "5.16.0" + resolved "https://registry.yarnpkg.com/d3/-/d3-5.16.0.tgz#9c5e8d3b56403c79d4ed42fbd62f6113f199c877" + integrity sha512-4PL5hHaHwX4m7Zr1UapXW23apo6pexCgdetdJ5kTmADpG/7T9Gkxw0M0tf/pjoB63ezCCm0u5UaFYy2aMt0Mcw== + dependencies: + d3-array "1" + d3-axis "1" + d3-brush "1" + d3-chord "1" + d3-collection "1" + d3-color "1" + d3-contour "1" + d3-dispatch "1" + d3-drag "1" + d3-dsv "1" + d3-ease "1" + d3-fetch "1" + d3-force "1" + d3-format "1" + d3-geo "1" + d3-hierarchy "1" + d3-interpolate "1" + d3-path "1" + d3-polygon "1" + d3-quadtree "1" + d3-random "1" + d3-scale "2" + d3-scale-chromatic "1" + d3-selection "1" + d3-shape "1" + d3-time "1" + d3-time-format "2" + d3-timer "1" + d3-transition "1" + d3-voronoi "1" + d3-zoom "1" + dag-map@^2.0.2: version "2.0.2" resolved "https://registry.yarnpkg.com/dag-map/-/dag-map-2.0.2.tgz#9714b472de82a1843de2fba9b6876938cab44c68" @@ -7128,7 +7391,7 @@ ember-cli-babel-plugin-helpers@^1.0.0, ember-cli-babel-plugin-helpers@^1.1.1: resolved "https://registry.yarnpkg.com/ember-cli-babel-plugin-helpers/-/ember-cli-babel-plugin-helpers-1.1.1.tgz#5016b80cdef37036c4282eef2d863e1d73576879" integrity sha512-sKvOiPNHr5F/60NLd7SFzMpYPte/nnGkq/tMIfXejfKHIhaiIkYFqX8Z9UFTKWLLn+V7NOaby6niNPZUdvKCRw== -ember-cli-babel@^7.10.0, ember-cli-babel@^7.13.0, ember-cli-babel@^7.18.0, ember-cli-babel@^7.22.1, ember-cli-babel@^7.23.0, ember-cli-babel@^7.23.1, ember-cli-babel@^7.26.11, ember-cli-babel@^7.26.3, ember-cli-babel@^7.26.5, ember-cli-babel@^7.26.6, ember-cli-babel@^7.7.3: +ember-cli-babel@^7.1.2, ember-cli-babel@^7.10.0, ember-cli-babel@^7.13.0, ember-cli-babel@^7.18.0, ember-cli-babel@^7.22.1, ember-cli-babel@^7.23.0, ember-cli-babel@^7.23.1, ember-cli-babel@^7.26.11, ember-cli-babel@^7.26.3, ember-cli-babel@^7.26.5, ember-cli-babel@^7.26.6, ember-cli-babel@^7.7.3: version "7.26.11" resolved "https://registry.yarnpkg.com/ember-cli-babel/-/ember-cli-babel-7.26.11.tgz#50da0fe4dcd99aada499843940fec75076249a9f" integrity sha512-JJYeYjiz/JTn34q7F5DSOjkkZqy8qwFOOxXfE6pe9yEJqWGu4qErKxlz8I22JoVEQ/aBUO+OcKTpmctvykM9YA== @@ -7666,6 +7929,17 @@ ember-composable-helpers@^5.0.0: ember-cli-babel "^7.26.3" resolve "^1.10.0" +ember-d3@^0.5.1: + version "0.5.1" + resolved "https://registry.yarnpkg.com/ember-d3/-/ember-d3-0.5.1.tgz#b23ce145863f082b5e73d25d9a43a0f1d9e9f412" + integrity sha512-NyjTUuIOxGxZdyrxLasNwwjqyFgay1pVHGRAWFj7mriwTI44muKsM9ZMl6YeepqixceuFig2fDxHmLLrkQV+QQ== + dependencies: + broccoli-funnel "^2.0.0" + broccoli-merge-trees "^3.0.0" + d3 "^5.0.0" + d3-selection-multi "^1.0.1" + ember-cli-babel "^7.1.2" + ember-data@~5.3.0: version "5.3.0" resolved "https://registry.yarnpkg.com/ember-data/-/ember-data-5.3.0.tgz#d7be6b77653a41ae8ed045ffb904f1adbdcb8920" @@ -8469,6 +8743,11 @@ execa@^7.1.1: signal-exit "^3.0.7" strip-final-newline "^3.0.0" +exists-sync@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/exists-sync/-/exists-sync-0.1.0.tgz#318d545213d2b2a31499e92c35f74c94196a22f7" + integrity sha512-qEfFekfBVid4b14FNug/RNY1nv+BADnlzKGHulc+t6ZLqGY4kdHGh1iFha8lnE3sJU/1WzMzKRNxS6EvSakJUg== + exit@^0.1.2: version "0.1.2" resolved "https://registry.yarnpkg.com/exit/-/exit-0.1.2.tgz#0632638f8d877cc82107d30a0fff1a17cba1cd0c" @@ -9757,7 +10036,7 @@ human-signals@^4.3.0: resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-4.3.1.tgz#ab7f811e851fca97ffbd2c1fe9a958964de321b2" integrity sha512-nZXjEF2nbo7lIw3mgYjItAfgQXog3OjJogSbKa2CQIIvSGWcKgeJnQlNXip6NglNzYH45nSRiEVimMvYL8DDqQ== -iconv-lite@0.4.24, iconv-lite@^0.4.24: +iconv-lite@0.4, iconv-lite@0.4.24, iconv-lite@^0.4.24: version "0.4.24" resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== @@ -12790,6 +13069,11 @@ run-parallel@^1.1.9: dependencies: queue-microtask "^1.2.2" +rw@1: + version "1.3.3" + resolved "https://registry.yarnpkg.com/rw/-/rw-1.3.3.tgz#3f862dfa91ab766b14885ef4d01124bfda074fb4" + integrity sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ== + rxjs@^6.4.0, rxjs@^6.6.0: version "6.6.7" resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.6.7.tgz#90ac018acabf491bf65044235d5863c4dab804c9" From 651cb499a06d08e83ed3f9cbc5945971923153c1 Mon Sep 17 00:00:00 2001 From: Vinit khandal <111434418+vinit717@users.noreply.github.com> Date: Thu, 18 Jan 2024 23:30:11 +0530 Subject: [PATCH 11/12] Refactor identity steps UI (#831) * refactor identity steps * refactor identity steps * refactor test and toast * change classname * refactor BEM --------- Co-authored-by: Satyam Bajpai --- app/components/identity-steps/step-five.hbs | 76 ++------ app/components/identity-steps/step-five.js | 52 ++---- app/components/identity-steps/step-four.hbs | 88 +++------ app/components/identity-steps/step-four.js | 61 +++++-- app/components/identity-steps/step-one.hbs | 14 +- app/components/identity-steps/step-seven.hbs | 86 --------- app/components/identity-steps/step-seven.js | 28 --- app/components/identity-steps/step-six.hbs | 112 +++++++++--- app/components/identity-steps/step-six.js | 66 +++---- app/components/identity-steps/step-three.hbs | 82 ++++++++- app/components/identity-steps/step-three.js | 27 +++ app/components/identity-steps/step-two.hbs | 21 ++- app/components/stepper-signup.hbs | 12 +- app/controllers/join.js | 12 +- app/styles/identity-card.module.css | 98 +++++++++- app/styles/onboarding-card.module.css | 100 +---------- app/styles/tooltip.module.css | 18 +- .../identity-steps/step-five-test.js | 118 ++----------- .../identity-steps/step-four-test.js | 145 +++++++-------- .../identity-steps/step-one-test.js | 4 +- .../identity-steps/step-seven-test.js | 167 ------------------ .../identity-steps/step-six-test.js | 164 +++++++++++++---- .../identity-steps/step-three-test.js | 139 +++++++++++++-- .../identity-steps/step-two-test.js | 18 +- 24 files changed, 808 insertions(+), 900 deletions(-) delete mode 100644 app/components/identity-steps/step-seven.hbs delete mode 100644 app/components/identity-steps/step-seven.js create mode 100644 app/components/identity-steps/step-three.js delete mode 100644 tests/integration/components/identity-steps/step-seven-test.js diff --git a/app/components/identity-steps/step-five.hbs b/app/components/identity-steps/step-five.hbs index 76e4f764..49f48472 100644 --- a/app/components/identity-steps/step-five.hbs +++ b/app/components/identity-steps/step-five.hbs @@ -1,66 +1,28 @@ -
-

- Deploy Profile Service +
+

+ Link Profile Service

-

- Set the chaincode on your profile service.
- Deploy it and enter your profile service URL +

+ Ensure that you have deployed your profile service, +
+ Click on link button to start the linking process for joining RealDevSquad.

-
- - - -
-
+
diff --git a/app/components/identity-steps/step-five.js b/app/components/identity-steps/step-five.js index b6bb02a0..074a443e 100644 --- a/app/components/identity-steps/step-five.js +++ b/app/components/identity-steps/step-five.js @@ -1,42 +1,21 @@ import Component from '@glimmer/component'; -import { tracked } from '@glimmer/tracking'; import { action } from '@ember/object'; +import { tracked } from '@glimmer/tracking'; import { inject as service } from '@ember/service'; -import { toastNotificationTimeoutOptions } from '../../constants/toast-notification'; import { APPS } from '../../constants/urls'; -import checkURL from '../../utils/check-url'; +import { TOAST_OPTIONS } from '../../constants/toast-options'; export default class IdentityStepsStepFiveComponent extends Component { @service toast; - @tracked isMouseOnTooltip = false; - @tracked profileURL = ''; - @tracked nextButtonDisabled = true; @tracked isLoading = false; - @action openTooltipInfo() { - this.isMouseOnTooltip = true; - } - - @action closeTooltipInfo() { - this.isMouseOnTooltip = false; - } - - @action changeProfileURL(e) { - this.profileURL = e.target.value; - if (this.profileURL === '' || !checkURL(this.profileURL)) { - this.nextButtonDisabled = true; - } else { - this.nextButtonDisabled = false; - } - } - - @action async handleEdit(e) { + @action async handleVerify(e) { e.preventDefault(); this.isLoading = true; + try { - const response = await fetch(`${APPS.API_BACKEND}/users/profileURL`, { - method: 'PATCH', - body: JSON.stringify({ profileURL: this.profileURL }), + const response = await fetch(`${APPS.API_BACKEND}/users/verify`, { + method: 'POST', headers: { 'Content-Type': 'application/json', }, @@ -44,27 +23,22 @@ export default class IdentityStepsStepFiveComponent extends Component { }); if (response.ok) { this.toast.info( - 'Updated profile URL!!', - '', - toastNotificationTimeoutOptions, + 'Your request has been queued successfully', + 'Info', + TOAST_OPTIONS, ); + this.args.startHandler(); } else { - this.toast.error( - 'Something went wrong. Please check console errors.', - '', - toastNotificationTimeoutOptions, - ); + throw new Error('Your request has not been queued'); } } catch (error) { - console.error(error); this.toast.error( 'Something went wrong. Please check console errors.', - '', - toastNotificationTimeoutOptions, + 'Error', + TOAST_OPTIONS, ); } finally { this.isLoading = false; - this.args.startHandler(); } } } diff --git a/app/components/identity-steps/step-four.hbs b/app/components/identity-steps/step-four.hbs index 03e601d9..f53d4e06 100644 --- a/app/components/identity-steps/step-four.hbs +++ b/app/components/identity-steps/step-four.hbs @@ -1,74 +1,36 @@ -
-

- Chaincode Generation +
+

+ Deploy Profile Service

-

- A private key that you need to use in your profile service URL and deploy - for source that you're the source of the URL +

+ Set the chaincode on your profile service.
+ Deploy it and enter your profile service URL

- {{#if @isChaincodeClicked}} -
-
- {{#if this.hideChaincode}} - ******************** - {{else}} - {{@chaincode}} - {{/if}} -
-
- - -
-
- {{else}} - + + - {{/if}} -
+
+ +
\ No newline at end of file diff --git a/app/components/identity-steps/step-four.js b/app/components/identity-steps/step-four.js index 7ba8f7f7..88f7d200 100644 --- a/app/components/identity-steps/step-four.js +++ b/app/components/identity-steps/step-four.js @@ -2,26 +2,61 @@ import Component from '@glimmer/component'; import { tracked } from '@glimmer/tracking'; import { action } from '@ember/object'; import { inject as service } from '@ember/service'; +import { APPS } from '../../constants/urls'; +import checkURL from '../../utils/check-url'; import { TOAST_OPTIONS } from '../../constants/toast-options'; export default class IdentityStepsStepFourComponent extends Component { @service toast; - @tracked hideChaincode = true; - @tracked isCopyClicked = false; - @tracked isChaincodePageButtonDisabled = true; - @tracked isTooltipVisible = true; + @tracked isMouseOnTooltip = false; + @tracked profileURL = ''; + @tracked nextButtonDisabled = true; + @tracked isLoading = false; - @action toggleEye() { - this.hideChaincode = !this.hideChaincode; + @action openTooltipInfo() { + this.isMouseOnTooltip = true; } - @action handleCopy() { - navigator.clipboard.writeText(this.Chaincode); - this.isCopyClicked = true; - this.isTooltipVisible = false; - this.isChaincodePageButtonDisabled = false; - if (this.isCopyClicked === true) { - this.toast.info('Copied', '', TOAST_OPTIONS); + @action closeTooltipInfo() { + this.isMouseOnTooltip = false; + } + + @action changeProfileURL(e) { + this.profileURL = e.target.value; + if (this.profileURL === '' || !checkURL(this.profileURL)) { + this.nextButtonDisabled = true; + } else { + this.nextButtonDisabled = false; + } + } + + @action async handleEdit(e) { + e.preventDefault(); + this.isLoading = true; + try { + const response = await fetch(`${APPS.API_BACKEND}/users/profileURL`, { + method: 'PATCH', + body: JSON.stringify({ profileURL: this.profileURL }), + headers: { + 'Content-Type': 'application/json', + }, + credentials: 'include', + }); + if (response.ok) { + this.toast.info('Profile Service Updated!!', 'Info', TOAST_OPTIONS); + this.args.startHandler(); + } else { + throw new Error('Profile update failed'); + } + } catch (error) { + console.error(error); + this.toast.error( + 'Something went wrong. Please check console errors.', + 'Error', + TOAST_OPTIONS, + ); + } finally { + this.isLoading = false; } } } diff --git a/app/components/identity-steps/step-one.hbs b/app/components/identity-steps/step-one.hbs index de36f6b6..5c36a7e9 100644 --- a/app/components/identity-steps/step-one.hbs +++ b/app/components/identity-steps/step-one.hbs @@ -1,11 +1,13 @@

Challenge Time !!

-

Thank you for providing all the details - Before joining the community we would want +

+ Thank you for providing all the details. + Before joining the community, we would want you to complete a small challenge which will also help you in setting up your - identity across Real Dev Squad + identity across Real Dev Squad.

-

Please click proceed to know about - the task +

+ Please click proceed to know about + the task.

- \ No newline at end of file + diff --git a/app/components/identity-steps/step-seven.hbs b/app/components/identity-steps/step-seven.hbs deleted file mode 100644 index c41451aa..00000000 --- a/app/components/identity-steps/step-seven.hbs +++ /dev/null @@ -1,86 +0,0 @@ -
- {{#each this.profileStatuses as |currentStatus|}} - {{#if (eq this.login.userData.profileStatus currentStatus.status)}} -
-

- {{currentStatus.heading}} -

- -
-
- {{#if - (eq this.login.userData.profileStatus this.PROFILE_STATUS.PENDING) - }} -

- Refresh to Check Verification Status -

-

- Your Profile Service Linked with Real Dev Squad Service -

- {{else if (eq - this.login.userData.profileStatus this.PROFILE_STATUS.BLOCKED - )}} -

- Your previous - Chaincode - is Blocked. -

- {{else if (eq - this.login.userData.profileStatus this.PROFILE_STATUS.VERIFIED - )}} -

- Congratulations! - Your Profile Service is Verified. -

-

- Take the Next Step and Join Our Discord Server. -

- {{/if}} -
- {{/if}} - {{/each}} -
- {{#if (eq this.login.userData.profileStatus this.PROFILE_STATUS.PENDING)}} - - {{else if (eq - this.login.userData.profileStatus this.PROFILE_STATUS.BLOCKED - )}} - - {{else if (eq - this.login.userData.profileStatus this.PROFILE_STATUS.VERIFIED - )}} - - {{/if}} -
-
\ No newline at end of file diff --git a/app/components/identity-steps/step-seven.js b/app/components/identity-steps/step-seven.js deleted file mode 100644 index c16cd2e2..00000000 --- a/app/components/identity-steps/step-seven.js +++ /dev/null @@ -1,28 +0,0 @@ -import Component from '@glimmer/component'; -import { inject as service } from '@ember/service'; -import { PROFILE_STATUS } from '../../constants/stepper-signup-data'; - -export default class IdentityStepsStepSevenComponent extends Component { - @service login; - @service router; - - PROFILE_STATUS = PROFILE_STATUS; - - profileStatuses = [ - { - status: PROFILE_STATUS.PENDING, - heading: 'Pending', - icon: 'hourglass-half', - }, - { - status: PROFILE_STATUS.BLOCKED, - heading: 'Blocked', - icon: 'ban', - }, - { - status: PROFILE_STATUS.VERIFIED, - heading: 'Successful', - icon: 'square-check', - }, - ]; -} diff --git a/app/components/identity-steps/step-six.hbs b/app/components/identity-steps/step-six.hbs index 49f48472..0659e0da 100644 --- a/app/components/identity-steps/step-six.hbs +++ b/app/components/identity-steps/step-six.hbs @@ -1,30 +1,86 @@ -
-

- Link Profile Service -

-

- Ensure that you have deployed your profile service, -
- Click on link button to start the linking process for joining RealDevSquad. -

- -
- +
+ {{#each this.profileStatuses as |currentStatus|}} + {{#if (eq this.activeProfileStatus currentStatus.status)}} +
+

+ {{currentStatus.heading}} +

+ +
+
+ {{#if + (eq this.activeProfileStatus this.PROFILE_STATUS.PENDING) + }} +

+ Refresh to Check Verification Status +

+

+ Your Profile Service Linked with Real Dev Squad Service +

+ {{else if (eq + this.activeProfileStatus this.PROFILE_STATUS.BLOCKED + )}} +

+ Your previous + Chaincode + is Blocked. +

+ {{else if (eq + this.activeProfileStatus this.PROFILE_STATUS.VERIFIED + )}} +

+ Congratulations! + Your Profile Service is Verified. +

+

+ Take the Next Step and Join Our Discord Server. +

+ {{/if}} +
+ {{/if}} + {{/each}} +
+ {{#if (eq this.activeProfileStatus this.PROFILE_STATUS.PENDING)}} + + {{else if (eq + this.activeProfileStatus this.PROFILE_STATUS.BLOCKED + )}} + + {{else if (eq + this.activeProfileStatus this.PROFILE_STATUS.VERIFIED + )}} + + {{/if}}
\ No newline at end of file diff --git a/app/components/identity-steps/step-six.js b/app/components/identity-steps/step-six.js index cae5e5a7..6399012d 100644 --- a/app/components/identity-steps/step-six.js +++ b/app/components/identity-steps/step-six.js @@ -1,49 +1,33 @@ import Component from '@glimmer/component'; -import { action } from '@ember/object'; -import { tracked } from '@glimmer/tracking'; import { inject as service } from '@ember/service'; -import { toastNotificationTimeoutOptions } from '../../constants/toast-notification'; -import { APPS } from '../../constants/urls'; +import { PROFILE_STATUS } from '../../constants/stepper-signup-data'; export default class IdentityStepsStepSixComponent extends Component { - @service toast; - @tracked isLoading = false; + @service login; + @service router; - @action async handleVerify(e) { - e.preventDefault(); - this.isLoading = true; + PROFILE_STATUS = PROFILE_STATUS; - try { - const response = await fetch(`${APPS.API_BACKEND}/users/verify`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - }, - credentials: 'include', - }); - - if (response.ok) { - this.toast.info( - 'Your request has been queued successfully', - '', - toastNotificationTimeoutOptions, - ); - } else { - this.toast.error( - 'Something went wrong. Please check console errors.', - '', - toastNotificationTimeoutOptions, - ); - } - } catch (error) { - this.toast.error( - 'Something went wrong. Please check console errors.', - '', - toastNotificationTimeoutOptions, - ); - } finally { - this.isLoading = false; - this.args.startHandler(); - } + get activeProfileStatus() { + const activeProfileStatus = this.login.userData.profileStatus; + return activeProfileStatus; } + + profileStatuses = [ + { + status: PROFILE_STATUS.PENDING, + heading: 'Pending', + icon: 'hourglass-half', + }, + { + status: PROFILE_STATUS.BLOCKED, + heading: 'Blocked', + icon: 'ban', + }, + { + status: PROFILE_STATUS.VERIFIED, + heading: 'Successful', + icon: 'square-check', + }, + ]; } diff --git a/app/components/identity-steps/step-three.hbs b/app/components/identity-steps/step-three.hbs index f55f14a7..b6444346 100644 --- a/app/components/identity-steps/step-three.hbs +++ b/app/components/identity-steps/step-three.hbs @@ -1,9 +1,73 @@ -

Chaincode Generation

-

A private that you need to use in your - profile service URL and deploy for source - that you're the source of the URL -

-

https://github.com/identity-service/instructions.md -

- +
+

+ Chaincode Generation +

+

+ A private key that you need to use in your profile service URL and deploy + for source that you're the source of the URL +

+ {{#if @isChaincodeClicked}} +
+
+ {{#if this.hideChaincode}} + ******************** + {{else}} + {{@chaincode}} + {{/if}} +
+
+ + +
+
+ {{else}} + + {{/if}} +
+ +
+
\ No newline at end of file diff --git a/app/components/identity-steps/step-three.js b/app/components/identity-steps/step-three.js new file mode 100644 index 00000000..f0b91f13 --- /dev/null +++ b/app/components/identity-steps/step-three.js @@ -0,0 +1,27 @@ +import Component from '@glimmer/component'; +import { tracked } from '@glimmer/tracking'; +import { action } from '@ember/object'; +import { inject as service } from '@ember/service'; +import { TOAST_OPTIONS } from '../../constants/toast-options'; + +export default class IdentityStepsStepThreeComponent extends Component { + @service toast; + @tracked hideChaincode = true; + @tracked isCopyClicked = false; + @tracked isChaincodePageButtonDisabled = true; + @tracked isTooltipVisible = true; + + @action toggleEye() { + this.hideChaincode = !this.hideChaincode; + } + + @action handleCopy() { + navigator.clipboard.writeText(this.args.chaincode); + this.isCopyClicked = true; + this.isTooltipVisible = false; + this.isChaincodePageButtonDisabled = false; + if (this.isCopyClicked === true) { + this.toast.info('Copied', '', TOAST_OPTIONS); + } + } +} diff --git a/app/components/identity-steps/step-two.hbs b/app/components/identity-steps/step-two.hbs index 3a09259b..45d13614 100644 --- a/app/components/identity-steps/step-two.hbs +++ b/app/components/identity-steps/step-two.hbs @@ -1,8 +1,13 @@ -

Challenge Time !!

-

To add/update your profile details, link - your profile service with Real Dev Squad service -

-

https://github.com/identity-service/instructions.md -

- +

Challenge Time !!

+

+ To add/update your profile details, link your profile service with Real Dev Squad service +

+ + https://github.com/Real-Dev-Squad/sample-profile-service + +

+ After exploring the sample profile service, proceed to the next step by clicking the button below. +

+ diff --git a/app/components/stepper-signup.hbs b/app/components/stepper-signup.hbs index 82b97038..de16fd89 100644 --- a/app/components/stepper-signup.hbs +++ b/app/components/stepper-signup.hbs @@ -66,26 +66,24 @@ {{else if (eq this.currentStep 8)}} {{else if (eq this.currentStep 9)}} - - {{else if (eq this.currentStep 10)}} - + {{else if (eq this.currentStep 10)}} + {{else if (eq this.currentStep 11)}} {{else if (eq this.currentStep 12)}} - - {{else if (eq this.currentStep 13)}} - - {{else if (eq this.currentStep 14)}} + {{else if (eq this.currentStep 13)}} {}); await render( hbs``, ); - assert.dom('[data-test=profile-service]').exists(); - assert.dom('[data-test=profile-service]').hasClass('profile-service-page'); + assert.dom('[data-test=profile-service-url-Linking]').exists(); + assert + .dom('[data-test=profile-service-url-Linking]') + .hasClass('profile-service-url-linking-page'); }); - test('render heading on profile service page', async function (assert) { + test('render heading on profile service url Linking page', async function (assert) { assert.expect(2); this.set('startHandler', () => {}); await render( hbs``, ); - assert.dom('[data-test=heading]').hasClass('profile-service-page__heading'); - assert.dom('[data-test=heading]').hasText('Deploy Profile Service'); + assert + .dom('[data-test=heading]') + .hasClass('profile-service-url-linking-page__heading'); + assert.dom('[data-test=heading]').hasText('Link Profile Service'); }); - test('render description on profile service page', async function (assert) { + test('render description on profile service url Linking page', async function (assert) { assert.expect(2); this.set('startHandler', () => {}); await render( @@ -35,69 +39,15 @@ module('Integration | Component | identity-steps/step-five', function (hooks) { ); assert .dom('[data-test=description]') - .hasClass('profile-service-page__description'); + .hasClass('profile-service-url-linking-page__description'); assert .dom('[data-test=description]') .hasText( - 'Set the chaincode on your profile service. Deploy it and enter your profile service URL', + 'Ensure that you have deployed your profile service, Click on link button to start the linking process for joining RealDevSquad.', ); }); - test('render input field on profile service page', async function (assert) { - assert.expect(5); - - this.set('startHandler', () => {}); - - await render( - hbs``, - ); - - assert - .dom('[data-test-input-field=profile-service]') - .hasClass('profile-service-page__inputContainer__input-url'); - assert - .dom('[data-test-input-field=profile-service]') - .hasAttribute('id', 'profile-service-url'); - assert - .dom('[data-test-input-field=profile-service]') - .hasProperty('type', 'text'); - assert - .dom('[data-test-input-field=profile-service]') - .hasProperty('value', ''); - assert - .dom('[data-test-input-field=profile-service]') - .hasProperty('placeholder', 'Enter your profile service URL'); - }); - - test('Display Tooltip Information on Mouse Hover', async function (assert) { - assert.expect(1); - - this.set('startHandler', () => {}); - - await render( - hbs``, - ); - - await triggerEvent('[data-test=tooltip]', 'mouseover'); - - assert.dom('[data-test=tooltip-info]').hasClass('tooltip-info'); - }); - - test('Not Display Tooltip Information on Mouse Out', async function (assert) { - assert.expect(1); - - this.set('startHandler', () => {}); - - await render( - hbs``, - ); - - await triggerEvent('[data-test=tooltip]', 'mouseout'); - - assert.dom('[data-test=tooltip-info]').hasClass('tooltip-info'); - }); - - test('Render Next button on profile page', async function (assert) { + test('Render Link button on profile service url Linking page', async function (assert) { assert.expect(3); this.set('startHandler', () => {}); @@ -107,41 +57,7 @@ module('Integration | Component | identity-steps/step-five', function (hooks) { ); assert.dom('[data-test-button=next]').exists(); - assert.dom('[data-test-button=next]').hasText('Next'); + assert.dom('[data-test-button=next]').hasText('Link'); assert.dom('[data-test-button=next]').hasProperty('type', 'button'); }); - - test("Ensure the 'Next' Button is Enabled Only When a Valid Profile URL is Entered", async function (assert) { - assert.expect(1); - - this.set('startHandler', () => {}); - - await render( - hbs``, - ); - - await typeIn( - '[data-test-input-field=profile-service]', - 'https://rds-profile-service.onrender.com', - ); - - assert.dom('[data-test-button=next]').hasProperty('disabled', false); - }); - - test("Ensure the 'Next' Button is Disabled When a InValid Profile URL is Entered", async function (assert) { - assert.expect(1); - - this.set('startHandler', () => {}); - - await render( - hbs``, - ); - - await typeIn( - '[data-test-input-field=profile-service]', - 'rds-profile-service.onrender.com', - ); - - assert.dom('[data-test-button=next]').hasProperty('disabled', true); - }); }); diff --git a/tests/integration/components/identity-steps/step-four-test.js b/tests/integration/components/identity-steps/step-four-test.js index 5274cde1..7b228a20 100644 --- a/tests/integration/components/identity-steps/step-four-test.js +++ b/tests/integration/components/identity-steps/step-four-test.js @@ -1,140 +1,119 @@ import { module, test } from 'qunit'; import { setupRenderingTest } from 'website-www/tests/helpers'; -import { render, click } from '@ember/test-helpers'; +import { render, typeIn } from '@ember/test-helpers'; import { hbs } from 'ember-cli-htmlbars'; -module('Integration | Component | identity-steps/step-four', function (hooks) { +module('Integration | Component | identity-steps/step-five', function (hooks) { setupRenderingTest(hooks); - test('render main container div on generate chaincode page', async function (assert) { + test('render main container div on profile service page', async function (assert) { assert.expect(2); this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); await render( - hbs``, + hbs``, ); - assert.dom('[data-test=chaincode]').exists(); - assert.dom('[data-test=chaincode]').hasClass('chaincode-page'); + assert.dom('[data-test=profile-service]').exists(); + assert.dom('[data-test=profile-service]').hasClass('profile-service-page'); }); - test('render heading on generate chaincode page', async function (assert) { + test('render heading on profile service page', async function (assert) { assert.expect(2); this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); await render( - hbs``, + hbs``, ); - assert.dom('[data-test=heading]').hasClass('chaincode-page__heading'); - assert.dom('[data-test=heading]').hasText('Chaincode Generation'); + assert.dom('[data-test=heading]').hasClass('profile-service-page__heading'); + assert.dom('[data-test=heading]').hasText('Deploy Profile Service'); }); - test('render description on generate chaincode page', async function (assert) { + test('render description on profile service page', async function (assert) { assert.expect(2); this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); await render( - hbs``, + hbs``, ); assert .dom('[data-test=description]') - .hasClass('chaincode-page__description'); + .hasClass('profile-service-page__description'); assert .dom('[data-test=description]') .hasText( - "A private key that you need to use in your profile service URL and deploy for source that you're the source of the URL", + 'Set the chaincode on your profile service. Deploy it and enter your profile service URL', ); }); - test('render Generation Chaincode button on chaincode page', async function (assert) { - assert.expect(2); + test('render input field on profile service page', async function (assert) { + assert.expect(5); + this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); - this.set('chaincode', 'Generate Chaincode'); - this.set('isChaincodeClicked', false); + await render( - hbs``, + hbs``, ); - assert.dom('[data-test-button=chaincode]').hasText('Generate Chaincode'); - assert.dom('[data-test-button=chaincode]').hasProperty('type', 'button'); + + assert + .dom('[data-test-input-field=profile-service]') + .hasClass('profile-service-page__input-url'); + assert + .dom('[data-test-input-field=profile-service]') + .hasAttribute('id', 'profile-service-url'); + assert + .dom('[data-test-input-field=profile-service]') + .hasProperty('type', 'text'); + assert + .dom('[data-test-input-field=profile-service]') + .hasProperty('value', ''); + assert + .dom('[data-test-input-field=profile-service]') + .hasProperty('placeholder', 'Enter your profile service URL'); }); - test('render disabled Next button on chaincode page', async function (assert) { - assert.expect(2); + test('Render Next button on profile page', async function (assert) { + assert.expect(3); + this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); + await render( - hbs``, + hbs``, ); + + assert.dom('[data-test-button=next]').exists(); assert.dom('[data-test-button=next]').hasText('Next'); - assert.dom('[data-test-button=next]').hasProperty('disabled', true); + assert.dom('[data-test-button=next]').hasProperty('type', 'button'); }); - test('Clicking "Generate Chaincode" button renders div with text and 2 button with icons on Chaincode page', async function (assert) { - assert.expect(6); + test("Ensure the 'Next' Button is Enabled Only When a Valid Profile URL is Entered", async function (assert) { + assert.expect(1); + this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => { - this.set('isChaincodeClicked', true); - }); await render( - hbs``, + hbs``, ); - await click('[data-test-button=chaincode]'); - assert.dom('[data-test=chaincode-container]').exists(); - assert - .dom('[data-test=chaincode-container__value]') - .hasClass('chaincode-container__value-invisible'); - assert - .dom('[data-test=chaincode-container__value]') - .hasText('********************'); - - assert - .dom('[data-test=chaincode-container__action]') - .hasClass('chaincode-container__action'); + await typeIn( + '[data-test-input-field=profile-service]', + 'https://rds-profile-service.onrender.com', + ); - assert.dom('[data-test-button=eye-icon]').hasClass('chaicode-button-icon'); - assert.dom('[data-test-button=copy-icon]').hasClass('chaicode-button-icon'); + assert.dom('[data-test-button=next]').hasProperty('disabled', false); }); - test('Clicking eye-icon button show generated code', async function (assert) { + test("Ensure the 'Next' Button is Disabled When a InValid Profile URL is Entered", async function (assert) { assert.expect(1); + this.set('startHandler', () => {}); - this.set('handleGenerateChaincode', () => {}); - this.set('isChaincodeClicked', true); - this.set('chaincode', 'hv2hz3xh1h'); + await render( - hbs``, + hbs``, ); - await click('[data-test-button=eye-icon]'); - assert.dom('[data-test=chaincode-container__value]').hasText('hv2hz3xh1h'); + await typeIn( + '[data-test-input-field=profile-service]', + 'rds-profile-service.onrender.com', + ); + + assert.dom('[data-test-button=next]').hasProperty('disabled', true); }); }); diff --git a/tests/integration/components/identity-steps/step-one-test.js b/tests/integration/components/identity-steps/step-one-test.js index 9b5750d9..9e2ebfe2 100644 --- a/tests/integration/components/identity-steps/step-one-test.js +++ b/tests/integration/components/identity-steps/step-one-test.js @@ -23,11 +23,11 @@ module('Integration | Component | identity-steps/step-one', function (hooks) { assert .dom('[data-test-getting-started-paragraph-1]') .hasText( - 'Thank you for providing all the details Before joining the community we would want you to complete a small challenge which will also help you in setting up your identity across Real Dev Squad', + 'Thank you for providing all the details. Before joining the community, we would want you to complete a small challenge which will also help you in setting up your identity across Real Dev Squad.', ); assert .dom('[data-test-getting-started-paragraph-2]') - .hasText('Please click proceed to know about the task'); + .hasText('Please click proceed to know about the task.'); assert.dom('[data-test-button=identity-proceed]').hasText('Proceed'); await click('[data-test-button=identity-proceed]'); assert.true(clicked, 'startHandler is called'); diff --git a/tests/integration/components/identity-steps/step-seven-test.js b/tests/integration/components/identity-steps/step-seven-test.js deleted file mode 100644 index 0aeb045e..00000000 --- a/tests/integration/components/identity-steps/step-seven-test.js +++ /dev/null @@ -1,167 +0,0 @@ -import { module, test } from 'qunit'; -import { setupRenderingTest } from 'website-www/tests/helpers'; -import { render, click } from '@ember/test-helpers'; -import { hbs } from 'ember-cli-htmlbars'; -import Service from '@ember/service'; - -class LoginStub extends Service { - userData = { profileStatus: 'PENDING' }; -} - -module('Integration | Component | identity-steps/step-seven', function (hooks) { - setupRenderingTest(hooks); - - hooks.beforeEach(function () { - this.owner.register('service:login', LoginStub); - }); - - test('renders heading on verification page when profile status is pending', async function (assert) { - this.set('handleRefresh', () => {}); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-heading]') - .hasClass('verification-page__heading'); - assert.dom('[data-test-verification-heading]').hasText('Pending'); - }); - - test('render description on verification page when profile status is pending', async function (assert) { - this.set('handleRefresh', () => {}); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-description-container]') - .hasClass('verification-page__description'); - assert - .dom('[data-test-verification-description-1]') - .hasText('Refresh to Check Verification Status'); - assert - .dom('[data-test-verification-description-2]') - .hasText('Your Profile Service Linked with Real Dev Squad Service'); - }); - - test('render Refresh button on verification page when profile status is pending', async function (assert) { - this.set('handleRefresh', () => {}); - await render( - hbs``, - ); - - assert.dom('[data-test-button=refresh]').hasText('Refresh'); - assert.dom('[data-test-button=refresh]').hasProperty('type', 'button'); - }); - - test('clicking Refresh button refresh the verification page when profile status is pending', async function (assert) { - const objToCheckFunctions = { - isHandleRefreshWorks: false, - }; - this.set('handleRefresh', () => { - objToCheckFunctions.isHandleRefreshWorks = true; - }); - await render( - hbs``, - ); - - await click('[data-test-button=refresh]'); - assert.dom('[data-test-button=refresh]').hasText('Refresh'); - assert.dom('[data-test-button=refresh]').hasProperty('type', 'button'); - assert.true( - objToCheckFunctions.isHandleRefreshWorks, - 'handleRefresh function is working fine!', - ); - }); - - test('renders heading on verification page when profile status is blocked', async function (assert) { - this.set('goToGenerateChaincodePage', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'BLOCKED'); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-heading]') - .hasClass('verification-page__heading'); - assert.dom('[data-test-verification-heading]').hasText('Blocked'); - }); - - test('render description on verification page when profile status is blocked', async function (assert) { - this.set('goToGenerateChaincodePage', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'BLOCKED'); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-description-container]') - .hasClass('verification-page__description'); - assert - .dom('[data-test-verification-description-1]') - .hasText('Your previous Chaincode is Blocked.'); - }); - - test('render Verify Again button on verification page when profile status is blocked', async function (assert) { - this.set('goToGenerateChaincodePage', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'BLOCKED'); - await render( - hbs``, - ); - assert.dom('[data-test-button=verify-again]').hasText('Verify Again'); - assert.dom('[data-test-button=verify-again]').hasProperty('type', 'button'); - }); - - test('renders heading on verification page when profile status is verified', async function (assert) { - this.set('startHandler', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'VERIFIED'); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-heading]') - .hasClass('verification-page__heading'); - assert.dom('[data-test-verification-heading]').hasText('Successful'); - }); - - test('render description on verification page when profile status is verified', async function (assert) { - this.set('startHandler', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'VERIFIED'); - await render( - hbs``, - ); - - assert - .dom('[data-test-verification-description-container]') - .hasClass('verification-page__description'); - assert - .dom('[data-test-verification-description-1]') - .hasText('Congratulations! Your Profile Service is Verified.'); - assert - .dom('[data-test-verification-description-2]') - .hasText('Take the Next Step and Join Our Discord Server.'); - }); - - test('render Next button on verification page when profile status is verified', async function (assert) { - this.set('startHandler', () => {}); - this.loginService = this.owner.lookup('service:login'); - this.set('loginService.userData.profileStatus', 'VERIFIED'); - await render( - hbs``, - ); - assert.dom('[data-test-button=next]').hasText('Next'); - assert.dom('[data-test-button=next]').hasProperty('type', 'button'); - }); -}); diff --git a/tests/integration/components/identity-steps/step-six-test.js b/tests/integration/components/identity-steps/step-six-test.js index 1f96cdee..0578491d 100644 --- a/tests/integration/components/identity-steps/step-six-test.js +++ b/tests/integration/components/identity-steps/step-six-test.js @@ -1,63 +1,167 @@ import { module, test } from 'qunit'; import { setupRenderingTest } from 'website-www/tests/helpers'; -import { render } from '@ember/test-helpers'; +import { render, click } from '@ember/test-helpers'; import { hbs } from 'ember-cli-htmlbars'; +import Service from '@ember/service'; -module('Integration | Component | identity-steps/step-six', function (hooks) { +class LoginStub extends Service { + userData = { profileStatus: 'PENDING' }; +} + +module('Integration | Component | identity-steps/step-seven', function (hooks) { setupRenderingTest(hooks); - test('render main container div on profile service url Linking page', async function (assert) { - assert.expect(2); - this.set('startHandler', () => {}); + hooks.beforeEach(function () { + this.owner.register('service:login', LoginStub); + }); + + test('renders heading on verification page when profile status is pending', async function (assert) { + this.set('handleRefresh', () => {}); await render( - hbs``, + hbs``, ); - assert.dom('[data-test=profile-service-url-Linking]').exists(); assert - .dom('[data-test=profile-service-url-Linking]') - .hasClass('profile-service-url-linking-page'); + .dom('[data-test-verification-heading]') + .hasClass('verification-page__heading'); + assert.dom('[data-test-verification-heading]').hasText('Pending'); }); - test('render heading on profile service url Linking page', async function (assert) { - assert.expect(2); - this.set('startHandler', () => {}); + test('render description on verification page when profile status is pending', async function (assert) { + this.set('handleRefresh', () => {}); await render( - hbs``, + hbs``, ); + + assert + .dom('[data-test-verification-description-container]') + .hasClass('verification-page__description'); + assert + .dom('[data-test-verification-description-1]') + .hasText('Refresh to Check Verification Status'); assert - .dom('[data-test=heading]') - .hasClass('profile-service-url-linking-page__heading'); - assert.dom('[data-test=heading]').hasText('Link Profile Service'); + .dom('[data-test-verification-description-2]') + .hasText('Your Profile Service Linked with Real Dev Squad Service'); }); - test('render description on profile service url Linking page', async function (assert) { - assert.expect(2); - this.set('startHandler', () => {}); + test('render Refresh button on verification page when profile status is pending', async function (assert) { + this.set('handleRefresh', () => {}); await render( - hbs``, + hbs``, ); + + assert.dom('[data-test-button=refresh]').hasText('Refresh'); + assert.dom('[data-test-button=refresh]').hasProperty('type', 'button'); + }); + + test('clicking Refresh button refresh the verification page when profile status is pending', async function (assert) { + const objToCheckFunctions = { + isHandleRefreshWorks: false, + }; + this.set('handleRefresh', () => { + objToCheckFunctions.isHandleRefreshWorks = true; + }); + await render( + hbs``, + ); + + await click('[data-test-button=refresh]'); + assert.dom('[data-test-button=refresh]').hasText('Refresh'); + assert.dom('[data-test-button=refresh]').hasProperty('type', 'button'); + assert.true( + objToCheckFunctions.isHandleRefreshWorks, + 'handleRefresh function is working fine!', + ); + }); + + test('renders heading on verification page when profile status is blocked', async function (assert) { + this.set('goToGenerateChaincodePage', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'BLOCKED'); + await render( + hbs``, + ); + assert - .dom('[data-test=description]') - .hasClass('profile-service-url-linking-page__description'); + .dom('[data-test-verification-heading]') + .hasClass('verification-page__heading'); + assert.dom('[data-test-verification-heading]').hasText('Blocked'); + }); + + test('render description on verification page when profile status is blocked', async function (assert) { + this.set('goToGenerateChaincodePage', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'BLOCKED'); + await render( + hbs``, + ); + + assert + .dom('[data-test-verification-description-container]') + .hasClass('verification-page__description'); assert - .dom('[data-test=description]') - .hasText( - 'Ensure that you have deployed your profile service, Click on link button to start the linking process for joining RealDevSquad.', - ); + .dom('[data-test-verification-description-1]') + .hasText('Your previous Chaincode is Blocked.'); }); - test('Render Link button on profile service url Linking page', async function (assert) { - assert.expect(3); + test('render Verify Again button on verification page when profile status is blocked', async function (assert) { + this.set('goToGenerateChaincodePage', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'BLOCKED'); + await render( + hbs``, + ); + assert.dom('[data-test-button=verify-again]').hasText('Verify Again'); + assert.dom('[data-test-button=verify-again]').hasProperty('type', 'button'); + }); + test('renders heading on verification page when profile status is verified', async function (assert) { this.set('startHandler', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'VERIFIED'); + await render( + hbs``, + ); + + assert + .dom('[data-test-verification-heading]') + .hasClass('verification-page__heading'); + assert.dom('[data-test-verification-heading]').hasText('Successful'); + }); + test('render description on verification page when profile status is verified', async function (assert) { + this.set('startHandler', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'VERIFIED'); await render( hbs``, ); - assert.dom('[data-test-button=next]').exists(); - assert.dom('[data-test-button=next]').hasText('Link'); + assert + .dom('[data-test-verification-description-container]') + .hasClass('verification-page__description'); + assert + .dom('[data-test-verification-description-1]') + .hasText('Congratulations! Your Profile Service is Verified.'); + assert + .dom('[data-test-verification-description-2]') + .hasText('Take the Next Step and Join Our Discord Server.'); + }); + + test('render Next button on verification page when profile status is verified', async function (assert) { + this.set('startHandler', () => {}); + this.loginService = this.owner.lookup('service:login'); + this.set('loginService.userData.profileStatus', 'VERIFIED'); + await render( + hbs``, + ); + assert.dom('[data-test-button=next]').hasText('Next'); assert.dom('[data-test-button=next]').hasProperty('type', 'button'); }); }); diff --git a/tests/integration/components/identity-steps/step-three-test.js b/tests/integration/components/identity-steps/step-three-test.js index 7eca9b30..c41c7e87 100644 --- a/tests/integration/components/identity-steps/step-three-test.js +++ b/tests/integration/components/identity-steps/step-three-test.js @@ -3,33 +3,138 @@ import { setupRenderingTest } from 'website-www/tests/helpers'; import { render, click } from '@ember/test-helpers'; import { hbs } from 'ember-cli-htmlbars'; -module('Integration | Component | identity-steps/step-three', function (hooks) { +module('Integration | Component | identity-steps/step-four', function (hooks) { setupRenderingTest(hooks); - test('stepThree renders', async function (assert) { - assert.expect(5); - let clicked = false; - this.set('startHandler', () => { - clicked = true; - }); + test('render main container div on generate chaincode page', async function (assert) { + assert.expect(2); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); + await render( + hbs``, + ); + + assert.dom('[data-test=chaincode]').exists(); + assert.dom('[data-test=chaincode]').hasClass('chaincode-page'); + }); + test('render heading on generate chaincode page', async function (assert) { + assert.expect(2); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); await render( - hbs``, + hbs``, ); + assert.dom('[data-test=heading]').hasClass('chaincode-page__heading'); + assert.dom('[data-test=heading]').hasText('Chaincode Generation'); + }); + test('render description on generate chaincode page', async function (assert) { + assert.expect(2); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); + await render( + hbs``, + ); assert - .dom('[data-test-getting-started-heading]') - .hasText('Chaincode Generation'); + .dom('[data-test=description]') + .hasClass('chaincode-page__description'); assert - .dom('[data-test-getting-started-paragraph-1]') + .dom('[data-test=description]') .hasText( - "A private that you need to use in your profile service URL and deploy for source that you're the source of the URL", + "A private key that you need to use in your profile service URL and deploy for source that you're the source of the URL", ); + }); + + test('render Generation Chaincode button on chaincode page', async function (assert) { + assert.expect(2); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); + this.set('chaincode', 'Generate Chaincode'); + this.set('isChaincodeClicked', false); + await render( + hbs``, + ); + assert.dom('[data-test-button=chaincode]').hasText('Generate Chaincode'); + assert.dom('[data-test-button=chaincode]').hasProperty('type', 'button'); + }); + + test('render disabled Next button on chaincode page', async function (assert) { + assert.expect(2); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); + await render( + hbs``, + ); + assert.dom('[data-test-button=next]').hasText('Next'); + assert.dom('[data-test-button=next]').hasProperty('disabled', true); + }); + + test('Clicking "Generate Chaincode" button renders div with text and 2 button with icons on Chaincode page', async function (assert) { + assert.expect(6); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => { + this.set('isChaincodeClicked', true); + }); + + await render( + hbs``, + ); + await click('[data-test-button=chaincode]'); + + assert.dom('[data-test=chaincode-container]').exists(); + assert + .dom('[data-test=chaincode-container__value]') + .hasClass('chaincode-container__value-invisible'); assert - .dom('[data-test-getting-started-paragraph-2]') - .hasText('https://github.com/identity-service/instructions.md'); - assert.dom('[data-test-button=identity-next]').hasText('Next'); - await click('[data-test-button=identity-next]'); - assert.true(clicked, 'click is called'); + .dom('[data-test=chaincode-container__value]') + .hasText('********************'); + + assert + .dom('[data-test=chaincode-container-action]') + .hasClass('chaincode-container-action'); + + assert.dom('[data-test-button=eye-icon]').hasClass('chaicode-button-icon'); + assert.dom('[data-test-button=copy-icon]').hasClass('chaicode-button-icon'); + }); + + test('Clicking eye-icon button show generated code', async function (assert) { + assert.expect(1); + this.set('startHandler', () => {}); + this.set('handleGenerateChaincode', () => {}); + this.set('isChaincodeClicked', true); + this.set('chaincode', 'hv2hz3xh1h'); + await render( + hbs``, + ); + await click('[data-test-button=eye-icon]'); + + assert.dom('[data-test=chaincode-container__value]').hasText('hv2hz3xh1h'); }); }); diff --git a/tests/integration/components/identity-steps/step-two-test.js b/tests/integration/components/identity-steps/step-two-test.js index 629f1645..8597bdcf 100644 --- a/tests/integration/components/identity-steps/step-two-test.js +++ b/tests/integration/components/identity-steps/step-two-test.js @@ -7,7 +7,7 @@ module('Integration | Component | identity-steps/step-two', function (hooks) { setupRenderingTest(hooks); test('stepTwo renders', async function (assert) { - assert.expect(5); + assert.expect(6); let clicked = false; this.set('startHandler', () => { clicked = true; @@ -16,6 +16,7 @@ module('Integration | Component | identity-steps/step-two', function (hooks) { await render( hbs``, ); + assert .dom('[data-test-getting-started-heading]') .hasText('Challenge Time !!'); @@ -26,9 +27,18 @@ module('Integration | Component | identity-steps/step-two', function (hooks) { ); assert .dom('[data-test-getting-started-paragraph-2]') - .hasText('https://github.com/identity-service/instructions.md'); - assert.dom('[data-test-button=identity-next]').hasText('Next'); - await click('[data-test-button=identity-next]'); + .hasText( + 'After exploring the sample profile service, proceed to the next step by clicking the button below.', + ); + + assert + .dom('[data-test-profile-service-link]') + .hasAttribute( + 'href', + 'https://github.com/Real-Dev-Squad/sample-profile-service', + ); + assert.dom('[data-test-button="identity-next"]').hasText('Next'); + await click('[data-test-button="identity-next"]'); assert.true(clicked, 'click is called'); }); }); From dd1df327c9358f8d709b7b96c2e9b85783e94148 Mon Sep 17 00:00:00 2001 From: Vinit khandal <111434418+vinit717@users.noreply.github.com> Date: Thu, 18 Jan 2024 23:44:27 +0530 Subject: [PATCH 12/12] Refactor application status card (#833) * refactor application status card * write test for status card * add test case foe unknown status * fix rejected icon color --------- Co-authored-by: Satyam Bajpai --- app/components/join-steps/status-card.hbs | 39 +++++++++-------- app/components/join-steps/status-card.js | 6 +-- app/styles/identity-card.module.css | 8 ++-- app/styles/status-card.module.css | 2 +- .../components/status-card-test.js | 42 ++++++++++++++++++- 5 files changed, 70 insertions(+), 27 deletions(-) diff --git a/app/components/join-steps/status-card.hbs b/app/components/join-steps/status-card.hbs index 665ec4cf..d3659d4d 100644 --- a/app/components/join-steps/status-card.hbs +++ b/app/components/join-steps/status-card.hbs @@ -8,12 +8,11 @@ > {{statusDetails.heading}}

- {{/if}} {{/each}} @@ -31,23 +30,27 @@

Your application is rejected

-

- Here's the feedback for your application -

-

- {{@feedback}} -

+ {{#if @feedback}} +

+ Here's the feedback for your application +

+

+ {{@feedback}} +

+ {{/if}} {{else if (eq @status this.APPLICATION_STATUS_TYPES.accepted)}}

Congratulations! Your application is accepted by us

-

- Here's the feedback for your application -

-

- {{@feedback}} -

+ {{#if @feedback}} +

+ Here's the feedback for your application +

+

+ {{@feedback}} +

+ {{/if}}

Take the Next Step and Join Our Discord Server.

@@ -67,4 +70,4 @@ }} />
-
\ No newline at end of file +
diff --git a/app/components/join-steps/status-card.js b/app/components/join-steps/status-card.js index 18bc3b11..c69872f2 100644 --- a/app/components/join-steps/status-card.js +++ b/app/components/join-steps/status-card.js @@ -12,17 +12,17 @@ export default class StatusCardComponent extends Component { { status: APPLICATION_STATUS_TYPES.pending, heading: 'Pending', - icon: 'mdi:timer-sand', + icon: 'hourglass-half', }, { status: APPLICATION_STATUS_TYPES.rejected, heading: 'Rejected', - icon: 'mdi:close-circle', + icon: 'ban', }, { status: APPLICATION_STATUS_TYPES.accepted, heading: 'Accepted', - icon: 'mdi:check-circle', + icon: 'square-check', }, ]; } diff --git a/app/styles/identity-card.module.css b/app/styles/identity-card.module.css index ad08b3aa..913899ff 100644 --- a/app/styles/identity-card.module.css +++ b/app/styles/identity-card.module.css @@ -102,8 +102,8 @@ profile-service-page__input-label { } .ban-icon { - padding: 1.4rem 0 0 0.5rem; - color: var(--color-red); + padding: 1.3rem 0 0 0.5rem; + color: var(--text-red); } .square-check-icon { @@ -120,13 +120,13 @@ profile-service-page__input-label { cursor: pointer; } -@media (width <= 425px) { +@media (width <= 480px) { .hourglass-half-icon { padding: 0.5rem 0 0 0.5rem; } .ban-icon { - padding: 0.4rem 0 0 0.5rem; + padding: 0.3rem 0 0 0.5rem; } .square-check-icon { diff --git a/app/styles/status-card.module.css b/app/styles/status-card.module.css index a824c3cb..178e1661 100644 --- a/app/styles/status-card.module.css +++ b/app/styles/status-card.module.css @@ -1,7 +1,6 @@ .status-card { display: flex; flex-direction: column; - gap: 1.5rem; align-items: center; } @@ -12,4 +11,5 @@ .status-card__description { text-align: center; line-height: 2rem; + padding-bottom: 12px; } diff --git a/tests/integration/components/status-card-test.js b/tests/integration/components/status-card-test.js index edf4dc7a..019428f6 100644 --- a/tests/integration/components/status-card-test.js +++ b/tests/integration/components/status-card-test.js @@ -57,7 +57,7 @@ module('Integration | Component | status-card', function (hooks) { .hasText("Here's the feedback for your application"); }); - test('it renders accepted status', async function (assert) { + test('it renders accepted status with feedback', async function (assert) { assert.expect(5); this.set('status', 'accepted'); @@ -83,4 +83,44 @@ module('Integration | Component | status-card', function (hooks) { .dom('[data-test-status-card-description-3]') .hasText('Feedback for accepted status'); }); + + test('it renders accepted status without feedback', async function (assert) { + assert.expect(4); + + this.set('status', 'accepted'); + this.set('feedback', null); + + await render(hbs` + + `); + + assert.dom('[data-test-status-card-heading]').hasText('Accepted'); + assert.dom('[data-test-icon="accepted"]').exists(); + assert + .dom('[data-test-status-card-description-1]') + .hasText('Congratulations! Your application is accepted by us'); + assert.dom('[data-test-status-card-description-2]').doesNotExist(); + }); + + test('it handles unknown status', async function (assert) { + assert.expect(2); + + this.set('status', 'unknown'); + this.set('feedback', 'This is unexpected'); + + await render(hbs` + + `); + + assert.dom('[data-test-status-card-heading]').doesNotExist(); + assert.dom('[data-test-icon]').doesNotExist(); + }); });