feat: add emoji reaction support for files
All checks were successful
Build and Publish Docker Image / deploy (push) Successful in 1m46s
All checks were successful
Build and Publish Docker Image / deploy (push) Successful in 1m46s
- Implement `ReactionService` to manage file reactions in the database.
- Add `POST /d/{boxID}/f/{fileID}/react` endpoint to handle user reactions.
- Add `GET /emoji/{pack}/{file}` endpoint to serve custom emoji assets.
- Support loading custom emoji packs dynamically from the data directory.
- Update README with instructions on configuring emoji reaction packs.
This commit is contained in:
@@ -65,6 +65,242 @@
|
||||
|
||||
.file-card {
|
||||
position: relative;
|
||||
padding-bottom: 2.6rem;
|
||||
}
|
||||
|
||||
.file-reaction-dock {
|
||||
position: absolute;
|
||||
right: 0.65rem;
|
||||
bottom: 0.55rem;
|
||||
z-index: 2;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: flex-end;
|
||||
max-width: calc(100% - 1.3rem);
|
||||
gap: 0.35rem;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.file-reactions {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: flex-end;
|
||||
min-width: 0;
|
||||
gap: 0.25rem;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.reaction-pill {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 0.2rem;
|
||||
min-height: 1.6rem;
|
||||
padding: 0.16rem 0.38rem;
|
||||
border: 1px solid color-mix(in srgb, var(--border) 84%, var(--primary));
|
||||
border-radius: 999px;
|
||||
background: color-mix(in srgb, var(--card) 88%, #000);
|
||||
color: var(--foreground);
|
||||
font-size: 0.75rem;
|
||||
font-weight: 700;
|
||||
box-shadow: 0 8px 22px rgba(0, 0, 0, 0.24);
|
||||
pointer-events: auto;
|
||||
}
|
||||
|
||||
.reaction-pill img {
|
||||
width: 1rem;
|
||||
height: 1rem;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.reaction-button {
|
||||
width: 2.1rem;
|
||||
height: 2.1rem;
|
||||
display: inline-grid;
|
||||
place-items: center;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 999px;
|
||||
background: color-mix(in srgb, var(--card) 92%, #000);
|
||||
color: var(--foreground);
|
||||
opacity: 0;
|
||||
transform: translateY(0.3rem) scale(0.94);
|
||||
box-shadow: 0 12px 30px rgba(0, 0, 0, 0.32);
|
||||
transition: opacity 150ms ease, transform 150ms ease, border-color 150ms ease, background 150ms ease;
|
||||
pointer-events: auto;
|
||||
}
|
||||
|
||||
.reaction-button svg {
|
||||
width: 1.15rem;
|
||||
height: 1.15rem;
|
||||
fill: none;
|
||||
stroke: currentColor;
|
||||
stroke-width: 1.9;
|
||||
stroke-linecap: round;
|
||||
stroke-linejoin: round;
|
||||
}
|
||||
|
||||
.file-card:hover .reaction-button,
|
||||
.file-card:focus-within .reaction-button,
|
||||
.reaction-button:focus-visible {
|
||||
opacity: 1;
|
||||
transform: translateY(0) scale(1);
|
||||
}
|
||||
|
||||
.reaction-button:hover,
|
||||
.reaction-button:focus-visible {
|
||||
border-color: var(--primary);
|
||||
background: var(--primary);
|
||||
color: var(--primary-foreground);
|
||||
}
|
||||
|
||||
.reaction-picker {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 70;
|
||||
width: min(21rem, calc(100vw - 1rem));
|
||||
}
|
||||
|
||||
html.reaction-picker-open,
|
||||
html.reaction-picker-open body {
|
||||
overflow: hidden;
|
||||
touch-action: none;
|
||||
}
|
||||
|
||||
.reaction-picker[hidden] {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.reaction-picker.is-mobile {
|
||||
inset: 0;
|
||||
width: auto;
|
||||
height: 100dvh;
|
||||
display: grid;
|
||||
place-items: end center;
|
||||
overflow: hidden;
|
||||
padding: 0.75rem 0.75rem max(1.5rem, env(safe-area-inset-bottom));
|
||||
background: rgba(0, 0, 0, 0.54);
|
||||
}
|
||||
|
||||
.reaction-picker-panel {
|
||||
overflow: hidden;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: var(--radius);
|
||||
background: color-mix(in srgb, var(--card) 97%, #000);
|
||||
box-shadow: 0 26px 70px rgba(0, 0, 0, 0.52);
|
||||
}
|
||||
|
||||
.reaction-picker.is-mobile .reaction-picker-panel {
|
||||
width: min(100%, 34rem);
|
||||
height: 75dvh;
|
||||
max-height: 75dvh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.reaction-picker-head {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 0.75rem;
|
||||
padding: 0.7rem;
|
||||
border-bottom: 1px solid var(--border);
|
||||
}
|
||||
|
||||
.reaction-picker-close {
|
||||
min-height: 2rem;
|
||||
padding: 0.3rem 0.55rem;
|
||||
font-size: 0.75rem;
|
||||
}
|
||||
|
||||
.reaction-picker-tabs {
|
||||
display: flex;
|
||||
gap: 0.35rem;
|
||||
overflow-x: auto;
|
||||
padding: 0.55rem 0.7rem 0;
|
||||
}
|
||||
|
||||
.reaction-tab {
|
||||
flex: 0 0 auto;
|
||||
min-height: 1.8rem;
|
||||
padding: 0.25rem 0.55rem;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 999px;
|
||||
background: var(--muted);
|
||||
color: var(--muted-foreground);
|
||||
font-size: 0.75rem;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.reaction-tab.is-active {
|
||||
border-color: var(--primary);
|
||||
background: var(--primary);
|
||||
color: var(--primary-foreground);
|
||||
}
|
||||
|
||||
.reaction-search {
|
||||
display: block;
|
||||
padding: 0.55rem 0.7rem;
|
||||
}
|
||||
|
||||
.reaction-search input {
|
||||
width: 100%;
|
||||
min-height: 2.15rem;
|
||||
padding: 0.35rem 0.55rem;
|
||||
}
|
||||
|
||||
.reaction-grid-wrap {
|
||||
max-height: 18rem;
|
||||
overflow: auto;
|
||||
padding: 0 0.7rem 0.7rem;
|
||||
}
|
||||
|
||||
.reaction-picker.is-mobile .reaction-grid-wrap {
|
||||
max-height: none;
|
||||
flex: 1;
|
||||
overscroll-behavior: contain;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
}
|
||||
|
||||
.reaction-grid {
|
||||
display: none;
|
||||
grid-template-columns: repeat(8, minmax(0, 1fr));
|
||||
gap: 0.25rem;
|
||||
}
|
||||
|
||||
.reaction-grid.is-active {
|
||||
display: grid;
|
||||
}
|
||||
|
||||
.reaction-picker.is-mobile .reaction-grid {
|
||||
grid-template-columns: repeat(6, minmax(0, 1fr));
|
||||
}
|
||||
|
||||
.reaction-emoji {
|
||||
aspect-ratio: 1;
|
||||
display: grid;
|
||||
place-items: center;
|
||||
min-width: 0;
|
||||
padding: 0.18rem;
|
||||
border: 1px solid transparent;
|
||||
border-radius: calc(var(--radius) - 0.25rem);
|
||||
background: transparent;
|
||||
}
|
||||
|
||||
.reaction-emoji:hover,
|
||||
.reaction-emoji:focus-visible {
|
||||
border-color: var(--border);
|
||||
background: var(--accent);
|
||||
}
|
||||
|
||||
.reaction-emoji[hidden] {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.reaction-emoji img {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: block;
|
||||
object-fit: contain;
|
||||
}
|
||||
|
||||
.thumb-link {
|
||||
|
||||
Reference in New Issue
Block a user