mirror of
https://github.com/Ocelot-Social-Community/Ocelot-Social.git
synced 2025-12-13 07:46:06 +00:00
Merge branch 'master' of github.com:Ocelot-Social-Community/Ocelot-Social into 6545-add-headline-to-chat-page
This commit is contained in:
commit
40b40eb486
@ -6,6 +6,9 @@ export const createMessageMutation = () => {
|
|||||||
CreateMessage(roomId: $roomId, content: $content) {
|
CreateMessage(roomId: $roomId, content: $content) {
|
||||||
id
|
id
|
||||||
content
|
content
|
||||||
|
saved
|
||||||
|
distributed
|
||||||
|
seen
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
@ -13,16 +16,28 @@ export const createMessageMutation = () => {
|
|||||||
|
|
||||||
export const messageQuery = () => {
|
export const messageQuery = () => {
|
||||||
return gql`
|
return gql`
|
||||||
query ($roomId: ID!) {
|
query ($roomId: ID!, $first: Int, $offset: Int) {
|
||||||
Message(roomId: $roomId) {
|
Message(roomId: $roomId, first: $first, offset: $offset, orderBy: createdAt_desc) {
|
||||||
_id
|
_id
|
||||||
id
|
id
|
||||||
|
indexId
|
||||||
content
|
content
|
||||||
senderId
|
senderId
|
||||||
username
|
username
|
||||||
avatar
|
avatar
|
||||||
date
|
date
|
||||||
|
saved
|
||||||
|
distributed
|
||||||
|
seen
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const markMessagesAsSeen = () => {
|
||||||
|
return gql`
|
||||||
|
mutation ($messageIds: [String!]) {
|
||||||
|
MarkMessagesAsSeen(messageIds: $messageIds)
|
||||||
|
}
|
||||||
|
`
|
||||||
|
}
|
||||||
|
|||||||
@ -463,6 +463,7 @@ export default shield(
|
|||||||
saveCategorySettings: isAuthenticated,
|
saveCategorySettings: isAuthenticated,
|
||||||
CreateRoom: isAuthenticated,
|
CreateRoom: isAuthenticated,
|
||||||
CreateMessage: isAuthenticated,
|
CreateMessage: isAuthenticated,
|
||||||
|
MarkMessagesAsSeen: isAuthenticated,
|
||||||
},
|
},
|
||||||
User: {
|
User: {
|
||||||
email: or(isMyOwn, isAdmin),
|
email: or(isMyOwn, isAdmin),
|
||||||
|
|||||||
@ -2,7 +2,7 @@ import { createTestClient } from 'apollo-server-testing'
|
|||||||
import Factory, { cleanDatabase } from '../../db/factories'
|
import Factory, { cleanDatabase } from '../../db/factories'
|
||||||
import { getNeode, getDriver } from '../../db/neo4j'
|
import { getNeode, getDriver } from '../../db/neo4j'
|
||||||
import { createRoomMutation } from '../../graphql/rooms'
|
import { createRoomMutation } from '../../graphql/rooms'
|
||||||
import { createMessageMutation, messageQuery } from '../../graphql/messages'
|
import { createMessageMutation, messageQuery, markMessagesAsSeen } from '../../graphql/messages'
|
||||||
import createServer from '../../server'
|
import createServer from '../../server'
|
||||||
|
|
||||||
const driver = getDriver()
|
const driver = getDriver()
|
||||||
@ -122,6 +122,9 @@ describe('Message', () => {
|
|||||||
CreateMessage: {
|
CreateMessage: {
|
||||||
id: expect.any(String),
|
id: expect.any(String),
|
||||||
content: 'Some nice message to other chatting user',
|
content: 'Some nice message to other chatting user',
|
||||||
|
saved: true,
|
||||||
|
distributed: false,
|
||||||
|
seen: false,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
@ -212,11 +215,15 @@ describe('Message', () => {
|
|||||||
{
|
{
|
||||||
id: expect.any(String),
|
id: expect.any(String),
|
||||||
_id: result.data.Message[0].id,
|
_id: result.data.Message[0].id,
|
||||||
|
indexId: 0,
|
||||||
content: 'Some nice message to other chatting user',
|
content: 'Some nice message to other chatting user',
|
||||||
senderId: 'chatting-user',
|
senderId: 'chatting-user',
|
||||||
username: 'Chatting User',
|
username: 'Chatting User',
|
||||||
avatar: expect.any(String),
|
avatar: expect.any(String),
|
||||||
date: expect.any(String),
|
date: expect.any(String),
|
||||||
|
saved: true,
|
||||||
|
distributed: true,
|
||||||
|
seen: false,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@ -253,17 +260,65 @@ describe('Message', () => {
|
|||||||
).resolves.toMatchObject({
|
).resolves.toMatchObject({
|
||||||
errors: undefined,
|
errors: undefined,
|
||||||
data: {
|
data: {
|
||||||
Message: expect.arrayContaining([
|
Message: [
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
id: expect.any(String),
|
id: expect.any(String),
|
||||||
|
indexId: 0,
|
||||||
content: 'Some nice message to other chatting user',
|
content: 'Some nice message to other chatting user',
|
||||||
senderId: 'chatting-user',
|
senderId: 'chatting-user',
|
||||||
username: 'Chatting User',
|
username: 'Chatting User',
|
||||||
avatar: expect.any(String),
|
avatar: expect.any(String),
|
||||||
date: expect.any(String),
|
date: expect.any(String),
|
||||||
|
saved: true,
|
||||||
|
distributed: true,
|
||||||
|
seen: false,
|
||||||
}),
|
}),
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
id: expect.any(String),
|
id: expect.any(String),
|
||||||
|
indexId: 1,
|
||||||
|
content: 'A nice response message to chatting user',
|
||||||
|
senderId: 'other-chatting-user',
|
||||||
|
username: 'Other Chatting User',
|
||||||
|
avatar: expect.any(String),
|
||||||
|
date: expect.any(String),
|
||||||
|
saved: true,
|
||||||
|
distributed: true,
|
||||||
|
seen: false,
|
||||||
|
}),
|
||||||
|
expect.objectContaining({
|
||||||
|
id: expect.any(String),
|
||||||
|
indexId: 2,
|
||||||
|
content: 'And another nice message to other chatting user',
|
||||||
|
senderId: 'chatting-user',
|
||||||
|
username: 'Chatting User',
|
||||||
|
avatar: expect.any(String),
|
||||||
|
date: expect.any(String),
|
||||||
|
saved: true,
|
||||||
|
distributed: false,
|
||||||
|
seen: false,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns the messages paginated', async () => {
|
||||||
|
await expect(
|
||||||
|
query({
|
||||||
|
query: messageQuery(),
|
||||||
|
variables: {
|
||||||
|
roomId,
|
||||||
|
first: 2,
|
||||||
|
offset: 0,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
errors: undefined,
|
||||||
|
data: {
|
||||||
|
Message: [
|
||||||
|
expect.objectContaining({
|
||||||
|
id: expect.any(String),
|
||||||
|
indexId: 1,
|
||||||
content: 'A nice response message to chatting user',
|
content: 'A nice response message to chatting user',
|
||||||
senderId: 'other-chatting-user',
|
senderId: 'other-chatting-user',
|
||||||
username: 'Other Chatting User',
|
username: 'Other Chatting User',
|
||||||
@ -272,13 +327,40 @@ describe('Message', () => {
|
|||||||
}),
|
}),
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
id: expect.any(String),
|
id: expect.any(String),
|
||||||
|
indexId: 2,
|
||||||
content: 'And another nice message to other chatting user',
|
content: 'And another nice message to other chatting user',
|
||||||
senderId: 'chatting-user',
|
senderId: 'chatting-user',
|
||||||
username: 'Chatting User',
|
username: 'Chatting User',
|
||||||
avatar: expect.any(String),
|
avatar: expect.any(String),
|
||||||
date: expect.any(String),
|
date: expect.any(String),
|
||||||
}),
|
}),
|
||||||
]),
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
query({
|
||||||
|
query: messageQuery(),
|
||||||
|
variables: {
|
||||||
|
roomId,
|
||||||
|
first: 2,
|
||||||
|
offset: 2,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
errors: undefined,
|
||||||
|
data: {
|
||||||
|
Message: [
|
||||||
|
expect.objectContaining({
|
||||||
|
id: expect.any(String),
|
||||||
|
indexId: 0,
|
||||||
|
content: 'Some nice message to other chatting user',
|
||||||
|
senderId: 'chatting-user',
|
||||||
|
username: 'Chatting User',
|
||||||
|
avatar: expect.any(String),
|
||||||
|
date: expect.any(String),
|
||||||
|
}),
|
||||||
|
],
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@ -308,4 +390,74 @@ describe('Message', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('marks massges as seen', () => {
|
||||||
|
describe('unauthenticated', () => {
|
||||||
|
beforeAll(() => {
|
||||||
|
authenticatedUser = null
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws authorization error', async () => {
|
||||||
|
await expect(
|
||||||
|
mutate({
|
||||||
|
mutation: markMessagesAsSeen(),
|
||||||
|
variables: {
|
||||||
|
messageIds: ['some-id'],
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
errors: [{ message: 'Not Authorized!' }],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('authenticated', () => {
|
||||||
|
const messageIds: string[] = []
|
||||||
|
beforeAll(async () => {
|
||||||
|
authenticatedUser = await otherChattingUser.toJson()
|
||||||
|
const msgs = await query({
|
||||||
|
query: messageQuery(),
|
||||||
|
variables: {
|
||||||
|
roomId,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
msgs.data.Message.forEach((m) => messageIds.push(m.id))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns true', async () => {
|
||||||
|
await expect(
|
||||||
|
mutate({
|
||||||
|
mutation: markMessagesAsSeen(),
|
||||||
|
variables: {
|
||||||
|
messageIds,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
errors: undefined,
|
||||||
|
data: {
|
||||||
|
MarkMessagesAsSeen: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has seen prop set to true', async () => {
|
||||||
|
await expect(
|
||||||
|
query({
|
||||||
|
query: messageQuery(),
|
||||||
|
variables: {
|
||||||
|
roomId,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
data: {
|
||||||
|
Message: [
|
||||||
|
expect.objectContaining({ seen: true }),
|
||||||
|
expect.objectContaining({ seen: false }),
|
||||||
|
expect.objectContaining({ seen: true }),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@ -13,13 +13,42 @@ export default {
|
|||||||
id: context.user.id,
|
id: context.user.id,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
const resolved = await neo4jgraphql(object, params, context, resolveInfo)
|
const resolved = await neo4jgraphql(object, params, context, resolveInfo)
|
||||||
|
|
||||||
if (resolved) {
|
if (resolved) {
|
||||||
|
const undistributedMessagesIds = resolved
|
||||||
|
.filter((msg) => !msg.distributed && msg.senderId !== context.user.id)
|
||||||
|
.map((msg) => msg.id)
|
||||||
|
if (undistributedMessagesIds.length > 0) {
|
||||||
|
const session = context.driver.session()
|
||||||
|
const writeTxResultPromise = session.writeTransaction(async (transaction) => {
|
||||||
|
const setDistributedCypher = `
|
||||||
|
MATCH (m:Message) WHERE m.id IN $undistributedMessagesIds
|
||||||
|
SET m.distributed = true
|
||||||
|
RETURN m { .* }
|
||||||
|
`
|
||||||
|
const setDistributedTxResponse = await transaction.run(setDistributedCypher, {
|
||||||
|
undistributedMessagesIds,
|
||||||
|
})
|
||||||
|
const messages = await setDistributedTxResponse.records.map((record) => record.get('m'))
|
||||||
|
return messages
|
||||||
|
})
|
||||||
|
try {
|
||||||
|
await writeTxResultPromise
|
||||||
|
} finally {
|
||||||
|
session.close()
|
||||||
|
}
|
||||||
|
// send subscription to author to updated the messages
|
||||||
|
}
|
||||||
resolved.forEach((message) => {
|
resolved.forEach((message) => {
|
||||||
message._id = message.id
|
message._id = message.id
|
||||||
|
if (message.senderId !== context.user.id) {
|
||||||
|
message.distributed = true
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return resolved
|
return resolved.reverse()
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
Mutation: {
|
Mutation: {
|
||||||
@ -32,10 +61,16 @@ export default {
|
|||||||
const writeTxResultPromise = session.writeTransaction(async (transaction) => {
|
const writeTxResultPromise = session.writeTransaction(async (transaction) => {
|
||||||
const createMessageCypher = `
|
const createMessageCypher = `
|
||||||
MATCH (currentUser:User { id: $currentUserId })-[:CHATS_IN]->(room:Room { id: $roomId })
|
MATCH (currentUser:User { id: $currentUserId })-[:CHATS_IN]->(room:Room { id: $roomId })
|
||||||
|
OPTIONAL MATCH (m:Message)-[:INSIDE]->(room)
|
||||||
|
WITH MAX(m.indexId) as maxIndex, room, currentUser
|
||||||
CREATE (currentUser)-[:CREATED]->(message:Message {
|
CREATE (currentUser)-[:CREATED]->(message:Message {
|
||||||
createdAt: toString(datetime()),
|
createdAt: toString(datetime()),
|
||||||
id: apoc.create.uuid(),
|
id: apoc.create.uuid(),
|
||||||
content: $content
|
indexId: CASE WHEN maxIndex IS NOT NULL THEN maxIndex + 1 ELSE 0 END,
|
||||||
|
content: $content,
|
||||||
|
saved: true,
|
||||||
|
distributed: false,
|
||||||
|
seen: false
|
||||||
})-[:INSIDE]->(room)
|
})-[:INSIDE]->(room)
|
||||||
RETURN message { .* }
|
RETURN message { .* }
|
||||||
`
|
`
|
||||||
@ -58,6 +93,32 @@ export default {
|
|||||||
session.close()
|
session.close()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
MarkMessagesAsSeen: async (_parent, params, context, _resolveInfo) => {
|
||||||
|
const { messageIds } = params
|
||||||
|
const currentUserId = context.user.id
|
||||||
|
const session = context.driver.session()
|
||||||
|
const writeTxResultPromise = session.writeTransaction(async (transaction) => {
|
||||||
|
const setSeenCypher = `
|
||||||
|
MATCH (m:Message)<-[:CREATED]-(user:User)
|
||||||
|
WHERE m.id IN $messageIds AND NOT user.id = $currentUserId
|
||||||
|
SET m.seen = true
|
||||||
|
RETURN m { .* }
|
||||||
|
`
|
||||||
|
const setSeenTxResponse = await transaction.run(setSeenCypher, {
|
||||||
|
messageIds,
|
||||||
|
currentUserId,
|
||||||
|
})
|
||||||
|
const messages = await setSeenTxResponse.records.map((record) => record.get('m'))
|
||||||
|
return messages
|
||||||
|
})
|
||||||
|
try {
|
||||||
|
await writeTxResultPromise
|
||||||
|
// send subscription to author to updated the messages
|
||||||
|
return true
|
||||||
|
} finally {
|
||||||
|
session.close()
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
Message: {
|
Message: {
|
||||||
...Resolver('Message', {
|
...Resolver('Message', {
|
||||||
|
|||||||
@ -2,8 +2,14 @@
|
|||||||
# room: _RoomFilter
|
# room: _RoomFilter
|
||||||
# }
|
# }
|
||||||
|
|
||||||
|
enum _MessageOrdering {
|
||||||
|
createdAt_asc
|
||||||
|
createdAt_desc
|
||||||
|
}
|
||||||
|
|
||||||
type Message {
|
type Message {
|
||||||
id: ID!
|
id: ID!
|
||||||
|
indexId: Int!
|
||||||
createdAt: String
|
createdAt: String
|
||||||
updatedAt: String
|
updatedAt: String
|
||||||
|
|
||||||
@ -16,6 +22,10 @@ type Message {
|
|||||||
username: String! @cypher(statement: "MATCH (this)<-[:CREATED]-(user:User) RETURN user.name")
|
username: String! @cypher(statement: "MATCH (this)<-[:CREATED]-(user:User) RETURN user.name")
|
||||||
avatar: String @cypher(statement: "MATCH (this)<-[:CREATED]-(:User)-[:AVATAR_IMAGE]->(image:Image) RETURN image.url")
|
avatar: String @cypher(statement: "MATCH (this)<-[:CREATED]-(:User)-[:AVATAR_IMAGE]->(image:Image) RETURN image.url")
|
||||||
date: String! @cypher(statement: "RETURN this.createdAt")
|
date: String! @cypher(statement: "RETURN this.createdAt")
|
||||||
|
|
||||||
|
saved: Boolean
|
||||||
|
distributed: Boolean
|
||||||
|
seen: Boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
type Mutation {
|
type Mutation {
|
||||||
@ -23,8 +33,15 @@ type Mutation {
|
|||||||
roomId: ID!
|
roomId: ID!
|
||||||
content: String!
|
content: String!
|
||||||
): Message
|
): Message
|
||||||
|
|
||||||
|
MarkMessagesAsSeen(messageIds: [String!]): Boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
type Query {
|
type Query {
|
||||||
Message(roomId: ID!): [Message]
|
Message(
|
||||||
|
roomId: ID!,
|
||||||
|
first: Int
|
||||||
|
offset: Int
|
||||||
|
orderBy: [_MessageOrdering]
|
||||||
|
): [Message]
|
||||||
}
|
}
|
||||||
|
|||||||
@ -8,6 +8,7 @@
|
|||||||
:template-actions="JSON.stringify(templatesText)"
|
:template-actions="JSON.stringify(templatesText)"
|
||||||
:menu-actions="JSON.stringify(menuActions)"
|
:menu-actions="JSON.stringify(menuActions)"
|
||||||
:text-messages="JSON.stringify(textMessages)"
|
:text-messages="JSON.stringify(textMessages)"
|
||||||
|
:message-actions="messageActions"
|
||||||
:messages="JSON.stringify(messages)"
|
:messages="JSON.stringify(messages)"
|
||||||
:messages-loaded="messagesLoaded"
|
:messages-loaded="messagesLoaded"
|
||||||
:rooms="JSON.stringify(rooms)"
|
:rooms="JSON.stringify(rooms)"
|
||||||
@ -21,6 +22,7 @@
|
|||||||
@fetch-messages="fetchMessages($event.detail[0])"
|
@fetch-messages="fetchMessages($event.detail[0])"
|
||||||
:responsive-breakpoint="responsiveBreakpoint"
|
:responsive-breakpoint="responsiveBreakpoint"
|
||||||
:single-room="singleRoom"
|
:single-room="singleRoom"
|
||||||
|
show-reaction-emojis="false"
|
||||||
@show-demo-options="showDemoOptions = $event"
|
@show-demo-options="showDemoOptions = $event"
|
||||||
>
|
>
|
||||||
<div slot="menu-icon" @click.prevent.stop="$emit('close-single-room', true)">
|
<div slot="menu-icon" @click.prevent.stop="$emit('close-single-room', true)">
|
||||||
@ -91,9 +93,11 @@ export default {
|
|||||||
{
|
{
|
||||||
name: 'deleteRoom',
|
name: 'deleteRoom',
|
||||||
title: 'Delete Room',
|
title: 'Delete Room',
|
||||||
}, */
|
},
|
||||||
|
*/
|
||||||
],
|
],
|
||||||
messageActions: [
|
messageActions: [
|
||||||
|
/*
|
||||||
{
|
{
|
||||||
name: 'addMessageToFavorite',
|
name: 'addMessageToFavorite',
|
||||||
title: 'Add To Favorite',
|
title: 'Add To Favorite',
|
||||||
@ -102,6 +106,7 @@ export default {
|
|||||||
name: 'shareMessage',
|
name: 'shareMessage',
|
||||||
title: 'Share Message',
|
title: 'Share Message',
|
||||||
},
|
},
|
||||||
|
*/
|
||||||
],
|
],
|
||||||
templatesText: [
|
templatesText: [
|
||||||
{
|
{
|
||||||
@ -144,6 +149,10 @@ export default {
|
|||||||
showDemoOptions: true,
|
showDemoOptions: true,
|
||||||
responsiveBreakpoint: 600,
|
responsiveBreakpoint: 600,
|
||||||
singleRoom: !!this.singleRoomId || false,
|
singleRoom: !!this.singleRoomId || false,
|
||||||
|
messagePage: 0,
|
||||||
|
messagePageSize: 20,
|
||||||
|
roomPage: 0,
|
||||||
|
roomPageSize: 999, // TODO pagination is a problem with single rooms - cant use
|
||||||
selectedRoom: null,
|
selectedRoom: null,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -178,32 +187,48 @@ export default {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
fetchMessages({ room, options = {} }) {
|
async fetchMessages({ room, options = {} }) {
|
||||||
this.messagesLoaded = false
|
if (this.selectedRoom !== room.id) {
|
||||||
setTimeout(async () => {
|
this.messages = []
|
||||||
try {
|
this.messagePage = 0
|
||||||
const {
|
this.selectedRoom = room.id
|
||||||
data: { Message },
|
}
|
||||||
} = await this.$apollo.query({
|
this.messagesLoaded = options.refetch ? this.messagesLoaded : false
|
||||||
query: messageQuery(),
|
const offset = (options.refetch ? 0 : this.messagePage) * this.messagePageSize
|
||||||
variables: {
|
try {
|
||||||
roomId: room.id,
|
const {
|
||||||
},
|
data: { Message },
|
||||||
fetchPolicy: 'no-cache',
|
} = await this.$apollo.query({
|
||||||
})
|
query: messageQuery(),
|
||||||
this.messages = Message
|
variables: {
|
||||||
} catch (error) {
|
roomId: room.id,
|
||||||
this.messages = []
|
first: this.messagePageSize,
|
||||||
this.$toast.error(error.message)
|
offset,
|
||||||
}
|
},
|
||||||
this.messagesLoaded = true
|
fetchPolicy: 'no-cache',
|
||||||
|
})
|
||||||
|
|
||||||
this.selectedRoom = room
|
const msgs = []
|
||||||
})
|
;[...this.messages, ...Message].forEach((m) => {
|
||||||
|
msgs[m.indexId] = m
|
||||||
|
})
|
||||||
|
this.messages = msgs.filter(Boolean)
|
||||||
|
|
||||||
|
if (Message.length < this.messagePageSize) {
|
||||||
|
this.messagesLoaded = true
|
||||||
|
}
|
||||||
|
this.messagePage += 1
|
||||||
|
} catch (error) {
|
||||||
|
this.messages = []
|
||||||
|
this.$toast.error(error.message)
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
refetchMessage(roomId) {
|
refetchMessage(roomId) {
|
||||||
this.fetchMessages({ room: this.rooms.find((r) => r.roomId === roomId) })
|
this.fetchMessages({
|
||||||
|
room: this.rooms.find((r) => r.roomId === roomId),
|
||||||
|
options: { refetch: true },
|
||||||
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
async sendMessage(message) {
|
async sendMessage(message) {
|
||||||
@ -231,6 +256,12 @@ export default {
|
|||||||
query() {
|
query() {
|
||||||
return roomQuery()
|
return roomQuery()
|
||||||
},
|
},
|
||||||
|
variables() {
|
||||||
|
return {
|
||||||
|
first: this.roomPageSize,
|
||||||
|
offset: this.roomPage * this.roomPageSize,
|
||||||
|
}
|
||||||
|
},
|
||||||
update({ Room }) {
|
update({ Room }) {
|
||||||
if (!Room) {
|
if (!Room) {
|
||||||
this.rooms = []
|
this.rooms = []
|
||||||
|
|||||||
@ -2,10 +2,11 @@ import gql from 'graphql-tag'
|
|||||||
|
|
||||||
export const messageQuery = () => {
|
export const messageQuery = () => {
|
||||||
return gql`
|
return gql`
|
||||||
query ($roomId: ID!) {
|
query ($roomId: ID!, $first: Int, $offset: Int) {
|
||||||
Message(roomId: $roomId) {
|
Message(roomId: $roomId, first: $first, offset: $offset, orderBy: createdAt_desc) {
|
||||||
_id
|
_id
|
||||||
id
|
id
|
||||||
|
indexId
|
||||||
senderId
|
senderId
|
||||||
content
|
content
|
||||||
author {
|
author {
|
||||||
|
|||||||
@ -1,8 +1,8 @@
|
|||||||
import gql from 'graphql-tag'
|
import gql from 'graphql-tag'
|
||||||
|
|
||||||
export const roomQuery = () => gql`
|
export const roomQuery = () => gql`
|
||||||
query {
|
query Room($first: Int, $offset: Int) {
|
||||||
Room {
|
Room(first: $first, offset: $offset) {
|
||||||
id
|
id
|
||||||
roomId
|
roomId
|
||||||
roomName
|
roomName
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user