From b966b4e9818472de948b18fb9105229fa39c15ad Mon Sep 17 00:00:00 2001 From: Ramon Wenger Date: Tue, 24 Aug 2021 13:31:05 +0200 Subject: [PATCH] Re-add comment test --- .../frontend/rooms/article-page.spec.js | 5 +- client/src/components/rooms/CommentInput.vue | 13 ++++- client/src/graphql/client.js | 1 + .../src/graphql/gql/mutations/addComment.gql | 9 +++ .../graphql/gql/queries/roomEntryQuery.gql | 12 ++++ client/src/main.js | 8 ++- client/src/pages/article.vue | 57 ++++++++++++++----- server/schema.graphql | 47 ++++++++++----- server/users/schema.py | 1 + 9 files changed, 122 insertions(+), 31 deletions(-) diff --git a/client/cypress/integration/frontend/rooms/article-page.spec.js b/client/cypress/integration/frontend/rooms/article-page.spec.js index 010a4874..38219d70 100644 --- a/client/cypress/integration/frontend/rooms/article-page.spec.js +++ b/client/cypress/integration/frontend/rooms/article-page.spec.js @@ -16,8 +16,7 @@ describe('Article page', () => { RoomEntryQuery: { roomEntry, }, - AddComment(input) { - console.log(input); + AddComment({input}) { return { addComment: { success: true, @@ -38,7 +37,7 @@ describe('Article page', () => { cy.setup(); }); - it.skip('goes to article and leaves a comment', () => { + it('goes to article and leaves a comment', () => { cy.mockGraphqlOps({ operations, }); diff --git a/client/src/components/rooms/CommentInput.vue b/client/src/components/rooms/CommentInput.vue index cd599e7e..da9dc6b9 100644 --- a/client/src/components/rooms/CommentInput.vue +++ b/client/src/components/rooms/CommentInput.vue @@ -7,11 +7,12 @@ data-cy="comment-textarea" class="comment-input__textarea" placeholder="Kommentar erfassen" + @input="updateValue($event.target.value)" /> Kommentar teilen + @click="$emit('submit', text)">Kommentar teilen @@ -28,6 +29,16 @@ components: { EmojiBar, }, + data() { + return { + text: '' + }; + }, + methods: { + updateValue(text) { + this.text = text; + } + }, }; diff --git a/client/src/graphql/client.js b/client/src/graphql/client.js index 2d759392..c0b7e3be 100644 --- a/client/src/graphql/client.js +++ b/client/src/graphql/client.js @@ -92,6 +92,7 @@ export default function (uri, networkErrorCallback) { switch (obj.__typename) { case 'InstrumentNode': case 'ModuleNode': + case 'RoomEntryNode': return `${obj.__typename}:${obj.slug}`; default: return defaultDataIdFromObject(obj); diff --git a/client/src/graphql/gql/mutations/addComment.gql b/client/src/graphql/gql/mutations/addComment.gql index 8d35714b..6535f630 100644 --- a/client/src/graphql/gql/mutations/addComment.gql +++ b/client/src/graphql/gql/mutations/addComment.gql @@ -1,5 +1,14 @@ mutation AddComment($input: AddCommentInput!) { addComment(input: $input) { success + comment { + text + id + owner { + firstName + lastName + } + created + } } } diff --git a/client/src/graphql/gql/queries/roomEntryQuery.gql b/client/src/graphql/gql/queries/roomEntryQuery.gql index 85076cf5..210c4e12 100644 --- a/client/src/graphql/gql/queries/roomEntryQuery.gql +++ b/client/src/graphql/gql/queries/roomEntryQuery.gql @@ -2,5 +2,17 @@ query RoomEntryQuery($slug: String!) { roomEntry(slug: $slug) { ...RoomEntryParts + comments { + edges { + node { + text + owner { + firstName + lastName + } + created + } + } + } } } diff --git a/client/src/main.js b/client/src/main.js index 4da9d4a8..2f06f7ce 100644 --- a/client/src/main.js +++ b/client/src/main.js @@ -131,7 +131,7 @@ function redirectUsersWithoutValidLicense(to) { function redirectStudentsWithoutClass() { return privateApolloClient.query({ query: ME_QUERY, - }).then(({data}) => data.me.schoolClasses.edges.length === 0 && data.me.permissions.length === 0); + }).then(({data}) => data.me.schoolClasses.edges.length === 0 && !data.me.isTeacher); } function redirectUsersToOnboarding(to) { @@ -169,21 +169,27 @@ router.beforeEach(async (to, from, next) => { localStorage.setItem(postLoginRedirectUrlKey, postLoginRedirectionUrl); } + Vue.$log.debug('redirecting to hello', to); next(redirectUrl); return; } if (to.name && to.name !== 'licenseActivation' && loginRequired(to) && await redirectUsersWithoutValidLicense()) { + Vue.$log.debug('redirecting to licenseActivation', to, null); + console.log('redirecting to licenseActivation', to, null); next({name: 'licenseActivation'}); return; } if (!joiningClass(to) && loginRequired(to) && await redirectStudentsWithoutClass()) { + Vue.$log.debug('redirecting to join-class', to); + Vue.$log.debug('await redirectStudentsWithoutClass()', await redirectStudentsWithoutClass()); next({name: 'join-class'}); return; } if ((to.name && to.name.indexOf('onboarding') === -1) && !joiningClass(to) && loginRequired(to) && await redirectUsersToOnboarding()) { + Vue.$log.debug('redirecting to onboarding-start', to); next({name: 'onboarding-start'}); return; } diff --git a/client/src/pages/article.vue b/client/src/pages/article.vue index d996790e..f4964b80 100644 --- a/client/src/pages/article.vue +++ b/client/src/pages/article.vue @@ -15,7 +15,11 @@ v-for="component in roomEntry.contents"/>
- + +
@@ -32,43 +36,70 @@ import ROOM_ENTRY_QUERY from '@/graphql/gql/queries/roomEntryQuery.gql'; import ADD_COMMENT_MUTATION from 'gql/mutations/addComment.gql'; import CommentInput from '@/components/rooms/CommentInput'; + import Comment from '@/components/rooms/Comment'; export default { components: { CommentInput, + Comment, 'text_block': TextBlock, 'image_block': ImageBlock, 'image_url_block': ImageUrlBlock, 'video_block': VideoBlock, 'link_block': LinkBlock, 'document_block': DocumentBlock, - UserMetaWidget + UserMetaWidget, }, data() { return { roomEntry: { author: { - name: 'Daniel Ramos', - avatar: '', - date: '4. Juli 2018 - 09:23' - } + name: 'Daniel Ramos', + avatar: '', + date: '4. Juli 2018 - 09:23', + }, }, }; }, + computed: { + comments() { + return (this.roomEntry && this.roomEntry.comments) ? this.roomEntry.comments.edges.map(edge => edge.node) : []; + }, + }, + methods: { createComment(text) { + console.log('text', text); this.$apollo.mutate({ mutation: ADD_COMMENT_MUTATION, variables: { input: { - roomEntry: this.roomEntry, - comment: text + roomEntry: this.roomEntry.id, + comment: text, + }, + }, + update: (store, {data: {addComment: {success, comment}}}) => { + if (success) { + const query = ROOM_ENTRY_QUERY; + const variables = { + slug: this.roomEntry.slug, + }; + const data = store.readQuery({ + query, + variables, + }); + if (data) { + data.roomEntry.comments.edges.unshift({ + node: comment, + }); + store.writeQuery({query, variables, data}); + } } - } + }, }); - } + }, }, apollo: { @@ -76,11 +107,11 @@ return { query: ROOM_ENTRY_QUERY, variables: { - slug: this.$route.params.slug - } + slug: this.$route.params.slug, + }, }; - } + }, }, }; diff --git a/server/schema.graphql b/server/schema.graphql index b058420b..963f2433 100644 --- a/server/schema.graphql +++ b/server/schema.graphql @@ -3,6 +3,18 @@ schema { mutation: CustomMutation } +input AddCommentInput { + comment: String! + roomEntry: ID! + clientMutationId: String +} + +type AddCommentPayload { + success: Boolean + comment: CommentNode + clientMutationId: String +} + input AddContentBlockInput { contentBlock: ContentBlockInput parent: ID @@ -255,6 +267,23 @@ type ClassMemberNode { isMe: Boolean } +type CommentNode implements Node { + text: String! + owner: PublicUserNode + created: DateTime! + id: ID! +} + +type CommentNodeConnection { + pageInfo: PageInfo! + edges: [CommentNodeEdge]! +} + +type CommentNodeEdge { + node: CommentNode + cursor: String! +} + type ContentBlockBookmarkNode implements Node { id: ID! user: PrivateUserNode! @@ -411,6 +440,7 @@ type CustomMutation { addRoomEntry(input: AddRoomEntryInput!): AddRoomEntryPayload deleteRoomEntry(input: DeleteRoomEntryInput!): DeleteRoomEntryPayload updateRoomEntry(input: UpdateRoomEntryInput!): UpdateRoomEntryPayload + addComment(input: AddCommentInput!): AddCommentPayload mutateContentBlock(input: MutateContentBlockInput!): MutateContentBlockPayload addContentBlock(input: AddContentBlockInput!): AddContentBlockPayload deleteContentBlock(input: DeleteContentBlockInput!): DeleteContentBlockPayload @@ -854,6 +884,7 @@ type ProjectNode implements Node { appearance: String! student: PrivateUserNode! final: Boolean! + schoolClass: SchoolClassNode entries(offset: Int, before: String, after: String, first: Int, last: Int): ProjectEntryNodeConnection! pk: Int entriesCount: Int @@ -886,6 +917,7 @@ type RoomEntryNode implements Node { room: RoomNode! author: PublicUserNode contents: GenericStreamFieldType + comments(offset: Int, before: String, after: String, first: Int, last: Int, owner: ID): CommentNodeConnection pk: Int } @@ -928,20 +960,9 @@ input SchoolClassInput { } type SchoolClassNode implements Node { - id: ID! name: String! - isDeleted: Boolean! code: String - users(offset: Int, before: String, after: String, first: Int, last: Int, username: String, email: String): PrivateUserNodeConnection! - moduleSet(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, slug_Icontains: String, slug_In: [String], title: String, title_Icontains: String, title_In: [String]): ModuleNodeConnection! - hiddenChapterTitles(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, title: String): ChapterNodeConnection! - hiddenChapterDescriptions(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, title: String): ChapterNodeConnection! - hiddenContentBlocks(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, title: String): ContentBlockNodeConnection! - visibleContentBlocks(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, title: String): ContentBlockNodeConnection! - hiddenObjectiveGroups(offset: Int, before: String, after: String, first: Int, last: Int, title: String, module_Slug: String): ObjectiveGroupNodeConnection! - hiddenObjectives(offset: Int, before: String, after: String, first: Int, last: Int, text: String): ObjectiveNodeConnection! - visibleObjectives(offset: Int, before: String, after: String, first: Int, last: Int, text: String): ObjectiveNodeConnection! - rooms(offset: Int, before: String, after: String, first: Int, last: Int, slug: String, appearance: String): RoomNodeConnection! + id: ID! pk: Int members: [ClassMemberNode] readOnly: Boolean @@ -1128,9 +1149,9 @@ type SyncModuleVisibilityPayload { type TeamNode implements Node { name: String! - isDeleted: Boolean! code: String id: ID! + isDeleted: Boolean! creator: PrivateUserNode members: [PublicUserNode] pk: Int diff --git a/server/users/schema.py b/server/users/schema.py index 9a7d34be..425f97b8 100644 --- a/server/users/schema.py +++ b/server/users/schema.py @@ -24,6 +24,7 @@ class SchoolClassNode(DjangoObjectType): class Meta: model = SchoolClass + only_fields = ['name', 'code', 'members', 'pk', 'read_only'] filter_fields = ['name'] interfaces = (relay.Node,)