Files
DiscordClone/convex/schema.ts
Bryan1029384756 b7a4cf4ce8
All checks were successful
Build and Release / build-and-release (push) Successful in 13m12s
feat(ui): add Button, Modal, Spinner, Toast, and Tooltip components with styles
- Implemented Button component with various props for customization.
- Created Modal component with header, content, and footer subcomponents.
- Added Spinner component for loading indicators.
- Developed Toast component for displaying notifications.
- Introduced Tooltip component for contextual hints with keyboard shortcuts.
- Added corresponding CSS modules for styling each component.
- Updated index file to export new components.
- Configured TypeScript settings for the UI package.
2026-04-14 09:02:14 -05:00

214 lines
6.4 KiB
TypeScript

import { defineSchema, defineTable } from "convex/server";
import { v } from "convex/values";
export default defineSchema({
userProfiles: defineTable({
username: v.string(),
clientSalt: v.string(),
encryptedMasterKey: v.string(),
hashedAuthKey: v.string(),
publicIdentityKey: v.string(),
publicSigningKey: v.string(),
encryptedPrivateKeys: v.string(),
isAdmin: v.boolean(),
status: v.optional(v.string()),
displayName: v.optional(v.string()),
avatarStorageId: v.optional(v.id("_storage")),
aboutMe: v.optional(v.string()),
customStatus: v.optional(v.string()),
joinSoundStorageId: v.optional(v.id("_storage")),
accentColor: v.optional(v.string()),
}).index("by_username", ["username"]),
categories: defineTable({
name: v.string(),
position: v.number(),
}).index("by_position", ["position"]),
channels: defineTable({
name: v.string(),
type: v.string(), // 'text' | 'voice' | 'dm'
categoryId: v.optional(v.id("categories")),
topic: v.optional(v.string()),
position: v.optional(v.number()),
}).index("by_name", ["name"])
.index("by_category", ["categoryId"]),
messages: defineTable({
channelId: v.id("channels"),
senderId: v.id("userProfiles"),
ciphertext: v.string(),
nonce: v.string(),
signature: v.string(),
keyVersion: v.number(),
replyTo: v.optional(v.id("messages")),
editedAt: v.optional(v.number()),
pinned: v.optional(v.boolean()),
}).index("by_channel", ["channelId"])
.index("by_channel_pinned", ["channelId", "pinned"])
.index("by_sender", ["senderId"]),
messageReactions: defineTable({
messageId: v.id("messages"),
userId: v.id("userProfiles"),
emoji: v.string(),
})
.index("by_message", ["messageId"])
.index("by_message_user_emoji", ["messageId", "userId", "emoji"])
.index("by_user", ["userId"]),
channelKeys: defineTable({
channelId: v.id("channels"),
userId: v.id("userProfiles"),
encryptedKeyBundle: v.string(),
keyVersion: v.number(),
})
.index("by_channel", ["channelId"])
.index("by_user", ["userId"])
.index("by_channel_and_user", ["channelId", "userId"]),
roles: defineTable({
name: v.string(),
color: v.string(),
position: v.number(),
permissions: v.any(), // JSON object of permissions
isHoist: v.boolean(),
}),
userRoles: defineTable({
userId: v.id("userProfiles"),
roleId: v.id("roles"),
})
.index("by_user", ["userId"])
.index("by_role", ["roleId"])
.index("by_user_and_role", ["userId", "roleId"]),
invites: defineTable({
code: v.string(),
encryptedPayload: v.string(),
createdBy: v.id("userProfiles"),
maxUses: v.optional(v.number()),
uses: v.number(),
expiresAt: v.optional(v.number()), // timestamp
keyVersion: v.number(),
}).index("by_code", ["code"])
.index("by_creator", ["createdBy"]),
dmParticipants: defineTable({
channelId: v.id("channels"),
userId: v.id("userProfiles"),
})
.index("by_channel", ["channelId"])
.index("by_user", ["userId"]),
typingIndicators: defineTable({
channelId: v.id("channels"),
userId: v.id("userProfiles"),
username: v.string(),
expiresAt: v.number(), // timestamp
}).index("by_channel", ["channelId"])
.index("by_user", ["userId"]),
voiceStates: defineTable({
channelId: v.id("channels"),
userId: v.id("userProfiles"),
username: v.string(),
isMuted: v.boolean(),
isDeafened: v.boolean(),
isScreenSharing: v.boolean(),
isServerMuted: v.boolean(),
watchingStream: v.optional(v.id("userProfiles")),
lastHeartbeat: v.optional(v.number()),
})
.index("by_channel", ["channelId"])
.index("by_user", ["userId"]),
channelReadState: defineTable({
userId: v.id("userProfiles"),
channelId: v.id("channels"),
lastReadTimestamp: v.number(),
})
.index("by_user", ["userId"])
.index("by_channel", ["channelId"])
.index("by_user_and_channel", ["userId", "channelId"]),
serverSettings: defineTable({
serverName: v.optional(v.string()),
afkChannelId: v.optional(v.id("channels")),
afkTimeout: v.number(), // seconds (default 300 = 5 min)
iconStorageId: v.optional(v.id("_storage")),
}),
customEmojis: defineTable({
name: v.string(),
storageId: v.id("_storage"),
uploadedBy: v.id("userProfiles"),
// `true` for animated (GIF / APNG) uploads so the settings UI
// can split Static vs Animated in separate sections. Optional
// so existing rows without the flag still validate — they
// surface as static in the UI by default.
animated: v.optional(v.boolean()),
createdAt: v.number(),
}).index("by_name", ["name"])
.index("by_uploader", ["uploadedBy"]),
polls: defineTable({
channelId: v.id("channels"),
createdBy: v.id("userProfiles"),
question: v.string(),
options: v.array(
v.object({
id: v.string(),
text: v.string(),
}),
),
allowMultiple: v.boolean(),
disclosed: v.boolean(),
closed: v.boolean(),
closesAt: v.optional(v.number()),
createdAt: v.number(),
})
.index("by_channel", ["channelId"])
.index("by_creator", ["createdBy"]),
pollVotes: defineTable({
pollId: v.id("polls"),
userId: v.id("userProfiles"),
optionIds: v.array(v.string()),
votedAt: v.number(),
})
.index("by_poll", ["pollId"])
.index("by_poll_and_user", ["pollId", "userId"]),
pollReactions: defineTable({
pollId: v.id("polls"),
userId: v.id("userProfiles"),
emoji: v.string(),
})
.index("by_poll", ["pollId"])
.index("by_poll_user_emoji", ["pollId", "userId", "emoji"])
.index("by_user", ["userId"]),
savedMedia: defineTable({
userId: v.id("userProfiles"),
// Convex storage URL — also the dedupe key for a single user.
url: v.string(),
kind: v.string(), // 'image' | 'video' | 'audio'
filename: v.string(),
mimeType: v.optional(v.string()),
width: v.optional(v.number()),
height: v.optional(v.number()),
size: v.optional(v.number()),
// Re-post path: keep the per-file AES key + iv so the same
// attachment metadata can be embedded in a future message
// without re-uploading. The file stays encrypted in storage —
// saving just bookmarks the metadata.
encryptionKey: v.string(),
encryptionIv: v.string(),
savedAt: v.number(),
})
.index("by_user", ["userId"])
.index("by_user_and_url", ["userId", "url"]),
});