#Basic support for django 1.8 - 1.10 has been implemented, and the tests are passing, but they are most certainly not comprehensive. The drop-in functionality with no configuration seems to work, but nothing else not covered in the tests has been tried in the wild, so proceed accordingly
When dealing with content from unacquainted sources(e.g., clients or designers)
one often gets images with absurd dimensions and/or filesizes: A 3000px-wide
play-button, a 10MB logo, etc. Media-helper attempts to mitigate this problem
by automating image-resizing and delivering the most appropriately sized image to
the browser.
It is also designed to be dropped into existing projects with minimal effort.
It's still in the alpha stage, but if you're careful it might make your life a
little bit easier while also speeding up your load times and reducing data
transfer.
- Quick start
- Changelog (Recent changes only)
- General Info
- How does it work?
- Requirements
- Installation
- Usage
- Configuration
- What it does not do (...yet?)
i. Requirements
- Jquery
- Pillow
1. Installation
# pip install django-media-helper
# install from source until more tests are written
pip install git+git://github.com/brmc/django-media-helper.git
2. settings.py:
#!python
INSTALLED_APPS = (
...
'media_helper', # should go after all your apps
)
...
TEMPLATE_CONTEXT_PROCESSORS = (
...
'media_helper.tools.context_processors.include_settings',
)
2. urls.py:
#!python
urlpatterns = patterns('',
...
url(r'^media-helper/', include('media_helper.urls')),
...
)
3. Templates:
<script src="//code.jquery.com/jquery-1.11.0.min.js"></script>
{% include 'media_helper/ajax.html' %}
That should be it for default functionality
- Added support for django 1.8-1.10
- Removed support for django < 1.8
- Added support for python 3.5
- Removed support for python 3.4, 3.3
- Fixed the long-description in setup.py that was causing installation to fail.
I'm still not quite sure why, so I just decided to hard-code it for now.
-
New management command to resize a specific image: --resize=PATH/FILENAME
-
Images and backgrounds will now be preloaded before being replaced to avoid flickering
-
Created the following jQuery plugins for resizing and replacing images and
backgrounds to allow more granular control:
swapImage(),
swapBackground(),
replaceAll()
Docs still need to be written for these though.
-
Fixed unicode decode error in management.commands.mediahelper
-
Changed resize quality from 85 to 'keep' because resized jpegs were often larger than their source files.
-
Finally wrote a regex for url stripping
-
Created tox envronments explicitly for django versions
-
Fixed firefox issues. Now correctly removes host name from background-images
-
Changed tests to work with and without tox
-
Updated tests to reflect new default settings
-
Fixed ajax success callbacks to use selectors defined in settings
-
Updated readme
- Flaked the shit out of everything
When dealing with content from unacquainted sources(e.g., clients or designers)
one often gets images with absurd dimensions and/or filesizes: A 3000px-wide
play-button, a 10MB logo, etc. Media-helper attempts to mitigate this problem
by automating image-resizing, delivering the most appropriately sized image to
the browser.
It is also designed to be dropped into existing projects with minimal effort.
It's still in the alpha stage, but if you're careful it might make your life a
little bit easier while also speeding up your load times and reducing data transfer.
It's basically broken into two parts....
First, this is a straight fork of django-cleanup
, an app that deletes files
from the server when the associated field or model is changed or deleted. It
retains its complete functionality with virtually no modifications.
From the django-cleanup docs:
**Warning! If you use transactions you may lose you files if transaction
will rollback. If you are concerned about it you need other solution for
old file deletion in your project.**
Most django projects I've seen don't use transactions and this app is
designed for such projects.
Images are resized, stored, and delivered via AJAX while the rest of your page
loads.
media_helper.tools.cleanup
connects pre_save and post_delete signals to special
functions(these functions delete old files) for each model whose parent app
is listed in INSTALLED_APPS above 'media_helper'.
When a user visits your domain, the low-res copy of each image will be initially
delivered to the browser. jQuery will then determine the rendered size of all
elements containing an image or background-image, bundle all these images their
respective dimensions, and send them to the server via an AJAX request. If no
image already exists, a new one will be generated. If something goes wrong,
the low-res image will remain in place.
The size of the image will be determined by the html element's rendered size,
and if the browser window is not maximized, it scales it up so image quality
won't be lost in case the user does maximize the window. This assumes
you're using a responsive design with images whose sizes are not statically
defined. In the near future I will accomodate for for alternate scenarios
So for those of you who prefer lists, here's what's going on:
- Deliver low-res images with initial http request
- Measure all elements containing image or background-image
- Send list of images, bundled with their sizes, via AJAX back to server
- Search for existing image with correct dimensions or create aa new one.
- note: To avoid an inundation of images being re-sized, dimensions will be
rounded to a certain value. see Configuration
One person was under the impression that 3 requests were made per image
(initial request, ajax, and the request for resized image), so to clarify,
that is not the case. Well, it is true if there is only one image per page,
but all ajax requests are bundled together, so the number of requests per image
approaches 2 as the number of images increases.
When you save an image, a couple things happen:
- The original image is copied to a new location and kept safe
- Several resized images will be initially generated (Info below)
- A default, low-res image will be also be created
Resized images other than the default image are named and stored according to
their size. For example, if an image foo.png
was resized to be 300px wide,
it would be found under the following request path
/<MEDIA_URL>/media-helper/<upload_to>/foo.png/300.png
It is warned that having purely numerical image names can negatively affect
search results. Nevertheless I have chosen to do so anyway because:
-
the default image retains its original name, and from my understanding
search engines index pages and images based on the results before the DOM
is modified by javascript, so this should not affect indexing. -
any information found in the image name is retained in the full path.
If I am mistaken, please let me know so I can rectify this as
soon as possible.
So if you are uncertain and SEO is of utmost importance to you, you might
reconsider using this app.
This app uses Pillow
for image resizing. If you've never used PIL or Pillow
before, aside from the base library itself, you're going to need to make sure
you have the appropriate development packages for each type of image you intend
to encode.
IMPORTANT: PIL/PIllow WILL 'successfully' install without the encoding
packages, but you won't be able to do much, so you need to install these
packages BEFORE you try to install Pillow.
Pay particular attention to this part:
http://pillow.readthedocs.org/en/latest/installation.html#external-libraries
And don't worry, Pillow pisses everyone off at some point.
Unless you plan to write the client-side AJAX requests yourself, you're going
to need jQuery. It's pretty basic stuff, so any reasonably recent version
should be fine. I haven't observed any conflict with jQuery 2.x, but
considering that the goal of this app is to accomodate for people who--with
no disrespect intended--don't know any better, sticking with jQuery 1.x might
be a better idea for the IE support.
<script src="//code.jquery.com/jquery-1.11.0.min.js"></script>
1. Install media_helper:
#pip install django-media-helper
# install from source until more tests are written:
pip install git+git://github.com/brmc/django-media-helper.git
2. Add media_helper
to settings.py
INSTALLED_APPS = (
...
'media_helper', # should go after your apps
)
media_helper
should be placed after all your apps. (At least after those apps
which need to remove files.)
*3.* Add media_helper.tools.context_processors.include_settings
to
TEMPLATE_CONTEXT_PROCESSORS
in settings.py
TEMPLATE_CONTEXT_PROCESSORS = (
...
'media_helper.tools.context_processors.include_settings',
)
3. In order to handle the ajax requests, add the following to your root urls.py
urlpatterns = patterns('',
...
url(r'^media-helper/', include('media_helper.urls')),
...
)
4. Now we also need some javascript, but the JS depends on some context variables,
so include the following template somewhere after jquery, like so:
<script src="//code.jquery.com/jquery-1.11.0.min.js"></script>
...
{% include 'media_helper/ajax.html' %}
This is safe to use inside django-compressor compress
tags. In fact, it's
recommended.
And I write JS like a neanderthal, so feel free to write your own AJAX. I don't
know if I'll get into the specifics in this version, but what I'm doing is
pretty straight-forward and basic.
If you are content with the default settings(which just happen to be discussed
next...spooky huh? Life is weird), for a new project, the rest is handled
automatically. You don't need to do anything else.
However if you're adding this app to an existing project, or something seems to
be broken, here are some management commands that might be useful.
note: there aren't any tests written for these commands yet. Use them at
your own risk. I've used them on three small sites in production without
problem.
This command is used to retrofit the media_helper app into a project that
already exists. With it you can resize all images found in the MEDIA_ROOT
directory, resize/adjust the quality of the placeholder image, delete all
resized images, and/or restore images to their original size, quality and
location. If all command options are used simultaneously, they will be
processed in the following order:
--restore
--delete
--resize-originals
--resize-all
And note that whenever --delete is passed, --restore will be forced so you
don't risk losing your original images.
usage:
python manage.py mediahelper
Restores the original images found in the media-helper sub directory to their
native path and then deletes the backup. All other images remain intact. This
means that the full-sized image will be delivered when the page is loaded.
Automatically resizes a single image according to scaling factors defined in
the settings.
You only need to include the upload_to directory for FILENAME
. For example
for an image uploaded to 'images/' would be resized like so:
--resize=images/file.jpg
Resizes all allowed images in MEDIA_ROOT.
Use this when you want to change the quality and/or size of the place holder
images. This just reinitializes everything, so for any changes to take affect
you need to adjust the appropriate settings in your settings.py
Restores the original images and deletes the media-helper directory tree.
Settings should be set in your settings.py
A boolean that determines whether a series of images will be generated when a
model is saved. The auto-sizing feature expects your images to be consistently
sized relative to one another and the final layout. If your images are all
willy-nilly, this probably won't do you much good. More info in the next
section.
default value: True
A list of scaling factors to be used to automatically scale images when they are
saved.
default values:
[0.3, 0.3125, 0.4, 0.426953125, 0.45, 0.5, 0.53125, 0.546875, 0.5625, 0.6, 0.625, 0.65625, 0.75, 0.8, 1.0]
These values were chosen with a maximum screen width of 2560px in mind where
each scaling factor corresponds to a common width. For example:
1.0 -> 2560px
.8 -> 2048px
.75 -> 1920px
...
.4 -> 1024px
etc...
The assumption is that if a layout is designed for a 2560px width, each
image is exactly cropped to fit the expected rendered dimensions of their html
element with no stuffing or stretching going on.
For example if you have a background-image, a logo, and a banner on a page where
the background-image takes up the entire window, the banner is 80% of the screen
width, and the logo is 10%, with the default settings media-helper
expects you
to upload images with the following widths:
image | width |
---|---|
background.jpg | 2560px |
banner.jpg | 2048px |
logo.png | 256px |
So for the logo and background image, the auto-scaling feature would create the
following sized images(widths are in px of course):
scaling factor | 0.3 | 0.3125 | 0.4 | 0.426953125 | [...] | 0.75 | 0.8 | 1.0 |
---|---|---|---|---|---|---|---|---|
background.jpg | 768 | 800 | 1024 | 1093 | [...] | 1920 | 2048 | 2560 |
logo.png | 77 | 80 | 103 | 110 | [...] | 193 | 205 | 256 |
Two things to point out,
-
it always rounds up to the nearest pixel
-
this doesn't take into account any additional rounding preferences, making it a good time to go into that.
This is probably one of the most important features because it is intended to
prevent a deluge of images being created, it is especially helpful for sites
with many images.
It is simply an integer representing the near # to round to.
default value: 10
This value was chosen rather arbitrarily, but I figure it's large enough to
account for slight variations in browsers.
This is the scaling factor for the low-res default/placeholder.
default value: .1
The minimum allowed size for an image (in pixels).
default value: 20
The folder under which images will be saved
default value: 'media-helper'
The quality of the low-res image.
default value: 50
default values for MEDIA_HELPER_DEFAULT
and MEDIA_HELPER_QUALITY
were also
chosen without any particular reasoning other than "make smaller." No science
included.
This tells media_helper which file extensions to recognize, but isn't checked
intelligently. It's simply a string comparison, but in case of errors, it
should fail gracefully.
default values: ['jpg', 'jpeg', 'png']
A string
of jQuery selectors for images to be resized (or ignored).
These can be any valid jQuery selector. So you can make them as simple or
complex as you wish.
For example, both 'img' and '.container .img:not(nth-child(2))' would work.
This also combines the ALLOWED_ENCODINGS encodings, so the default settings
produce:
$(img[src$=".jpg"], img[src$=".jpeg"], img[src$=".png"], )
default value: "img"
Like above, a string of jQuery selectors for elements with background-images
to be resized.
These can be any valid jQuery selector. So you can make them as simple or
complex as you wish.
For example, both 'div' and '.container .img:not(nth-child(2))' would work.
default value: "div"
-
It does not scale up. It only shrinks images.
-
It doesn't resize images found under STATIC_URL
-
It doesn't handle javascript image zooming libraries very well, particularly
if they source the same image or the display is set to none