Metadata-Version: 2.1
Name: django-commoncontent
Version: 0.3.0
Summary: basic data model and templates for Django's generic views to produce a decent-looking website with minimal effort
Author-email: Vince Veselosky <vince@control-escape.com>
Project-URL: Homepage, https://github.com/veselosky/django-commoncontent
Project-URL: Bug Tracker, https://github.com/veselosky/django-commoncontent/issues
Classifier: Programming Language :: Python :: 3
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Operating System :: OS Independent
Requires-Python: >=3.8
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: django-bootstrap-icons>=0.7.8
Requires-Dist: django-imagekit>=5.0.0
Requires-Dist: django-sitevars>=1.0.2
Requires-Dist: django-taggit>=5.0.0
Requires-Dist: django<5.3,>=4.2
Requires-Dist: pillow>=10.1.0
Requires-Dist: pygments>=2.18.0
Requires-Dist: python-dateutil>=2.8.1
Requires-Dist: pyyaml>=6.0.1
Requires-Dist: mistletoe>=1.3.0
Provides-Extra: dev
Requires-Dist: django-debug-toolbar>=4.4.6; extra == "dev"
Requires-Dist: django-environ>=0.11.2; extra == "dev"
Requires-Dist: django-extensions>=3.2.3; extra == "dev"
Requires-Dist: django-tinymce>=4.1.0; extra == "dev"
Requires-Dist: djlint>=1.34.1; extra == "dev"
Requires-Dist: docutils>=0.20.1; extra == "dev"
Requires-Dist: pre-commit>=3.5.0; extra == "dev"
Requires-Dist: pytest>=8.3.2; extra == "dev"
Requires-Dist: pytest-cov>=5.0.0; extra == "dev"
Requires-Dist: pytest-django>=4.8.0; extra == "dev"

# Common Content: A reusable content app for Django

Common Content provides models, views, and templates for common web content in a
reusable Django app, giving you a blog-like web site out of the box.

Django ships with `flatpages`, the most simplistic content model possible. Third party
tools like [Wagtail](https://wagtail.org/) or [DjangoCMS](https://www.django-cms.org/)
provide flexible and complex content management workflows, but require a lot of custom
code. Common Content is designed to fill the gap between these options. If you need
pages and feeds for your Django site, but don't need (or want) the power and complexity
of a full content management system, Common Content may be the right tool.

Common Content includes a set of templates using the
[Bootstrap CSS framework](https://getbootstrap.com) based on the examples found on the
bootstrap website. The templates work with any objects that implement the common data
model. They can easily be coupled with Django's generic class-based views to produce a
clean and professional site quickly and easily. If you use the included concrete models
with the default templates and the included views, you'll be starting with a basic
blog-style site.

## Project Status: Alpha

Common Content is still Alpha-level software. It has many completed features that are
usable in production. However, test coverage is not yet complete, the feature set is
still growing, and there are likely to be breaking changes before the software reaches a
stable 1.0 release. Be aware that migrating to newer versions may require some work.

## Included Content Models, Views, and Templates

Common Content defines a data model based on [Schema.org](https://schema.org), extended
in some places using fields from the [open graph protocol](https://ogp.me), or other web
standards. It provides concrete Django models for common web objects, as well as an
abstract CreativeWork and BasePage model for extending the available page types.

Common Content provides the following Content:

- `HomePage` - A dedicated model for site home pages. Sites can have multiple HomePages
  with different publication dates, allowing for scheduled updates.
- Generic `Page` - A model for evergreen pages like "About".
- `Article` - Like a Post in Wordpress, the Article is the main content type of the
  site. Articles can also be linked together in a Series.
- `Section` - Articles are contained in sections. A section is like a category.
- RSS Feeds - Common Content provides a site RSS Feed of published articles, and a
  separate RSS feed for each Section.
- Sitemaps - Common Content also provides Sitemaps for use with Django's sitemap
  framework.
- `Image` - An Image model is provided to house image uploads and their metadata,
  including copyright information. Resized renditions of each image are provided by
  `django-imagekit` (see the section below on Images).
- `Attachment` - A model for storing non-image file uploads and their metadata.
- `Author` - A model to encapsulate author information, including default copyright
  information. Authors are optional. Author pages show a profile and list of authored
  Articles. Each Author also has an RSS Feed.

Common Content provides views and templates (using Bootstrap 5) for each type, along
with a few different options for list display. The templates include Open Graph metadata
and Schema.org metadata for social sharing and SEO. The templates load the Bootstrap CSS
and JavaScript from a CDN by default, so there's nothing else to install and little or
no front-end to deal with (but you can override this in your own templates, see the
`bootstrap_styles` block below).

In addition to the content pages, Common Content also provides:

- `Menu` and `Link` - These models define a list of links that can be used to implement
  site navigation menus, footer links, etc.
- Redirects - Includes the Django redirects app, with a custom middleware to support
  temporary redirects.
- `AbstractCreativeWork` and `BasePage` - These are abstract Django models that you can
  subclass to create new content types that are compatible with Common Content templates
  and tools.
- Admin Support - Common Content provides admin pages for its models, and documentation
  for use by the `admindocs` app.
- `django-sitevars` - Site variables store bits of information that are reused across
  the site. Things like the default copyright notice, site tagline, analytics IDs, etc.
  can be stored in SiteVars. All variables for a site are injected into every template
  context using the SiteVars Context Processor so they are available on every page, even
  pages not provided by Common Content.
- `django-tinymce` Support - If `django-tinymce` is installed, Common Content will use
  its WYSISYG editor in its admin pages. There is an optional JSON view that exposes a
  list of recently upload images for use by the TinyMCE editor.

## Development

This project uses [uv from Astral](https://docs.astral.sh/uv/) for managing development.
Install uv before working with this repo.

After checking out the code, you can bootstrap a development environment by running:

```bash
uv sync
uv run ./manage.py migrate
```

## Installation

Add the following to your `settings.py`:

```python
import commoncontent.apps
INSTALLED_APPS = [
  *commoncontent.apps.CONTENT,
  # Optionally use tinymce in the admin
  "tinymce",
  # Other Django apps here
  "django.contrib.redirects",  # Required
  "django.contrib.sites",  # Required
  # sitevars must come AFTER contrib.sites for admin to work
  "sitevars",
]

# Ensure your middleware includes the following:
MIDDLEWARE += [
  "django.contrib.sites.middleware.CurrentSiteMiddleware",
  "commoncontent.redirects.TemporaryRedirectFallbackMiddleware",
]

# If using tinymce
TINYMCE_DEFAULT_CONFIG = commoncontent.apps.TINYMCE_CONFIG

# Add `commoncontent.apps.context_defaults` to your context processors. You will also
# need to add the request, auth, and messages context processors if not already there.
# The inject_sitevars processor should come after context_defaults so you can override
# the default values using sitevars.
# Probably looks like this:

TEMPLATES = [
  { "BACKEND": "django.template.backends.django.DjangoTemplates",
    "DIRS": [],
    "APP_DIRS": True,
    "OPTIONS": {
      "context_processors": [
        "django.template.context_processors.debug",
        "django.template.context_processors.request",
        "django.contrib.auth.context_processors.auth",
        "django.contrib.messages.context_processors.messages",
        "commoncontent.apps.context_defaults",
        "sitevars.context_processors.inject_sitevars",
      ],
    },
  },
]
```

In your project's `urls.py`, insert the Common Content URLs where you want them. To have
Common Content manage your home page and top-level pages, make it the LAST url pattern,
and list it like this:

```python
from commoncontent import views as generic

urlpatterns = [
  # Common Content provides templates for auth pages
  path("accounts/", include("django.contrib.auth.urls")),
  # You need to add the admin yourself. Remember to add the docs!
  path('admin/doc/', include('django.contrib.admindocs.urls')),
  path("admin/", admin.site.urls),
  # TinyMCE urls if desired
  path("tinymce/", include("tinymce.urls")),
  # All other urls are handed to Common Content
  path("", include("commoncontent.urls")),
]
```

## Usage

The following documentation should help you use Common Content.

### The base template

The Common Content base template is divided into major blocks that can be replaced in
your child templates. The blocks, in order of appearance, are:

- `title`: The content of the HTML page title tag.
- `bootstrap_styles`: An HTML block that by default pulls in the bootstrap stylesheet.
  You can override it to customize your version of bootstrap (e.g. with BootSwatch) or
  include `{{ block.super }}` and then add some tweaks.
- `opengraph`: Used to expose the page's open graph metadata in the HTML head. This is
  done in the default template. You won't need to override this unless you define custom
  open graph types that need special handling.
- `schema`: Holds the schema.org metadata.
- `extra_head`: Override this if you need to include custom CSS, web fonts, etc. in the
  page's HTML head.
- `body_class`: A block you can override to apply a custom class to the HTML body
  element, if needed by your custom CSS.
- `header`: A block to contain your site's masthead and main navigation, the header
  elements common to every page. Override in your site's base template.
- `precontent`: A block that appears before the main content of the page. Override this
  on home pages or section pages where you need to insert some feature above the
  standard content, such as a call to action, introduction to the topic, featured
  stories, etc.
- `content`: The main content body of the page. By default, will `include` the content
  sub-template specified in the template context. For an Article this will be the
  article text, for a Section page this will be the list of Articles in the Section.
- `postcontent`: A block that appears after the main content of the page. Often used to
  include "Read Next" or "Related" recirculation modules or a call to action.
- `footer`: A block to contain your site's footer links and blanket copyright
  statements, elements common to every page. Override in your site's base template.
- `bootstrap_js`: Loads the Bootstrap JS bundle. Override if needed.
- `extra_js`: A block just before the closing body tag. Use this to include deferred
  JavaScript or other elements you want loaded after the main page content.

### Block Templates

Common Content ships with several partial templates that can be included as the content
of the base blocks to construct a page quickly from reusable modules. These can be
loaded from the `commoncontent/blocks/` template directory.

NOTE: Although the Common Content base template is an excellent starter, these block
templates are very basic. This is an area for future expansion. For now, they are
usable, but you probably will want to build your own.

The available block templates are:

- `article_list_album.html`: Displays Articles' image, title, and description as an
  array of cards. Useful for Section pages and home pages.
- `article_list_blog.html`: Displays Articles' title and excerpt in a news feed style,
  like an old-school blog home page.
- `article_text.html`: Displays the title, featured image, and full text of an Article.
  Used on the Article detail page.
- `author_list_album.html`: Displays a profile image and short bio for each author. Used
  on the Author list page.
- `author_profile.html`: Displays the author profile photo and full bio. Used on the
  Author detail page.
- `empty.html`: An empty template to leave a block empty.
- `footer_simple.html`: A very simple footer.
- `header_simple.html`: A very simple header.

### Site Vars

We depend on [django-sitevars](https://pypi.org/project/django-sitevars/) for storing
site-specific variables or chunks of content. You can also store any variable for use
with your own templates. Variable can be injected into template contexts using the
context processor in `sitevars.context_processors.inject_sitevars`, or using the
`{% sitevar "name" %}` template tag. Variables used by the default templates/models are
listed below. Defaults for these variables are injected in the template contexts by
`commoncontent.apps.context_defaults`.

- `base_template` - The base template to use for generic pages. Defaults to
  `commoncontent/base.html`.
- `brand` - Site's brand name. Uses `site.name` if not set.
- `copyright_holder` - Custom name for the copyright holder if using the default
  copyright notice. Falls back to `site.name` if not provided.
- `copyright_notice` - Text to include in the copyright notice section of the footer
  (replaces the default copyright notice).
- `custom_stylesheet` - A site-specific CSS file. This is pulled into the `extra_head`
  block after bootstrap and Common Content's CSS, so you can use it to override default
  styles and customize the look and feel of each site. It is included using
  `{% static custom_stylesheet %}` so the value should be relative to the STATIC_ROOT.
- `default_icon` - Name of a Bootstrap icon to use by default if the object provides
  none.
- `detail_precontent_template` - Default template to use for the `precontent` block for
  detail pages.
- `detail_content_template` - Default template to use for the `content` block for detail
  pages.
- `detail_postcontent_template` - Default template to use for the `postcontent` block
  for detail pages.
- `footer_template` - Default template to use for the `footer` block.
- `header_template` - Default template to use for the `header` block.
- `list_postcontent_template` - Default template to use for the `postcontent` block for
  list pages.
- `list_precontent_template` - Default template to use for the `precontent` block for
  list pages.
- `list_content_template` - Default template to use for the `content` block for list
  pages.
- `paginate_by` - Items per page on list pages, same as Django's ListView, see
  [pagination](https://docs.djangoproject.com/en/dev/ref/paginator/) in the Django docs.
- `paginate_orphans` - Same as Django's ListView, see
  [pagination](https://docs.djangoproject.com/en/dev/ref/paginator/) in the Django docs.

### Images and Media

Common Content takes advantage of
[django-imagekit](https://pypi.org/project/django-imagekit/) to manage images. Models
are provided for storing file metadata including copyright information.

Common Content uses presets to produce images in specific sizes as recommended by
[Buffer's Social Media Image Guidelines](https://buffer.com/library/ideal-image-sizes-social-media-posts/).

These named image presets are available:

- small: 160x90
- medium: 400x225
- large: 960x540
- opengraph: 1200x630
- hd720p: 1280x720
- hd1080p: 1920x1080
- portrait_small: 90x160
- portrait_medium: 225x400
- portrait_large: 540x960
- portrait_cover: 1000x1500
- portrait_social: 1080x1350
- portrait_hd: 1080x1920
