A web-based presentation tool built with Lively. Write your slides in Markdown, present them in the browser, and control everything from a separate presenter display.
- Markdown slides with YAML frontmatter for metadata and template selection.
- Presenter display with current slide, next slide preview, notes, and timing.
- Real-time sync between display and presenter via WebSockets.
- Code highlighting with @socketry/syntax, including animated focus regions for code walkthroughs.
- Multiple templates — title, section, two-column, code, translation, image, and default.
- Timing and pacing — per-slide duration metadata with elapsed/remaining time and pacing indicators.
- Full-screen support — press
Fon the display view. - Keyboard navigation — arrow keys, space, Page Up/Down.
Please see the project documentation for more details.
-
Getting Started - This guide explains how to use
presentlyto create and deliver web-based presentations using Markdown slides. -
Animating Slides - This guide explains how to animate content within slides using the
morphtransition and the slide scripting system.
Please see the project releases for all releases.
- Add optional
translationsection to thedefaulttemplate.
- Rework build effects to use direct CSS class animation rather than
view-transition-class.build-fade,build-fly-up, etc. are now regular@keyframesclasses applied to the revealed element, rather than view transition pseudo-element selectors. This decouples in-slide sequential animation from slide-level morph transitions. - Rename
SlideElements#buildtoSlideElements#showfor clarity —boxes.show(0)/boxes.show(3)more clearly describes the outcome from the audience's perspective. - Add
SlideElements#builder(options)— returns aSlideBuilderwith default options (group, effect) and a cached position, so callers can usenext()instead of tracking the step count manually. - Add
SlideBuilder#show(count, overrides)— set visibility state to an arbitrary position. Returns aPromisethat resolves when the reveal animation completes (or immediately when no effect is given). - Add
SlideBuilder#next(overrides)— reveal the next element with the builder's default effect, optionally overridden per call. O(1): only touches the single newly revealed element. Returns aPromise. - Add
SlideBuilder#play(interval, callback)— reveals all remaining elements in sequence withintervalmilliseconds between each. An optional callback is invoked after each reveal; returnfalseto stop playback early. Requires the builder to be created viaslide.find(...).builder()so timeouts are tracked and cancelled on slide change. - Add
SlideBuilder#finishedgetter — returnstruewhen all elements have been revealed. - Add
Slide#after(delay, callback)— schedules a callback after a delay in milliseconds and returns aSlideChain. Subsequent.after(delay, callback)calls on the chain fire relative to the previous step, making sequential reveal timing easy to read and adjust. - Add
Slide#setTimeout(callback, delay)— a tracked replacement for the globalsetTimeout. All timeouts registered this way are automatically cancelled when the slide changes, preventing stale callbacks from firing after navigation. The globalsetTimeoutin slide scripts is shadowed by this method automatically. - Add
Slide#cancelTimeouts()— cancels all pending timeouts registered by the slide's script. Called automatically by the presentation engine on every slide change.
- Add
bake presently:slides:speakerstask to print a timing breakdown grouped by speaker. Each speaker's slides are listed in presentation order with individual and total durations, making it easy to balance talk time in multi-speaker presentations. Slides without aspeakerkey are grouped under(no speaker).
- Add optional
speakerfront matter key to slides. When present, the current speaker's name is shown in the timing bar. If the next slide has a different speaker, a handoff indicator (e.g.→ Next Speaker) is shown alongside, giving presenters an at-a-glance cue for tag-team talks.
- Add
bake presently:slides:notestask to extract all presenter notes into a single Markdown document, with each slide's file path as a heading. Useful for reviewing or sharing speaker notes outside of the presentation. - Presenter notes are now kept as a Markdown AST internally and rendered to HTML on demand, so the notes you write are faithfully round-tripped rather than converted to HTML at parse time.
- Add
diagramtemplate with aposition: relativecontainer — direct<div>children areposition: absoluteby default for free-form layouts. - All slide templates now have
position: relativeon the slide inner container, allowing absolutely positioned overlays in any template. - Add slide scripting: a fenced
```javascript ```block at the end of presenter notes is extracted and executed in the browser after each slide renders. The script receives aslideobject scoped to the slide body. - Add
Slide#find(selector)— a pure CSS selector query returning aSlideElementscollection with no side effects. - Add
SlideElements#build(n, options)— shows the firstnmatched elements, hides the rest, and assignsview-transition-namefor morph transition matching. Acceptsgroup(name prefix) andeffect(entry animation) options. - Add build effects via
view-transition-class:fade,fly-left,fly-right,fly-up,fly-down,scale. Requires Chromium 125+; degrades gracefully to instant appear in other browsers. - Rename
magic-movetransition tomorph. - Italic text in presenter notes is styled in amber to distinguish stage directions from spoken words.
- Add transitions guide and animating slides guide to documentation.
- Use Markly's native front matter parser (
Markly::FRONT_MATTER) instead of manual string splitting, parsing each slide document once and extracting front matter directly from the AST. - Use the last
---hrule in the AST as the presenter notes separator, so earlier---dividers in slide content are preserved correctly. - Add support for Mermaid diagrams in slides.
- Initial release.
- Slide files are Markdown with YAML front matter for metadata (
template,duration,title,skip,marker,transition,focus). - Slide content is split into named sections by top-level headings, rendered to HTML via Markly.
- Presenter notes are separated from slide content by a
---divider. - Magic move transitions between slides.
- Navigation control in the presenter view.
- Code highlighting with line-range focus support.
- Live state synchronisation between display and presenter views over WebSockets.
- lively — The real-time application framework that powers Presently.
- falcon — The web server used to host presentations.
- syntax-js — Syntax highlighting for code slides.
- markly — CommonMark parser used for slide content.
We welcome contributions to this project.
- Fork it.
- Create your feature branch (
git checkout -b my-new-feature). - Commit your changes (
git commit -am 'Add some feature'). - Push to the branch (
git push origin my-new-feature). - Create new Pull Request.
To run the test suite:
bundle exec susTo make a new release:
bundle exec bake gem:release:patch # or minor or majorIn order to protect users of this project, we require all contributors to comply with the Developer Certificate of Origin. This ensures that all contributions are properly licensed and attributed.
This project is best served by a collaborative and respectful environment. Treat each other professionally, respect differing viewpoints, and engage constructively. Harassment, discrimination, or harmful behavior is not tolerated. Communicate clearly, listen actively, and support one another. If any issues arise, please inform the project maintainers.
