Live preview your page changes as you make them.

pip install wagtail-livepreview==0.1


Wagtail Live Preview

Wagtail Live Preview lets you view your page changes as you make them in the Wagtail Admin.

Using React or Vue? This won't work for you, nor was it designed to. This Live Preview package is designed for simple Wagtail websites.

How it works

Tell it how often to save a snapshot of the page you're working on and how often to poll for updates in the Live Preview.

It does not piggy back on Wagtails Revision system, but you can tell it to save a page revision every x number of saves so you never accidentally lose your work (or if you just want to log your progress and maybe revert back to a previous content iteration).

The package itself is called wagtail-livepreview to let everyone know this is a Wagtail specific package. But the code references livepreview instead of wagtail_livepreview as to not confuse Wagtail features with what's in this package.


  1. Install the package
pip install wagtail-livepreview
  1. Add it to your INSTALLED_APPS above the 'wagtail.admin' app.
    # ...
    # ...
  1. Add {% load livepreview_tags %} to your base.html template. And add {% livepreview_js %} right above your </body> tag in base.html
{% load static wagtailuserbar livepreview_tags %}

<!DOCTYPE html>
<html class="no-js" lang="en">

    <body class="{% block body_class %}{% endblock %}">

        {% livepreview_js %}
  1. You'll need to apply migrations.
python manage.py migrate


You can take an action before and after a Live Preview using a generic Wagtail hook.

def after_live_preview_save(request, page):
    """Event to happen before the live preview is served."""

def before_live_preview_save(request, page):
    """Event to happen after the live preview is served."""

Caution: It's a bad idea to provide a process intensive task in these hooks since these hooks may end up being called as frequently as once per second. It might be best to offload your tasks in these hooks to a task runner.


Checking if a view is a Live Preview or not

You'll want to adjust your template so you aren't triggering your analytics every second. You can prevent this with:

{% if not livepreview %}
    .. analytics in here
{% else %}
    <div id="warning">This is a live preview</div>
{% endif %}

You can also use {{ request.livepreview }} in your template to check against the request.


Global Settings

# base settings.py

# How often (in milliseconds) should the livepreview check for page updates? Default is 1000ms.
# If you'd like to turn on auto-revision saving every x number of Live Preview saves, set this as True. Default is False.
# How many Live Preview saves should happen before a new revision is automatically saved? Default is 10. Requires LIVEPREVIEW_SAVE_AS_REVISIONS = True.
# Render Live Previews into a temporary file, and attempt to serve that file. Default is true.
# If True, LIVEPREVIEW_TIMEOUT can be as low as 250ms.
# If False, the minimum LIVEPREVIEW_TIMEOUT is 1000ms.

Model Settings

You can disable Live Preview for specific page models. For example, you might have a simple Blog Index Page with just a title field. Or a page that redirects to another page. In these scenarios you might not want Live Preview enabled.

class YourPage(Page):
    # ...
    LIVEPREVIEW_DISABLED = True  # Disable Live Preview on a per-model basis


Feel free to open a PR. I'm not overly picky about how things are done as long as it works and it's kept relatively simple.

The JavaScript can use a lot of improvements (small and large) so that'd be a great place to start.


If anyone wants to get involved and pick off a few of these tasks, that'd be awesome. But no pressure.

  • Implement proper http protocol responses
  • Remove jQuery in favor of vanilla JS
  • Put the iframe in the content_panels ONLY tab.
  • Make the iframe be 100% height of the content section
  • Add better JavaScript error handling and events
  • When LIVEPREVIEW_USE_FILE_RENDERING is enabled, save the LivePreview file with minified html (Not using middleware, and respects spacing in elements like <pre> and <textarea>)