Unify calendar engines across /events and Custom Views #78
Open
opened 2026-05-25 11:23:29 +00:00 by mAi
·
2 comments
No Branch/Tag Specified
main
mai/planck/coder-b5-b6-train-share
mai/archimedes/fixer-port-engine
mai/maxwell/coder-b4-akte-mode
mai/lorenz/coder-b3-event-triggered
mai/euler/fixer-builder-add
mai/brunel/fixer-prod-500s-after-b1
mai/galileo/coder-b1-b2-mvp-train
mai/pasteur/fixer-pkg-litigationplann
mai/newton/coder-b0-scenario-db
mai/edison/inventor-prd-columnar
mai/knuth/coder-workflow-tracker
mai/atlas/inventor-extend-tools
mai/cronus/inventor-unified
mai/atlas/inventor-deadline-system
mai/atlas/inventor-followup-rules
mai/athena/consultant-deadline
mai/brunel/fixer-dark-mode-support
mai/knuth/coder-cronus-fristenrechn
mai/ritchie/coder-mig-153-proceeding
mai/atlas/inventor-proceeding
mai/cronus/inventor-fristenrechner
mai/curie/coder-mig152-clone-dedupe
mai/darwin/researcher-lexy-draft
mai/knuth/coder-dedupe-null
mai/cronus/coder-composer-slice-f
mai/cronus/coder-composer-slice-e
mai/cronus/coder-composer-slice-d
mai/curie/coder-slice-b6-url-rename
mai/curie/coder-slice-b5-go-rename
mai/cronus/coder-composer-slice-c
mai/curie/coder-slice-b4-destructive-drop
mai/cronus/coder-composer-slice-b
mai/cronus/coder-composer-slice-a
mai/cronus/inventor-prd-for
mai/knuth/coder-verfahrensablauf
mai/ritchie/coder-make-backup
mai/diesel/fixer-dark-mode-css
mai/curie/coder-slice-b3-read-cutover
mai/diesel/fixer-verfahrensablauf
mai/curie/coder-slice-b2-dual-write
mai/cronus/coder-slice-d-scenarios
mai/knuth/coder-backfill-applies
mai/hermes/gitster-verfahrensablauf
mai/cronus/coder-berufung-labels-refactor
mai/diesel/hotfix-2-mig-134-missing
mai/curie/coder-slice-b1-procedural-events
mai/cronus/coder-slice-c-upc-snapshot
mai/brunel/hotfix-rename-upc-apl
mai/cronus/coder-slice-b3-primary-party
mai/cronus/coder-slice-b2-catalog-query
mai/cronus/inventor-litigation-slice-b
mai/curie/researcher-slice-b-zero
mai/cronus/inventor-litigation
mai/artemis/gitster-remove-admin
mai/ritchie/coder-sort-post-trigger
mai/knuth/coder-conditional-label
mai/hermes/coder-verfahrensablauf
mai/brunel/rebase-121-conditional
mai/knuth/coder-conditional-rule
mai/hermes/gitster-dark-mode-fix
mai/ritchie/coder-submission-form
mai/artemis/gitster-re-surface
mai/brunel/fixer-views-any-filters
mai/cronus/coder-cicd-slice-a
mai/knuth/coder-wave-1-tier-1-rule
mai/ritchie/coder-upc-damages-add
mai/cronus/inventor-ci-cd-pre
mai/brunel/rebase-108-language
mai/hermes/gitster-admin-rules-list
mai/artemis/gitster-submission
mai/icarus/gitster-verfahrensablauf
mai/orpheus/gitster-search-input
mai/atlas/coder-event-card-choices-slice-ab
mai/hermes/gitster-date-range
mai/demeter/gitster-submission
mai/knuth/coder-hl-patents-style
mai/hermes/gitster-draft-editor
mai/atlas/inventor-per-event-card
mai/knuth/coder-deadline-rule-tier
mai/cronus/coder-procedural-events-slice-a
mai/hermes/gitster-deadline-form
mai/artemis/gitster-add-missing-i18n
mai/demeter/gitster-paliadin-chat
mai/brunel/wave0-tier0-deadline-fixes
mai/artemis/coder-docker-compose-yml
mai/icarus/coder-inbox-overhaul-slice-a
mai/atlas/coder-date-range-picker-slice-a
mai/brunel/fixer-de-inf-lg-cfi
mai/cronus/inventor-procedural
mai/hermes/gitster-event-type-modal
mai/cronus/coder-backup-mode
mai/curie/researcher-bulletproof
mai/hermes/gitster-draft-editor-focus-jump
mai/cronus/inventor-backup-mode
mai/hermes/gitster-submissions
mai/artemis/gitster-deadline-form
mai/brunel/fixer-submission-preview
mai/brunel/fixer-test-data-reset
mai/artemis/gitster-approval-withdraw
mai/demeter/gitster-events
mai/hermes/gitster-sidebar-loses
mai/hermes/gitster-browse-a
mai/brunel/fixer-submissions-demo
mai/icarus/inventor-inbox-overhaul
mai/atlas/inventor-symmetric-date
mai/artemis/gitster-demote-daten
mai/hermes/gitster-team-view-mailto
mai/knuth/coder-global-schriftsatze
mai/knuth/coder-schriftsatze
mai/ritchie/coder-author-demo-docx
mai/knuth/coder-add-schriftsatze
mai/knuth/coder-add-checklist
mai/knuth/coder-anchor-lookup-must
mai/tesla/dashboard-resize-clamp
mai/knuth/coder-demote-projekt
mai/knuth/coder-paliadin-chat
mai/knuth/coder-print-views
mai/knuth/coder-add-proceeding
mai/knuth/coder-submission
mai/ritchie/coder-extend-team-email
mai/knuth/coder-changelog-catch-up
mai/tesla/dashboard-overlap
mai/pasteur/fixercoder-dashboard
mai/newton/inventor-configurable
mai/dirac/inventorcoder-user
mai/gauss/inventorcoder-team-admin
mai/kepler/inventorcoder-project
mai/darwin/roadmap-ccr-en
mai/euler/coder-small-ux-polish
mai/darwin/fristenrechner-cleanup
mai/darwin/fixercoder-priority-bug
mai/leibniz/inventor-caldav-multi
mai/hertz/inventor-unified-modal
mai/archimedes/inventor-excel-data
mai/boltzmann/inventor-gap-tolerant
mai/copernicus/submission-slice-1
mai/fermi/interactive-session
mai/hertz/inventor-suggest-changes
mai/copernicus/inventor-submission
mai/mendel/test-strategy-slice-1
mai/mendel/inventor-test-strategy
mai/ampere/custom-views-improvements
mai/joule/mig-097-apply-huygens-s
mai/ohm/workstream-b-rename
mai/huygens/workstream-a-backfill
mai/kelvin/t-204-phase-2-proceeding
mai/bohr/ingest-t-paliad-203-rule
mai/curie/fristenrechner-gap
mai/maxwell/inbox-grey-out
mai/rutherford/slice-9-follow-up-b-re
mai/dirac/slice-9-follow-up-a
mai/bose/determinator-cascade-slice-3
mai/bose/determinator-cascade-slice-2
mai/bose/determinator-row-cascade
mai/lorenz/fristen-phase-3-slice-9
mai/curie/fristen-phase-3-slice-12
mai/planck/aichat-phase-b-paliad
mai/young/fristen-phase-3-slice-11b
mai/lorenz/fristen-phase-3-slice-11a
mai/lorenz/fristen-phase-3-slice-10
mai/lorenz/fristen-phase-3-slice-8
mai/lorenz/fristen-phase-3-slice-7
mai/lorenz/fristen-phase-3-slice-6
mai/lorenz/fristen-phase-3-slice-5
mai/lorenz/fristen-phase-3-slice-4
mai/lorenz/fristen-phase-3-slice-3
mai/lorenz/fristen-phase-3-slice-2
mai/lorenz/fristen-phase-3-slice-1
mai/pauli/fristen-phase2-design
mai/tesla/project-timeline-chart
mai/pauli/fristen-logic-audit
mai/pauli/determinator-b1-row-by
mai/noether/tools-cleanup-slice-1
mai/kelvin/inventor-tools-surface
mai/planck/paliadin-per-user-rls
mai/maxwell/bug-bundle-filterbar
mai/faraday/project-timeline-chart
mai/schroedinger/smarttimeline-slice-4
mai/bohr/smarttimeline-slice-3
mai/gauss/smarttimeline-slice-2
mai/riemann/filterbar-phase-2-slice
mai/lagrange/smarttimeline-design-the
mai/curie/researcher-determinator
mai/noether/collapse-regel-typ-on
mai/riemann/inventor-universal
mai/minkowski/project-level-our-side
mai/dirac/inventor-inline-paliadin
mai/feynman/fristenrechner
mai/minkowski/navbar-dashboard-reorg
mai/shannon/approval-rework
mai/einstein/consultant-deadline-data
mai/curie/researcher-upc-rop-audit
mai/noether/paliadin-real-claude
mai/noether/inventor-paliadin
mai/hilbert/inventor-approval-policy
mai/shannon/bug-frist-due-date
mai/fritz/bug-fristen-termine
mai/godel/inventor-projects-page
mai/fritz/bug-paliadin-chat
mai/kepler/inventor-profession-vs
mai/noether/inventor-paliadin-in-app
mai/fritz/bulk-team-email-send-to
mai/noether/inventor-local-chat-for
mai/noether/inventor-data-display
mai/fritz/bug-derived-team-members
mai/fritz/bug-sidebar-visibly
mai/noether/inventor-project
mai/shannon/bug-project-team-add
mai/cronus/inventor-dual-control
mai/fritz/bug-edit-mode-on
mai/cronus/inventor-holidays-per
mai/ritchie/phase-h-ai-deadline
No results found.
Milestone
No items
No Milestone
Projects
Clear projects
No project
Notifications
Due Date
No due date set.
Dependencies
No dependencies set.
Reference: m/paliad#78
Reference in New Issue
Block a user
Blocking a user prevents them from interacting with repositories, such as opening or commenting on pull requests or issues. Learn more about blocking a user.
No description provided.
Delete Branch "%!s()"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
m's report (2026-05-25 12:44)
(Companion issue for the date-range filter simplification is separate — inventor design phase.)
Context
This duplicates / reinforces #55 ("Align calendar-view rendering between Events/Termine page and Custom Views' calendar view type"). m has now reported it again as a live bug — pick this up now.
What to do
/events?view=calendar→ likelyfrontend/src/events.tsx+frontend/src/client/events.ts(calendar branch)frontend/src/client/views/calendar-view.tsor similar under the views directoryfrontend/src/components/CalendarView.tsxor similar) consumed by both/eventsand the Custom-View calendar.Files most likely touched
frontend/src/events.tsxfrontend/src/client/events.tsfrontend/src/client/views/calendar-view.ts(or whatever the Custom-Views calendar lives in)frontend/src/styles/global.css— consolidate any calendar-specific CSSHard rules
go build ./... && go test ./internal/... && cd frontend && bun run buildclean.mai/<worker>/calendar-engine-unify.Out of scope
Reporting
mai report completedwith branch + SHAs + the chosen canonical engine + verification path (open /events?view=calendar AND a Custom View with calendar layout → both render identically).When filing the commit comment on this issue, also reference / close out the duplicate concern in #55 if appropriate (don't close the issue — set the
donelabel instead per project convention).Investigation — calendar engines are already unified (t-paliad-224 / m/paliad#55)
Before touching code I checked the live state on paliad.de + the worktree. Both
/events?view=calendarand Custom-Viewshape=calendaralready render through the same engine —frontend/src/client/calendar/mount-calendar.ts. t-paliad-224 (m/paliad#55) landed this on 2026-05-20 and is deployed.Proof — both prod bundles ship the same
mountCalendarcodeDownloaded
https://paliad.de/assets/events.jsandhttps://paliad.de/assets/views.jsand grepped for every calendar-feature string. Every one appears exactly once in each bundle (i.e. one copy ofmountCalendar, no orphan):views-calendar-toolbarviews-calendar-cell--todayviews-calendar-pill--moreviews-calendar-week-gridviews-calendar-day-listviews-calendar-mobile-noticeviews-calendar-back-to-monthcal.day.no_entriesThe minified bodies around
views-calendar-toolbarare byte-equivalent modulo minifier variable letters (B vs _, D vs S, …). Same module compiled twice into two pages.Source-side
frontend/src/client/events.ts:11importsmountCalendarfrom./calendar/mount-calendar.frontend/src/client/events.ts:605callsmountCalendar(host, items, { urlState: true, defaultView: "month" }).frontend/src/client/views/shape-calendar.ts:12callsmountCalendar(host, items, { defaultView: render.calendar?.default_view ?? "month", urlState: true })— same module, same options.renderCalendar()/openCalPopup()/events-cal-dot-*code from before t-224 is gone (verified bygrep -rn frist-cal\|events-calreturning nothing infrontend/src/)..views-calendar-*remains for the full-page calendar..frist-cal-*was deleted.What can still feel different to a user — and isn't engine
/events?view=calendarcalls/api/eventsand only returns rows of kinddeadlineandappointment— so the pill palette only ever shows the lime-deadline + blue-appointment dots. A Custom-View withsources=[deadline, appointment, project_event, approval_request]calls/api/views/<slug>/runand can render purpleproject_eventand amberapproval_requestpills too. Same engine, different data.client/filter-bar/. Both surfaces sit above an identical.views-calendarelement.defaultView: "month"; Custom Views honourrender_spec.calendar.default_view(defaults tomonth, but a saved view can pin toweek). If your view's spec stored a non-month default, you'd land on a different initial view even though the engine is the same.Recommendation
The unify-the-engine work is done; the canonical engine is
mount-calendar.tsand the orphans are removed. No code change shipped on this branch — there's nothing left to consolidate without first knowing what specific divergence you're seeing.Could you post a screenshot of
/events?type=appointment&view=calendar&cal_view=month&cal_date=2026-07-01side-by-side with your custom view at the same anchor? If a real visual diff is still present I'll dig into it; if it turns out to be the data-source / chrome difference above, we'd want a different scope (e.g. let/events?view=calendarwiden to includeproject_event+approval_request, or align the filter chrome).No
donelabel set — re-labeling toneeds-reviewand leaving the floor with you.— gitster (demeter)
Re-verification — engine still unified, build clean (hermes shift)
Picked this up after demeter's investigation on 2026-05-25 (#78 (comment)). Re-verified the state at the current main HEAD (
d1d0cf9) — same conclusion:/events?view=calendarand Custom-Viewshape=calendargo through one engine,frontend/src/client/calendar/mount-calendar.ts, called from both surfaces.Code-side proof (re-checked now)
/eventsKalender chipfrontend/src/client/events.ts:11imports +:622callsmountCalendar(host, items, { urlState: true, defaultView: "month" })shape=calendarfrontend/src/client/views/shape-calendar.ts:12callsmountCalendar(host, items, { defaultView: render.calendar?.default_view ?? "month", urlState: true })No
.frist-cal-*/.events-cal-*orphan classes anywhere infrontend/src(grep -rn). Only one orphan-y identifier remains —.events-calendar-wrap— and that's just the host<div>that mountCalendar paints into on /events. It's a 2-line margin rule (global.css:13353), not a separate engine.Bundle-side proof (locally built + downloaded prod)
Every calendar-engine string appears exactly once per bundle in both my local
bun run buildoutput and the current prod bundles athttps://paliad.de/assets/{events,views}.js:views-calendar-toolbarviews-calendar-cell--todayviews-calendar-pill--moreviews-calendar-week-gridviews-calendar-day-listviews-calendar-mobile-noticeviews-calendar-back-to-monthdashboard-cal-*(any)The
dashboard-cal-*row is the interesting one — see next section.Build hygiene (per issue hard rules)
go build ./...— cleango test ./internal/...— auth/branding/calc/changelog/db/handlers/services allokcd frontend && bun run build— clean (2892 keys, data-i18n attributes clean, dist/ written)Nothing to ship on
mai/hermes/calendar-engine-unify— branch is empty of commits.The only other calendar engine in the codebase
Grepping
frontend/srcend-to-end, there's one other thing that paints a calendar:renderMiniCalendar()infrontend/src/client/dashboard.ts:602. It's the mini multi-month dot grid that the dashboard widgets (upcoming-deadlines,upcoming-appointments) render when their per-widget view is set tocalendar. DOM is.dashboard-cal-*, not.views-calendar-*. Very different UX:/events+/views/{slug})+Nmore, each dot is the link?cal_view=,?cal_date=frontend/src/client/calendar/mount-calendar.ts(579 LoC)frontend/src/client/dashboard.ts:602-680(~80 LoC)It is a deliberately separate engine because the widget needs a glanceable, fixed-height card; mountCalendar's toolbar + full pills would not fit. Database check: no user currently has any widget set to view=
calendar(SELECT … FROM paliad.user_dashboard_layouts WHERE layout_json::text ILIKE '%calendar%'→ empty), so this isn't what you're seeing in your example URL either.What could still look different — perceptually
/events?type=appointment&view=calendaronly renders rows of kindappointment(one pill colour). A Custom-View withsources=[deadline, appointment, project_event, approval_request]renders up to 4 pill colours on the same date. Same engine, different inputs..views-calendar-*element is identical.defaultView: "month". Custom Views honourrender_spec.calendar.default_view— a saved view can pin toweek..events-calendar-wraphasmargin: 0.25rem 0 1rem;.views-shape-hosthasmargin-top: 16px. Cosmetic only — could be aligned in a separate one-line PR if you want them visually flush.Where this leaves us
The "unify the engines" work was already shipped by t-paliad-224 / m/paliad#55 (2026-05-20). I won't ship a no-op commit on
mai/hermes/calendar-engine-unify— the branch is clean and the worktree retires here.If you still see a visual diff between the two URLs in your report, the most useful thing for the next worker would be two screenshots at the same
cal_date=2026-07-01anchor — one of/events?type=appointment&view=calendar&cal_view=month&cal_date=2026-07-01and one of whatever Custom-View URL you're comparing against. Then either:renderMiniCalendarintomountCalendarwith a "mini" mode (non-trivial — different UX contract), orLabels: leaving as-is (only
deferred/doneexist on this repo, noneeds-review). Per project convention I won't close the issue; flip todonewhen you've confirmed there's nothing left to chase.— gitster (hermes), branch
mai/hermes/gitster-unify-calendar, no commits.Commit: n/a (no-op; verification only).