thumbspage — Turn Folders into HTML Image Galleries |
Version: | 2.3, May 9, 2022 (patched Dec-2023: changes) |
License: | Provided freely, but with no warranties of any kind |
Author: | © M. Lutz (learning-python.com) 2016-2023 |
Install: | Unzip thumbspage's download, install the Pillow library |
Run with: | Python 3.X, on any platform supporting this and Pillow |
See also: | web page, live demos, screenshots, examples, code |
Welcome to the thumbspage manual. This guide is meant for content creators and consumers of all kinds. It includes an overview of the program, detailed usage information, and a look at what's been upgraded in each release. Whether you consider yourself a developer or user, you'll find resources here to help you get started building and viewing image galleries with thumbspage.
Because thumbspage adds items to the images source folder and may rotate photos there, you are encouraged to read this guide first—especially its usage caution—before running thumbspage on valued picture collections. But if you're in a hurry: for a quick preview, try a live demo, packaged examples, or the code. You can also click the image above to test-drive the original use case, and take a brief tour of this program at its web page.
This section introduces the basics of thumbspage's roles and operation. Read this first if you're looking for a quick summary.
In short, this program allows you to view or display a folder of images in a web browser, using a format which is simple by design yet noticeably better than browser defaults, and does not require a web server except when viewing images online.
thumbspage turns an images folder into an easily viewed gallery. It automatically makes image thumbnails, an HTML thumbnail-links index page, and HTML image-viewer pages for all the images in a source folder, all of which can be broadly customized by content creators. The mostly static results can be viewed offline or online in any web browser.
In a theoretical sense, thumbspage is a program that builds another program: its results are a combination of HTML, CSS, and JavaScript, which are run by a browser. More tangibly, thumbspage simply creates presentation scaffolding for user-friendly views of a folder's images.
In a bit more detail, given a folder of image files, this program generates an HTML index page in one of two flavors, with generated thumbnail links for each image in the folder. This page's links open the full-size images using either browser-native (i.e., built-in) display, or generated HTML viewer pages.
Viewer pages in turn dynamically scale the image to your display; include links for gallery navigation, automatic slideshows, and one-page fullscreen; and open image-info popups on filename clicks, and image-description popups on button taps when enabled. On mobile and PC touch screens, swipe gestures on the image trigger viewer-page actions too.
Apart from image scaling and interactive tools, the net effect is intentionally static: generated results reflect the folder's contents at build time only, but do not require a web server, and can be viewed both offline and online in any desktop or mobile browser. As such, this program can be used both for both websites and non-web use cases, including program documentation, photo sharing, and general viewing.
When run using the techniques explored ahead, the builder program skips non-image files; uses optional header and footer HTML inserts in the index page; uses text note files for any or all images in viewer pages; makes an optional bullet list for subfolders in the images folder; and creates the output index page in the images folder itself, along with a subfolder for thumbnail images and viewer pages. The resulting gallery is complete and self-contained, and ready to be viewed or published.
After running this program, you can view or publish its results in a variety of ways:
index.html
by default).
_thumbspage
and index.html
by default, respectively).
As a client example, the website that hosts thumbspage uses it for some 50 galleries today, some of which are viewable both online at the site, and offline in program download packages. For a sampling, see this site's demos. Builders can also find gallery-build pointers in thumbspage's publishing scripts, though its simple console interaction or command lines may suffice for most uses.
thumbspage began as a builder of simple index pages that used generated thumbnail images for their links, but its feature set has evolved over time in response to usage experience. Among its new highlights added in recent releases:
setting=value
command-line arguments override config-file
settings;
tooltips can be enabled on all
pages;
and more.
You can read the full story on new releases ahead. Move on to the next section to start learning how to use thumbspage to view and display your photos.
This section covers the basics of viewing the galleries which thumbspage creates. It's oriented towards gallery consumers (a.k.a. users), though creators might find its functionality overview helpful too. If you're looking for details on running the program to make galleries, though, see the builders' section.
By today's standards, thumbspage galleries are straightforward and intuitive to use. They work and may be viewed on any desktop or mobile browser, and both online and offline. No extensions must be installed in browsers to view thumbspage galleries, though JavaScript should be enabled for the best user-interface (UI) experience, and some browsers have known quirks that can impact gallery behavior (there's more on JavaScript and browsers in Viewing Tips ahead).
Once built, a thumbspage gallery consists of an index page opened initially, and one viewer page per image opened from either thumbnails on the index page, or direct page-address URLs. The next sections describe how these two page types are used.
Index pages are the usual entry point into a gallery. Here's one captured on desktop and mobile browsers, and its live version.
Index pages display thumbnail images, either in a fixed table which scrolls both horizontally and vertically as needed, or arranged to fit the display's size per the 2.1 update ahead. Either way, index-page thumbnails normally open the next section's larger image-viewer pages when tapped (or clicked). By default, index pages also display the image folder's basename, and a bullet-list of links that open subfolders nested in the page's folder, if any are present.
Beginning with 2.0, thumbspage index pages also usually display a floating Top button near page bottom after a scroll, much like that in this guide. Click (or tap) Top to jump to the top of the index page immediately—to read its preface or open its first image, for example. Top appears only after you scroll down far enough, so it may not show up on smaller index pages.
An index page's subfolders links
may lead to supplemental info or nested galleries, and their behavior varies per
usage mode. When viewing galleries online, subfolder links typically open
a nested gallery's index page automatically. When viewing
offline, these links open a simple directory listing, and you'll
manually click nested galleries' index pages (e.g., index.html
)
if present. These links also open directory listings for non-gallery folders
on some online servers; tap filenames to open. There's more on subfolders
ahead.
While many index pages follow the preceding norm, individual galleries can also customize it with tailored header and footer content above and below the thumbnails table. They might also use thumbnails that open raw displays instead of viewer pages, may display tooltips on mouseovers, and can opt out of either floating Top buttons or subfolder bullet lists. Experiment with galleries you visit to see which options they support.
Update: as of version 2.1, thumbspage index pages may also use either a fixed or dynamic layout. By default, index pages use the original fixed layout, which always displays the same number of columns. In dynamic layout, the thumbnails table does not scroll horizontally; instead, the number of columns is chosen to match the display size, and changes when the display is resized. Dynamic layout is an option which can be enabled when a gallery is built. If you run across this alternative, you can shrink or expand your window on PCs and mobiles with popup app windows to view fewer or more columns. On mobile in general, switching device orientation has a similar effect. See the demo.
Index-page thumbnail clicks and taps by default open viewer pages (a.k.a. image-viewer pages in this doc) to display a single image. Here's one captured on desktop and mobile, along with its live version.
Viewer pages display the image's filename, the image itself, and a bottom toolbar of action buttons, and respond to swipe gestures on the image on touch screens. The image is automatically scaled to fill the window's or display's available space, and grows or shrinks in both directions with desktop-window resizes, mobile-device rotations, fullscreen toggles, and popup or split-screen app-window resizes on some mobiles. The toolbar has four to six named action items (Note and Full are optional), and scrolls horizontally to reveal buttons if needed on smaller displays. Viewer pages may display tooltips on mouseovers too; this is enabled by default when galleries are built.
Everything on a viewer page does something when tapped and images respond to swipes, as the following sections will describe.
All told, there are seven or eight actions available to users on viewer pages, five of which have swipe-gesture equivalents—and two of which are invoked with display widgets instead of buttons:
control
or command
with +
/-
),
and may yield a larger image depending on browser and display size.
Use the browser's Back operation (e.g., button or swipe) to return to the gallery's
viewer page. The Full toolbar button also zooms the image in most contexts, but
we have to move ahead to see how.
As of 2.2, the image-tap event is also triggered by an up-swipe on the image in
supporting browsers when image notes are unused (see ahead).
The remaining viewer-page actions can all be requested by tapping named toolbar buttons (or swiping equivalently), and one is optional and may be omitted when a gallery is built:
Keep in mind that this feature is optional and new: Note buttons will be present and up-swipes will open notes only if notes were enabled at gallery build time. If notes are unused, up-swipes open raw views as in 2.2; if notes are used, you can still open raw views with an image tap. Also note that Note popups don't stay open during Auto slideshows; tap Auto to stop and read a note along the way, per the next item here.
When active, a slideshow applies to all thumbspage viewer pages visited in a single browser tab. They stop (really, pause) when you leave a gallery's pages, and may be cancelled or restarted at any time by pressing Auto again in any viewer page. The Prev and Next toolbar buttons work during, and do not cancel, an active slideshow; use these to skip or move to images during the show. You can also use Index or your browser's Back function to pick an image from the gallery's index page at which to continue the slideshow in progress. In most contexts, a Back or any other navigation that reenters a gallery resumes a slideshow left in progress too; press Auto to cancel whenever you like (and see ahead for a rare exception on Safari).
Update: as of version 2.1, the Auto button changes to be underlined as an indicator whenever the slideshow is on, and changes back to normal text when the slideshow is off. This helps make the show's state apparent, especially when it's first started or is resumed on return from another page.
Because of these limitations, the Full button and feature may be omitted on a per-gallery basis (it's enabled by default). When Full is available, it provides a per-image blowup/quick-zoom option (e.g., before and after), that may be easier than raw-image zooming via image taps. Users looking for a true fullscreen experience, however, may be better served by using browsers which support a manual fullscreen that spans multiple pages, including those in thumbspage slideshows; see the next section's fullscreen tip for pointers. Note that Full may be hidden on small displays; scroll the toolbar left as needed.
Update: like Auto, the Full button changes to be underlined as of 2.1 as an indicator that its fullscreen mode is on, and resets its font when its fullscreen is cancelled, whether by Full or outside the page (e.g., with an Escape key). Unlike Auto, though, fullscreen mode is obvious, and Full's font changes are not supported on all browsers. Moreover, even where supported, Full will not change for, and will not cancel, a fullscreen mode started independently of the page; for better or worse, user-initiated and page-initiated fullscreens differ, and browsers handicap the latter in the name of security.
Beginning with version 2.2, thumbspage viewer pages respond to swipe gestures on all devices and browsers which support them. Swipes require a touch screen (touch pads aren't supported), but they work broadly on both smartphones and Windows PCs, and in all dozen browsers tested except Internet Explorer (subject to one up-swipe limitation covered ahead).
To perform a swipe, start the gesture in the image-display area—that is, on the image itself—in any viewer page. The direction of the swipe determines its action:
You can test-drive swipes on a touch-screen device by running any of the demos listed here. Where available, swipes are simply a convenient alternative to other page actions. All former widget and toolbar taps described above still work as they did in prior versions, as do browser long-presses, pinch/spread zooms, and back/forward swipes. Mouse clicks naturally still work on non-touch displays too.
Usage tip: some browsers implement back/forward swipes as an alternative to buttons. thumbspage left/right swipes won't clash with these as long as both are performed well. Be sure to trigger browser back/forward with a short flick from screen edges, and begin thumbspage swipes in the page's image display. If this is difficult, you can also do back/forward by swiping above or below the thumbspage image-display area, or simply use browser back/forward buttons.
As usual with UIs, first-hand experience is also a great way to learn how to use thumbspage galleries; browse the examples in the thumbspage package, and test-drive the live demos at thumbspage's host site for more on index- and viewer-page interaction. The next section provides more info on gallery views.
This section collects additional assorted tips for viewing thumbspage galleries. Content you'll find here:
Because viewer pages suffer most when JavaScript is off, they give you an alert when yours is disabled; here's one on desktop and mobile. JavaScript can be misused, but is fully harmless in code that thumbspage generates itself, and is essential for the sort of interactive-UI functionality offered by thumbspage galleries. In fact, most of today's web isn't quite usable without it—for better or worse. Please turn JavaScript on for the best thumbspage experience.
Update: as of version 2.3, JavaScript is now fully required for using the viewer pages created by thumbspage. This is normal on the web today, and was already effectively required in prior versions, because images did not scale and most widgets were inoperative. 2.3 made it official, with an enhancement that renders navigation buttons no-ops without JavaScript too, leaving no usable content. For more background, see the builder's note.
Caveats: while no browsers or devices are known to be unable to display thumbspage galleries, some browsers have relevant quirks mentioned in other notes ahead; usability is naturally subject to browser and platform changes; and viewability is prone to decline with screen size (all bets are off on your watch). That said, HTML may not be a portability utopia, but thumbspage manages to make it work.
F11
key (with fn
if
needed) puts the browser in fullscreen mode, which persists for all
pages visited until explicitly disabled with another F11
.
This works in Chrome, Firefox, Edge, Internet Explorer, and likely others,
and allows thumbspage Auto slideshows to run completely in fullscreen mode.
F11
key (with fn
if
needed) switches browsers to and from persistent fullscreen mode, just
as it does on Windows. This works in Chromium, Firefox, and likely others,
and, as on Windows, allows thumbspage Auto slideshows to run completely
in fullscreen mode.
control
+command
+f
.
A shift
+command
+f
may additionally hide the toolbar in Chrome (but not Firefox, currently),
and you can also go fullscreen by clicking the upper-left-corner
green button or using menu fullscreen icons.
For additional tips and other platforms and browsers, try a web search, or consult your browser; browsers change regularly, and new fullscreen options may crop up over time. Manual and persistent fullscreen isn't available everywhere today, but it generally beats thumbspage's one-page Full where possible. You may still find thumbspage's Full useful as a quick zoom, though, especially on browsers that lack a manual option today. With any luck, those browsers will make Full fully superfluous in the not-too-distant tomorrow.
Update: on Android, the Opera web browser recently gained an amazing fullscreen mode for landscape that works much like those in desktop browsers, and spans multiple pages. For details, see this usage note at thumbspage's host site. The Opera browser is largely the same as Chrome under the hood, and runs thumbspage galleries just as well—and arguably better with its landscape fullscreen. Yet another reason to consider parting with the herd on Android.
file://
URLs offer more options. On mobile devices, however,
offline viewing isn't always as seamless, but you'll have to move on to the next
two notes for the Android and iOS offline stories. macOS Safari users: see also the
caveat ahead for the rare use case of
browser Back returns to offline galleries.
content://
URIs.
While these may be spun as security measures, that argument has been tiredly
used for many an Android
breakage.
thumbspage galleries are known to work offline on Androids Nougat, Oreo, Pie, and 10, but its releases and devices vary too widely to give universal advice here. If your offline gallery fails, try the following work-arounds that have proved successful in some contexts:
file://url
if present)
file:///sdcard/...
)
/storage/emulated/0
; see the
primer)
https://...
)
Especially on newer Androids, also make sure the browser has permission to access your files in storage. Firefox, for example, may require this on Android 10. Try Settings/Apps and navigate to storage permissions, or similar on your device. This has grown more crucial as later Androids have moved to lock down storage to app-specific sandboxes.
If you're technically inclined, you might also try running a
web-server app
on your device and viewing your gallery on localhost
or a
dedicated port number; this is wholly untested (but arguably awesome).
Caveat: the upcoming Android 11 will clamp permissions down further
(read the latest edicts
here),
but its impact on browser apps remains to be seen.
Update: for recent findings on both local-file views and running web-server apps on Android, see the off-page coverage here, which is part of an Android 11 review. In short, you can run a web server on Android, but your functionality may be limited and your disconnections many. Android 11 has also narrowed your options for local-file views in browsers, in the name of a security rationale; inoperable phones may be secure, but they're also inoperable.
Update:
the thumbspage team recently discovered another route to offline viewing on Android.
On Android 11, simply copy your content into the browser's own sandboxed folder, using
a file explorer that has permission to do so (Solid Explorer and Cx File Explorer do
at this writing). Then, use that folder's local path in a file://
URL
in the browser to view offline. For example, copy content to these locations for
Chrome and Opera (alas, this won't work for Firefox on 11, because it strips
the file://
):
file:///storage/emulated/0/Android/data/com.android.chrome/your-folder-and-file-here file:///storage/emulated/0/Android/data/com.opera.browser/your-folder-and-file-hereThis works for Chrome on Android 10 too, but isn't required for Opera or Firefox on 10, where they have full access to arbitrary storage paths (unlike both Chrome on 10, and all browsers on 11). This isn't the same as arbitrary file access, of course; you're limited to files dropped into a browser's folder. This is also an arguably terrible idea: your content in the sandbox folder will be deleted if you uninstall the browser, and is prone to being nuked if the browser ever resets its own folder. Hacker beware!
Update: in late 2021, some Android file explorers now provide seemingly heroic work-arounds on Android 11 that allow you to use a browser to view web pages stored locally on your phone—including thumbspage galleries. For example, among file-explorer apps tested:
content://
URIs that implement Android's content-provider paradigm.
This scheme is not a substitute for a web server, opens some complex pages slowly, and
can break down for some nested website content (e.g., folder indexes may oddly invoke
a download that fails). But it also suffices for viewing basic content and thumbspage
galleries on your phone.
Thus, your first step when trying to view locally stored thumbspage galleries on Android 11 and later today is to simply try opening them in a file-explorer app to see if they are supported automatically. For more details on the constantly evolving struggle between Android permissions and file explorers, see also the updates here, here, and here.
Nor is this an isolated case.
As another example, the thumbspage team recently found and reported a
settings-related bug in 2020's Android Firefox—though not
thumbspage specific, this browser sometimes leaves an odd blank
strip at the bottom of thumbspage viewer pages if you enable its
"Scroll to hide toolbar" option (more details
here).
Thumbspage's host site also reported a bug in version 53
of Android Opera—though it does not impact thumbspage galleries (and was
fixed in version 60), this browser strangely truncated the remainder of a page at a
/*
character sequence in text formatted with <pre>
.
At the end of the day, thumbspage is completely dependent on browsers which change frequently and arbitrarily, and have an unfortunate history of fragility (indeed, at times it seems a website could be laid low by a good sneeze). For best results, clear settings that cause issues on Android, use similarly obscure browser behavioral switches with care, and try other browsers as a last resort when browser bugs—and enhancements—break your viewing experience.
For more tips on using thumbspage galleries, viewers are also encouraged to browse the rest of this document; although it's focused on gallery builds from here on, you can also find usage details along the way.
This section is meant for gallery creators, who run thumbspage to build new galleries. It describes thumbspage install requirements, inputs and results, customization options, and other operational details. It's also a comprehensive tutorial that doesn't assume you are already a command-line wizard—which means more advanced readers may want to skim parts useful to others. If you're looking for info on how to use galleries, try the viewers' section first.
Examples: if you enjoy learning by example, you can also study the thumbspage demo sites online here and here for sparse but fast tutorials. These sites list their run logs, and use custom headers and settings which you don't need to code if the defaults work for your galleries.
This section covers the basic logistics of using the thumbspage program to build galleries. The pages of the galleries it builds work on all browsers and platforms and require no extra installs or infrastructure, as covered earlier.
thumbspage itself is a Python program that runs on all major platforms, and is provided in source-code form which you run with your local Python. To install the program, download its zipfile from the Download section of the following web page and unzip it on your computer:
learning-python.com/thumbspage.htmlThe thumbspage program also requires installs of Python 3.X (any X) to run its source code, plus the third-party Pillow (a.k.a. PIL) image library to extend the installed Python 3.X with image-processing tools. If they're not already present, fetch and install both these items from the following sites, respectively (or search the web for other links):
www.python.org/downloads/ pypi.python.org/pypi/Pillow
thumbspage can be run to build galleries on any platform that runs Python 3.X and Pillow, and has the required folder and file access permissions. For example, this program has been verified to run on Windows, macOS, Linux, and Android. The latter has unique requirements and permission rules; see the notebox below for usage on Android, as well as the similarly constrained iOS. For a related Pillow install pointer, see also this page.
As discussed ahead, as of version 1.7 thumbspage also uses the pure-Python and third-party piexif library to update image Exif tags, but this library's code is included and shipped with thumbspage itself on all platforms, and does not require a separate install.
A note for programmers: Pillow is used from Python for thumbnail generation,
image rotation, and Exif-tag processing—which all occur at gallery-build
(i.e., program-run) time. Images are scaled at image-display time instead by
JavaScript code in generated viewer pages—which mix HTML, CSS, and
JavaScript in a tangled and browser-specific morass that is the norm
for web development today. The stew manages to work well anyhow.
thumbspage galleries can be both viewed and built on mobile devices.
This guide often focuses on the mobile portability of generated pages viewed, but
it's also possible to run the program itself on your Android smartphone or
tablet from a Python-aware app.
For instance, thumbspage galleries can be built in the
Termux app, after running both
of the following commands (more details
here,
though the
After this, thumbspage commands work in Termux the same as elsewhere.
You can also launch thumbspage command lines on Android in
Pydroid 3,
after running the same
Once you've installed thumbspage and its required tools per the preceding
section, you're ready to start
turning your image folders into galleries.
This section demonstrates how to run thumbspage on your computer.
It also goes over command-line basics for users new to the technique.
For a basic launch of thumbspage, run script
thumbspage's main options are selected with five console replies, or their enter-key defaults,
on each run. The following example session gives user inputs in bold font. In its
prompts, inputs are described in
This section documents the input replies in the preceding usage example, numbering
them from the first
Special cases: a solitary
For reply #3, give the number of thumbs you want to display in each
row on the index page. The thumbs table automatically expands and shrinks
with the page, and scrolls both horizontally and vertically
where needed. Note that this input is not requested if you opt to
use 2.1's new dynamic index-page layout alternative,
because this layout determines the number of columns automatically from
page size.
Exception: as of version 2.3, thumb sizes provided as configs by file or
argument can now give a single integer instead of a tuple, if both sizes
are the same. Sizes input at the console, however, must still be 2-tuples
as described here. See 2.3's update for details.
If you change your mind or make a mistake while inputting replies:
you can cancel a run by typing a break or EOF key sequence (e.g., hold
Besides the basic console interaction of the prior section, thumbspage also
allows you to pass in the folder name as a sole command-line argument instead
of a prompt reply. This allows you to use shell auto-completion on long folder
names, though it's probably more useful in a console window than an IDE.
Folder argument or not, you can also automate a launch by using command-line
In the last of these modes, parameters in the input file
Naturally, you can combine all these with shell syntax for pipes and output
redirection (
Finally, on Unix systems—and Windows systems that have Unix subsystems or
similar tools—you can also provide console inputs immediately
following a command line, by using "here documents" of the sort available in
Bash and other Unix shells. Using this in a script file allows you to code
canned input parameters in the script itself, without having to juggle a
separate file. The following demos the idea,
but see your shell's docs for more details if needed; in short, it assumes
Fine print: use just
You can catch the last of these ideas in action in the shell and Python scripts
here,
here, and
here, used to
automatically generate thumbspage's example galleries.
thumbspage's full set of build scripts here may
also provide pointers for more advanced automated builds, including
automatic zips and uploads for external thumbspage clients.
Although thumbspage's basic console interaction covered
above
should suffice for many use cases, automating its launches
is straightforward, and may be necessary when thumbspage is used as
a nested tool in a larger build system.
For more background on the shell techniques shown here, try the off-site overviews of
streams,
redirection,
variables, and
here docs.
For more thumbspage usage examples, browse its
Update: thumbspage 1.7 added the optional
folder-name command-line argument. To better reflect
this, 1.7 also moved folder name to be the first prompt when not provided
on the command line; this invalidates some older session logs where folder
name was asked later, but the difference is trivial.
2.0 added nicer input-error reporting,
replacing exception tracebacks. 2.1 omits the thumbs-per-row input
when dynamic layout is chosen, because
the input is moot in this mode, and omits all inputs but the first as
irrelevant for imageless folders.
Update: thumbspage 2.2 added optional
command-line config arguments, of the form
For full coverage of this feature, see the 2.2 release note.
Update: thumbspage 2.3 added new config settings
that can be used to override console inputs. When provided, in the config
file or command-line arguments, input values are taken from the settings, and no input
is requested at the console's stdin. For example, a command line of the following form
overrides two console inputs with arguments:
This provides are portable and uniform alternative
to the shell redirect and here-document techniques described in this section.
For more details, see the 2.3 release note and its command
examples.
As noted above, 2.3 also allows thumbs maximum size to be provided as a single
integer instead of a tuple, when given by configs; see 2.3's note.
Because thumbspage is run from a command line, you need to know the basics
of folder pathnames in the console realm. Luckily, this is much simpler than
it may sound; as noted, the pathname you input at
prompt #1 can be either:
For instance, when running thumbspage via command lines,
you can
Absolute paths are generally required when running thumbspage from an IDE such
as PyEdit, if they run
code in the program's folder. Also see your file explorer's options for copy/paste
of a pathname to which you've navigated; it can often avoid having to type a long
pathname at thumbspage prompts.
As you might expect, the
If you'd like more background on command-line use, you'll find both online and
brick-and-mortar resources that go into more detail. Here, the next section
moves on to show you how to make your galleries more unique,
after a brief technical sidebar.
For console prompt #4, thumbnails sizes are
input as a pair of numeric pixel dimensions,
Without getting into code, here are a few examples of how this works; thumbnail
sizes input on the left of
You can see more results and experiment with this on your own using
this script.
In short, input dimensions you provide are essentially maximums:
they define an area in which to scale thumbs; they are only ever
decreased to match original images; and one may be adjusted down to keep
the aspect ratio the same as the original and avoid distorting the image.
The most common thumbspage customization options are available as console inputs
on each run, as described in the preceding section.
This section covers additional customization options enabled by file edits.
A set of additional customizations are available as Python settings in file
Updates: in addition to the configs file,
thumbspage 2.2 supports config command-line arguments, of the
form
For more custom behavior, add unique HTML code to the top and bottom
of the index page by placing it in files named
Subtlety: galleries which both use a custom footer and enable a floating Top
button for the index page may also need to allow for extra end-of-page
blank space in the footer's code to prevent the button from covering final
non-fixed content; see the version 2.0 note ahead.
The generated thumbnail index table's code is self-contained, and requires
no support code in a custom header.
Conversely, a custom
For an example of this technique in action, see the
online demo site
here.
Beginning with 2.0, it's also possible to code scripts that replace an HTML
comment automatically added to the
As of version 1.6, viewer pages can also be changed arbitrarily by editing
the template file
Starting in 2.0, you can also customize viewer pages by
programmatically replacing the automatic
As of version 2.0, you can also customize the optional floating
Top button of index pages, by editing the new source-code file
As of version 2.3, you can optionally provide plain-text note files for
any or all of your image files, to provide descriptions or narration
for the images in your gallery. These notes are displayed as a
popup in
viewer pages when users tap the tollbar Note
button,
or up-swipe on the image on touch
displays.
When present, note files appear alongside images in the images folder,
and are named with a
This feature may be disabled, and Note buttons and their up-swipe
appears only if this is enabled and at least on note file
is present in the images folder. For a demo of notes in action,
see this
gallery.
For more details on notes, see the 2.3 release note.
Related: if you use notes, you may also be interested in 2.3's
thumbs-only index pages.
To tie together the ideas covered so far, the following sketches the
structure of an images folder processed by thumbspage, with generated
parts in bold font and default names and behavior applied:
Open the generated
You can view raw examples of generated index and viewer pages and
This section collects assorted pointers for thumbspage gallery builders.
Though arguably random, it covers some of the most common issues
and border cases that may arise when using the program. Notes you'll
find here (be sure to also browse Version History
for tips omitted in this section):
Update: as of version 1.7, viewer pages are
even more functional—with info-dialog popups on filename taps, and
easier Raw-display access via image clicks—and repair some former
large-font issues (see the release notes).
These pages are now broadly recommended.
Update: as of version 2.0, "Raw" button taps are now
fully replaced by image taps, and new "Auto" and "Full" buttons provide slideshows
and one-page fullscreen, respectively; see 2.0's release notes,
and use viewer pages.
Also note that because filenames used on labels are not line-wrapped, their
width largely determines column spacing on the index page. As a rule of
thumb (pun intended...), use shorter filenames for narrower columns.
Update: on a related point, version 2.0 now wraps or
scrolls absurdly long file and folder names; see the release note.
-y
suggested there isn't recommended—it skips asking if you wish to
proceed):
pkg install python ndk-sysroot clang make libjpeg-turbo
pip install Pillow
pip
command in its Terminal or using its Pip GUI, and
other Python apps may provide additional options. To sample the flavor of thumbspage
builds in Termux, check out the Android demos
here,
here, and
here, plus the console log
here.
Running thumbspage
Usage Example
thumbspage.py
from a command line with no command-line arguments.
It can be run from a console (e.g., Terminal on macOS
and Linux, Command Prompt on Windows, and whatever qualifies as
a command shell in Android apps); and after opening its file in most Python IDEs
(e.g., PyEdit and
IDLE).
[]
, and defaults for empty replies are
given in ()
(Windows users: be sure to use py -3
at the start
of your command line, \
instead of /
in your pathnames,
and a C:
or other drive letter if needed):
/.../content$ python3 /MY-STUFF/Code/thumbspage/thumbspage.py
Images folder path [. or dir] (enter=.)? trnpix
Clean thumbs folder [y or n] (enter=y)? y
Thumbs per row [int] (enter=4)?
Thumb max size [x, y] (enter=(100, 100))?
Use image-viewer pages [y or n] (enter=y)? y
Running
Cleaning: trnpix/_thumbspage/1996-first-pybook.png
Cleaning: trnpix/_thumbspage/1996-first-pybook.png.html
...
Skipping: .DS_Store
Making thumbnail: trnpix/_thumbspage/1996-first-pybook.png
Making thumbnail: trnpix/_thumbspage/1998-puertorico-1.jpg
...
Skipping: _cut
Skipping: _HOW.txt
...
Generating thumbnails index page
Generating view page for: 1996-first-pybook.png
Generating view page for: 1998-puertorico-1.jpg
...
Finished: see the results in the images folder, "trnpix".
Prompts and Replies
?
prompt to the last. As we'll see in the
next section, the first of these can now
be provided with a command-line argument instead; this bumps the others up
in the list, but the options work the same either way. Here are the
parameters you'll input to run thumbspage:
folder
means that
folder in the directory where the script is being run, and
/folder/folder
and
C:\folder\folder
denote absolute paths to your image folder on Unix and Windows, respectively.
.
here means the directory where the script
is being run, which is also the default for an empty reply (i.e., pressing your enter
key only). You can also use other path syntax such as two dots for one level up
(e.g., ../folder
is a sibling); see the pathname primer
ahead if you're new to such things.
Tip: if you name a folder which has no images,
you'll make an index page with a subfolder list.
100,100
makes thumbs 100x100 pixels large, but either side may
be rendered less than 100 large if needed to avoid distortion. There's more on
what that actually means in the sidebar at the end of this
section.
control
and type a c
or d
) at any prompt,
and thumbspage reports input errors nicely as demonstrated in this
console log.
For more examples of console runs on various platforms, see the logs
folder.
Other Usage Modes
<
syntax to provide canned input parameters in a text file,
one per line (technically, by redirecting the stdin
stream).
These options yield at least three ways to start the program:
$ python3 thumbspage.py
Images folder path [. or dir] (enter=.)? trnpix
...4 other parameters prompted from input...
$ python3 thumbspage.py trnpix
...4 other parameters prompted from input...
$ python3 thumbspage.py < inputs.txt
inputs.txt
would like look the following; use an empty line to accept a prompt's default,
and omit the first line if you provide the folder name as an argument in the
command line instead:
trnpix
y
4
y
|
and > file
).
For example, the following Unix session both sends input parameters to the
script from one file, and saves its output to another; it also
assumes shell variable $C
has been set to name the folder where
you unzip programs like thumbspage, and opts for 128-pixel thumbnails, 3 per row:
$ cat inputs.txt
y
3
128, 128
y
$ python3 $C/thumbspage/thumbspage.py trnpix < inputs.txt > report.txt
$ more report.txt
...output here...
$C
is your install folder again, and provides canned inputs as tabbed
lines between EOF
markers:
$ cat generate.sh
#!/bin/bash
python3 $C/thumbspage/thumbspage.py trnpix <<-EOF
2
128, 128
EOF
$ bash generate.sh # or just "generate.sh" if it's made executable via chmod
...output here...
<<EOF
and omit tabs if your shell
requires it, and don't expect a 2> /dev/null
to suppress
input prompts here. The latter won't work, because Python's input()
function prompts to stdout
(well, often; this is inconsistent,
and an open bug).
The tagpix program supports
prompt drops this way, but only because it uses a custom input()
that
prompts to stderr
. Such redirects are dodgy in any event;
messages for uncaught errors, including Python exceptions,
may also be lost in the bargain.
examples
folder.
The console input prompts and arguments we've met in this section are
really just the first level of options in thumbspage.
The sections Customization
and Building Tips that follow
cover additional configuration options and usage details.
First, the next section explores more basics for users new to running
programs from command lines.
setting=value
.
When used, these arguments are coded in the command line after the optional folder
path, and override same-named settings in the configs file.
With this extension, thumbspage command lines now take the following general form:
python3 thumbspage.py imagefolder? setting=value setting=value...
python3 thumbspage.py imagefolder inputCleanThumbsFolder=True inputThumbsPerRow=4
A Brief Primer on Pathnames
photos
for an item in the folder you're working in;
.
for the current folder itself; ..
for one level up;
and ../photos
for a filesystem sibling
/Users/you/photos
on Unix; C:\Users\you\photos
on Windows; and
/sdcard/photos
on Android
cd
(change directory) to the folder
containing your source image folder,
and give a folder path relative to where you are working:
$ cd /MY-STUFF/camerauploads
$ python3 /Code/thumbspage/thumbspage.py
Images folder path [. or dir] (enter=.)? imagefolderhere
Or, run the program anywhere and give an absolute path to your
images folder:
$ python3 /Code/thumbspage/thumbspage.py
Images folder path [. or dir] (enter=.)? /MY-STUFF/camerauploads/imagefolderhere
thumbspage.py
script's path in console command lines
can be relative or absolute too, depending on your console's current directory. For instance,
py -3 thumbspage.py
suffices to start the program on Windows if run in
the thumbspage install folder, though your images folder will reside elsewhere—and
may be relative or absolute:
> cd thumbspage-install-folder
> py -3 thumbspage.py
Images folder path [. or dir] (enter=.)? C:\MY-STUFF\camerauploads\imagefolderhere
(width, height)
,
where parentheses and blank spaces are optional.
The underlying Pillow imaging library
always adjusts your inputs as needed to preserve the original image's aspect ratio,
instead of warping images.
=>
are automatically changed to sizes used on the right:
For an original image of size 100w x 50h (wider):
(300, 100) => (200, 100)
(100, 300) => (100, 50)
(50, 50) => (50, 25)
(100, 100) => (100, 50)
(300, 300) => (300, 150)
For an original image of size 50w x 100h (higher):
(300, 100) => (50, 100)
(100, 300) => (100, 200)
(50, 50) => (25, 50)
(100, 100) => (50, 100)
(300, 300) => (150, 300)
For an original image of size 100w x 100h (square):
(300, 100) => (100, 100)
(100, 300) => (100, 100)
(50, 50) => (50, 50)
(100, 100) => (100, 100)
(300, 300) => (300, 300)
Customization
User Configurations File
user_configs.py
.
See that file for its options, documentation on their roles, and
their preset values.
As examples, that file defines Unicode encodings; gives the names of the
generated index page and thumbs folder; turns subfolder-link lists on
or off; as of 1.5, configures most colors; and as of 1.6, allows images
to expand beyond actual sizes, and users to control auto-rotation of
images. Versions 1.7 and
later
add numerous options you'll generally find later in the configs file.
setting=value
.
When used, these arguments override same-named settings in the configs
file, and provide an alternative and simpler way to customize settings
that may vary per gallery build.
For more on this feature, see the 2.2 release note.
Header and Footer Insert Files
HEADER.html
and FOOTER.html
, respectively, and storing these files
in the images folder alongside your images.
You can use one, both, or neither of these files; if not present, generic
HTML and text is automatically generated in the index page around the
thumbs table. For details on how to code these files, see the demos in
folder examples/
here. In brief:
HEADER.html
<!doctype>
,
<html>
, and the <head>
section with all the usual
components if used by your index page—<meta>
tags for mobile viewport
and content-type Unicode encoding, <style>
, <title>
,
<script>
, and so on.
This preamble should be followed by the start of the <body>
section
with any informational <p>
paragraphs or other content.
<body>
margins
(e.g., to appease curved mobile displays), as in
this demo's
source code; default headers do not, because there isn't any substantial text at page top.
<meta>
tag should generally be the same as the outputEncoding
setting in the
user configurations file, because the latter is used when
saving the whole index page, including its generated thumbnails table. See version 1.3
release notes ahead. The UTF-8 default is recommended in all contexts.
FOOTER.html
</body>
and </html>
.
For example, footer content might include navigation-toolbar
links,
or informational text;
see both links for examples.
More Header Options
HEADER.html
file can use CSS style code to
alter the thumbs table in ways not supported by basic user settings (e.g.,
to tailor index-table font). The way you'll do this depends on the
index-page layout model chosen:
<td>
element for the entire index page, using
CSS code like this in a custom header file's <head>
:
<style> td { font-weight: bold; } <style>
<div>
,
with CSS code of this sort in a header file:
<style> #thumbslinks { font-style: italic; } <style>
<style> body { font-family: Arial; } <style>
<head>
of default index headers
and precoded in viewer pages for this purpose. See this note
ahead for more details, as well as this script
for an example that leverages this hook to insert analytics code in generated index
pages prior to uploads.
More generally, this lets programmers add arbitrarily custom code to default index-page
headers, and may avoid custom headers in some use cases.
Viewer-Page Template File
template-viewpage.html
in this program's install folder
(use "view source" in your browser to see its code).
For example, such edits might add site icons, analytics code,
or navigation widgets specific to your usage or site.
Edit with care (this file's code is fragile!) and rerun thumbspage
to use your customized template.
<head>
comments noted in
the prior section and described here. Though this requires some
programming skills, it is probably a
safer option than changing the viewer template in place, especially if you
ever upgrade to a new release (your edits won't be overwritten by an unzip).
Floating-Top Template File
template-floatingtop.html
,
which host the button's implementation code. This isn't normally
necessary, because nearly every property of the Top button can be
customized with settings in the configs file, per the 2.0
change note.
Image Note Files
.note
extension following the
corresponding image file's name. For instance the note for an image
file photo1.jpg
should be named photo1.jpg.note
.
Note files may contain any Unicode text, including emojis, though HTML
code is shown verbatim (links won't work). Lines are collapsed into
paragraphs, with the exception that a blank line coded as two adjacent
line breaks (i.e., \n\n
) is taken to be a paragraph
separator, and renders as a blank line in the popup.
Summary: thumbspage Folders
Your image source folder/
Your image files...
Optional image.note files...
Optional HEADER.html
Optional FOOTER.html
index.html
_thumbspage/
Thumbnail-image files...
Viewer-page files...
index.html
file to view the thumbspage gallery,
and package the entire images source folder to distribute. Exceptions:
HEADER.html
, FOOTER.html
, and image
.note
files, if used, are required at gallery build time only,
and need not be present when a gallery is viewed. This follows from the fact
that header and footer contents are embedded in generated index pages,
and image notes are embedded in generated viewer pages. Shipping these
files to content consumers is optional.
_thumbspage/
folders online
here,
here, and
here,
and their live renditions here.
In addition to the images-folder content sketched above, some files in the thumbspage
install (unzipped) folder are available for user customizations:
thumbspage install folder/
Implementation files...
user_configs.py
template-viewpage.html
template-floatingtop.html
Edit user_configs.py
and the HTML template files
as you like for your galleries. Not shown above, your image folders may also
contain subfolders that might show up in bullet lists on the index page;
but to explore special cases like this we have to move on to the next section.
Building Tips
"
), especially if they are to be
viewed on multiple devices. By contrast, galleries uploaded to a web server need
satisfy only their server's filename rules.
That said, thumbspage galleries can display only image types supported by both the Pillow library used to build their thumbnails, and the web browsers used to view their pages. While Pillow happily creates thumbnails for nearly every image type under the sun (with some exceptions: see the HEIC update ahead), web browsers are much more limited.
TIFFs, for example, yield correct thumbnails and display well in Safari, Edge, and Internet Explorer, but cannot be displayed by Chrome or Firefox. The upstart WebP format similarly is supported by Pillow (and hence by thumbspage builds); but support is an add-on in some versions of Edge and still growing in Safari (and hence in thumbspage views). See this shot for current results in Chrome (top left), Firefox, and Safari (right).
Because of these constraints, the support story today reads as follows:
Some browsers may ask to auto-open some exotic image types in another program, but this isn't the same as in-page support. If your TIFFs (or other) images don't display in a browser you use, your best recourse is to convert them to a more widely supported format for use in thumbspage, such as PNG or JPEG. A "Save as" in your local image editor will generally suffice. For a demo of common supported image types, see examples/mixedtypes. For more on browser image support, try this page or this search.
Also note that the above pertains only to image display. Support for image metadata—of the Exif-tag sort displayed by info popups—varies by image type too. JPEGs are generally tag rich, for example, but PNGs, TIFFs, and WebPs may be more spotty, and image libraries may not recognize newer formats. thumbspage collects metadata where possible, and omits it elsewhere.
One browser-specific caution: a somewhat dated Safari (2015's version 9) on macOS has been seen to crash altogether when trying to display a PBM in the mixed-types demo pages, for unknown reasons that are well beyond the scope of the thumbspage project. Safari updates, image conversions, and other browsers are the suggested remedies.
Update:
in 2021, support for Apple's HEIF/HEIC image format common on iPhones is still not present
in either stock Pillow or Python's mimetypes
module (the latter means it won't
appear in galleries at all), but it might be usable in thumbspage with a third-party library and
programming solutions.
Barring such extensions, your best bets may be to convert these images to JPEGs,
or change your Camera or Photos settings to save or transfer in JPEG format, respectively.
Try searches
here and
here for pointers.
Update: beginning in 2.3, files in the images folder
with a .note
extension are special, and provide image notes displayed in the UI
when present; see the 2.3 coverage. Technically, header
and footer files are similarly special.
Exception: subfolders named with a leading _
or .
character are not included in the bullet list; the former is considered developer
private per Python convention, and the latter is hidden per Unix convention.
See version 1.1's note ahead for more subfolders lists in general;
version 1.7's update to skip .*
subfolders too;
and version 2.1's customizable space between subfolder-list items.
.../folder
and .../folder/
work only when a web server is present. Use the more complete and explicit form
...folder/index.html
to also (or only) view results offline.
To link to individual images, use either their image-file or viewer-page URLs;
the latter gives access to the gallery and its formatting. In sum:
...folder/index.html # the index page ...folder/image.jpg # an image itself ...folder/_thumbspage/image.jpg.html # an image's viewer page
Because rotation changes the source
image file in-place for browser inline display, the original image
is by default first saved with a .original
filename
extension in the source folder as a backup copy. See
user_configs.py
for
settings that allow users to disable this feature and/or its backups;
the included simple
utility script
that restores all originals from backups;
and 1.6 release notes and later updates to them in
Version History ahead for more background
(version 2.1, for example, adds deletion of embedded thumbnails on rotations).
Conversely, folder cleaning is not required if images are only added to the images folder—thumbnails will be made for new images only, and thumbs for previously added images will be retained. This can save substantial time when extending large image folders. But when in doubt, clean; this script's work is done once, before any views.
_thumbspage
by default, in
user_configs.py
. It is not
named with a leading .
character to make it a Unix hidden file,
because hidden files are rude (you should really see what programs do to
your computer), and may be skipped by some compression and backup
programs (e.g., see ziptools and Mergeall at
learning-python.com).
The default name can be freely changed to be hidden if .
tradeoffs are acceptable in your usage (but you didn't hear that here).
The issue: when published online, thumbspage's thumbnail-index pages load quickly, but its image-viewer pages must download images in full before scaling them to browser windows. As a consequence, some large images in thumbspage galleries may download slowly on some servers or clients. In early 2020, for example, large (e.g., 2-6M) image transfers on the business-hosting server that houses thumbspage were usually very quick (e.g., 2-3 seconds), but sporadically throttled down to an outright crawl (e.g., 20-30 seconds). It's unclear if this speed hit was due to traffic on the server itself or the broadband and cellular client networks accessing it, but similar delays may occur in other contexts. Such delays can obviously be discouraging to site visitors, and posting large images in general might even be rude to users with limited or metered bandwidth.
The remedy: if the full-size images in galleries you build with thumbspage load too slowly, your recourses include moving to a faster hosting server, and downscaling the size or quality of very large images. Of these, downscaling images may be easiest, cheapest, and politest. The Pillow library, for instance, has tools that you can leverage in a Python script to reduce image filesize by scaling down image dimensions and quality. Better yet, get the new shrinkpix program for a precoded solution that uses Pillow this way to reduce the size of one or all image files in your website. However you opt to shrink, be sure to regenerate thumbspage galleries to reflect your newly shrunk images' information displayed in info popups.
A faster, and perhaps dedicated, server may help too, but not for users on metered connections or slow client networks; for such visitors, smaller images (e.g., < 500K) may be the only fix, and even then will solve speed issues only if network throttling isn't extreme. The website hosting thumbspage has resisted downscaling its images in the past for the sake of both image quality and usage demonstration, but at this writing is in the process of shrinking images globally. Downscaled or not, if images here are still slow for you, you can always view most of this site's thumbspage demos by downloading and unzipping to your machine and viewing locally. This includes thumbspage's own examples—fetch its full package for quick off-line image views.
In a better world, online images would by now be immune to the scourges of metered access and traffic bottlenecks. In the world in which we code, profit will probably always trump societal need, and the web's mass popularity will probably always best its technological progress.
Update: per testing so far, shrinking large images with the shrinkpix program mentioned above both removes observed slowdowns, and seems to yield faster views. The primary downsides are minor and rare reductions in full-size image quality (described in shrinkpix's caveats), and potentially degraded quality for thumbnails. thumbspage 1.7 solved the latter issue by converting some images to "RGBA" color mode temporarily when making their thumbnails; this produces results as good as for unshrunk originals, and has the added benefit of improving thumbnail quality for some unshrunk GIF images too. See the 1.7 update note and shrinkpix docs for more details.
Update: with the benefit of hindsight, most of the speed issues related to images at thumbspage's site stemmed from web-host speed, not image-file size. Speed issues were finally resolved in full in Spring 2020, by moving this site from its former GoDaddy host to an AWS Lightsail VPS server. The speed increase for thumbspage galleries was staggering, and far surpassed the gain from reduced image-file size (which was so overshadowed by host sloth, that it could not be reliably measured). Still, shrinking images was important for users on metered-bandwidth clients; images at this site are all now 500K or less (and average about half that size), and that's much less to ask of casual browsers than the former 6M bandwidth bombs.
Update: for completeness, it's worth noting that you can shrink a website's images dynamically with Apache's mod_pagespeed (a.k.a. PageSpeed) module. This ambitious module attempts to optimize size of fetched images by automatically resizing and caching for future requests. Unfortunately, you have little control over this process—or the quality of the images it produces. Worse, this module also mangles page code for transmission (a rude nonstarter for sites that double as page-coding resources), and abruptly crashed this site's Apache server when applied by default by a Bitnami install stack (despite the module's experimental "Incubating" status). Your mileage may vary, but static image-size reduction seems a more deterministic—and safer—approach.
Besides its HTML galleries, you can also view and click thumbspage's
generated thumbnails in GUI mode (i.e., without a web browser), by
running the included tkinter
example from the
book
where parts of thumbspage first appeared. This GUI works on Windows
(any version), Unix (macOS and Linux), and probably on
Android (using a launcher script in the Pydroid 3
app).
To start, pass it any image folder's pathname as a command-line argument,
as follows:
/.../thumbspage$ python3 viewer_thumbs.py examples/unicode/images # Unix c:\...\thumbspage> py -3.3 viewer_thumbs.py examples\unicode\images # Windows
This is, however, a very basic and limited viewer GUI (and today even confesses as much). The book's later PyPhoto example is a much more useful GUI, with navigation, image scaling and zooms, and scrollbars for both indexes and images (as seen here and here). More recently, a much-upgraded PyPhoto is now available standalone, at the following site (it's a bundled PyGadget):
http://learning-python.com/pygadgets.html
Also note that thumbspage itself can serve as an image viewer too. Because its results can be viewed both online and offline in any desktop or mobile browser, they can be used for both remote and same-device viewing; your page and browser become the GUI and its host.
Update: running the GUI-viewer command lines above
requires the full examples/
folder, which may not be included in
your thumbspage download package (it may be better accessed online as of version
2.1 due to its
size). See the main
web page
to fetch this folder if desired and required, or use the pathname of a different
image folder of your own.
Update:
as of 2.1, the GUI viewer always stores its thumbnails in a subfolder named by
the THUMBS
setting in the configs file,
which is also used by the HTML gallery builder, and is preset to _thumbspage
.
This ensures that the viewer won't make a separate thumbnails folder, unless you change
the setting between gallery and viewer runs. If changed, the viewer's thumbs folder
may be redundant, but won't appear in gallery index pages as long as THUMBS
starts with a leading _
or .
(the viewer's former default did not).
More on subfolder lists here.
Finally, a word from the legal department. thumbspage has been tested extensively and used successfully on a wide variety of photo collections, and will likely perform well on yours too. It is provided freely because it can help you view and display your image libraries. Given the many ways that computers can fail, however, a word of caution is in order:
Please note:
by design, this program will modify your images folder
in-place, and may modify some of its images after backups.
To the folder, it
adds
an HTML index-page file (named index.html
by default), along with a
subfolder (named _thumbspage
by default) containing thumbnails and
HTML viewer pages. As preconfigured, it also
rotates any tilted images and
deletes their embedded
thumbnails, after saving unmodified backup copies
with .original
extensions. Run this program on folder
copies if you don't want it to change your valued photo collections
directly, and read this guide for full program-usage details.
All that being said, keep in mind that you can easily delete the file
and folder added; original versions of rotated images can be easily restored
with the included script
restore-prerotate-originals.py
;
and both image rotations and thumbnail deletions can be disabled in
user_configs.py
.
Moreover, if you always run thumbspage on a copy of your source
images folder, your originals will never be changed by the program in any way.
Still, the importance of your photos merits a complete understanding of
any tool that modifies them—this one included.
Like much in life, thumbspage improves with time and experience. This section describes changes made in thumbspage releases, most recent first. It goes into implementation details that may be primarily of interest to developers, though users can find additional context here too (and the border between thumbspage users and developers is porous at best).
Before jumping into the releases, here are two global admin notes up front.
First, for readers of code—you can generally find changes made to code by searching
source files for a version number enclosed in square brackets. For example, looking
for [1.6]
will turn up version 1.6 code changes in most
code files; a release year or date in brackets may also work in
some contexts, especially for older releases.
Second, for detail-minded fetchers—if you find that a zipfile or its contents are dated later than the current version's latest release date, it just means that trivial non-code fixes were applied in a rezip. This almost certainly implies minor doc typos (which have a nasty habit of hiding until just after a release). Significant changes, and all code changes, are always called out with a new release date or number here.
For readers who want to save a Top click, here's an index to the version coverage in this section:
thumbspage 2.3, published initially on February 8, 2022, includes three major new features, plus smaller upgrades. This section provides overviews of its new image notes, console-input overrides, enhanced tooltips, and other changes. You can also find a demo of image notes and tooltips in the 2.3 examples folder.
Updates: 2.3 was later repackaged:
cgi
standard-library module
(despite the module's three-decade tenure and millions of clients).
Without the patch, this triggers warnings in Pythons 3.11 and 3.12,
and aborts in 3.13 and later.
A handful of typos were also fixed in docs,
but no program behavior was changed.
\
followed by a non-special character is no longer
allowed; doesn't retain the \
; issues a warning in Python 3.12;
and may be an error in 3.13. The trivial fix is to use forms
r''
or \\
. No program behavior was changed by the fix.
With 2.3, galleries can optionally include notes for images, which are displayed as popups in viewer pages when requested by either taps on Note buttons or up-swipes on touch devices. This provides a way to describe or narrate any or all of the images in your galleries, and opens up an new presentation dimension for both builders and users.
This section covers this feature in full, but if you wish to test-drive notes up front, visit the 2.3 upgrades example, as well as the online training-pictures gallery. Other examples do not yet have notes, because this wasn't available until 2.3; galleries that predate 2.3 were regenerated with 2.3, but those without notes work as they did before. You can also read the users perspective in the viewing section here and here, and the parallel coverages in the customizations section and the configs file.
In more detail, image-viewer pages may now include a Note toolbar button,
which displays the plain text of an imagefilename.note
file near the bottom of the page, if such a file is present in the images
folder at gallery build time.
This feature is enabled by coding one or more note files. It
can be disabled by the config file/argument setting
useImageNotes
, and is automatically disabled if no
.note
files exist for any images in the folder (this avoids
showing Note buttons on viewer pages which would all be no-ops).
When notes are enabled, up-swipe on the image on touch displays is
also changed to open the note, instead of raw-image view. This is symmetric
with down-swipes that open the info popup at top of page. Raw view is
still available via a simple image tap when notes are used, but the
2.2 upSwipeOnAllBrowsers
setting is ignored in this case, because
up-swipe opens the Note popup instead, and works on all browsers;
unlike raw view, no page switch is required for notes.
More formally, image notes is a backward-compatible extension, with behavior that works as follows:
IfuseImageNotes
is False OR no images have.note
files: - No Note buttons are displayed in viewer-page toolbars - Any.note
files in the images folder are ignored - Up-swipes work as they did in 2.2, opening raw-image views IfuseImageNotes
is True AND any image has a.note
file: For images with a.note
file: - The viewer page shows a Note button with normal font and color - Note taps pop up the.note
file's content as plain text - Up-swipes also open the note file's content, not raw-image views For images without a.note
file: - The viewer page still shows a Note button, but with line-through - Note taps and up-swipes work, but show default text "(No note)"
In other words, viewer pages are the same as they were in 2.2 if no note
files are present, but gain Note buttons and redefine up-swipe if notes
are enabled. Because useImageNotes
is preset to True in the
configs file, notes are by default enabled and disabled by the presence
and absence of .note
files in the images folder, respectively.
The setting provides an extra level of control when needed.
Two fine points: first, notes are displayed during Full one-page fullscreens,
but do not stay open during Auto slideshows, because images would be both
shaded and obscured; tap Auto to stop the show if you wish to read a Note.
Second, .note
files, like custom header and footer
files, need not
be present when a gallery is viewed; because .note
files' text
is embedded in generated viewer pages, these files are required at build
time only.
To code a note for an image, simply create a plain-text file with a
.note
extension in the images folder, alongside the image
itself. For example, make text file images/photo.jpg.note
to give the Note text for image file images/photo.jpg
.
Here's a typical note file from 2.3's
demo.
You can create notes for any, all, or none of your images; images without
an associated note display Note buttons with a
line-through and display
default note
text,
and no Note buttons appear if there are no note
files.
In terms of content, any Unicode character can appear in a note file, including
emojis.
In addition, characters special to JavaScript or HTML are automatically escaped
and render verbatim. '
and \
, for instance, appear
literally and do not impact note text, and <A>
displays as is
rather than producing a hyperlink (notes are plain text by design, for simplicity).
For a demo of special-character escapes, see the note
here.
Note content loaded from .note
files is decoded per
the Unicode config
setting noteEncoding
,
which is preset to UTF-8. You can use any
text editor to create note files, but be sure that files are saved in a
Unicode encoding compatible with this setting if non-ASCII characters
are present.
When note-file contents are displayed, the viewer-page popup collapses all
text into a single paragraph, except that an empty line coded as two adjacent
line breaks in the note (i.e., \n\n
in Python, with no spaces
between), is treated as a
paragraph break in the popup, and renders as a blank line.
You can use this to code as many paragraphs as you like,
though as a guideline, notes should be relatively short, especially
for use on mobile (a picture is, after all, already worth 1k words).
The Note popup itself expands and shrinks with the window, and its text
scrolls vertically if too tall for the page. If desired, the width of the
popup can be adjusted by the config file/argument setting
noteBoxVSpace
, which gives the CSS size for both left and right
margins, and can be any size type (e.g., 10%
or 5px
).
The preset 15%
makes the popup span 70% of the page, and reflects
a compromise between mobile (where wider is better) and desktop (where the
popup tracks the width of the window).
Tip: if your gallery includes images notes, filename labels might be superfluous on its index page. See the thumbs-only index page option ahead to omit these labels if they don't add any value to your gallery.
One potential downside of using individual .note
files
is that they are somewhat scattered, making it difficult to treat
them as a set (e.g., to spellcheck all notes before publishing a gallery).
To address this, 2.3 also includes a utility script,
batchnotes.py
, which
collects all of an image folder's note files into a single text file,
with separator lines and sorted by filename. The result can be easily
pasted into a spellchecker, or otherwise managed.
See the script for usage, and see its output for the 2.3 upgrades example, as well as the online training-pictures demo. In principle, this script's output could be split into individual note files (and hence provide an alternative way to code them), but no use case for this has arisen thus far.
Like info popups, 2.3's Note popups initially inherited the color scheme of viewer pages at large, whose preset was white text and border on black. This generally makes sense because additional colors can clash with gallery images, but some colors may make text difficult to read on some displays, especially for larger notes. Bright white text on full black, for instance, may be too glaring for some readers. To compensate, 2.3's latest May 2022 release allows builders to customize the background, foreground text, and border colors of both Note and info popups. Any or all of these three colors may now differ arbitrarily from those of the viewer pages in which the popups appear.
To tailor these popups' colors, assign the new popup*Color
settings
in either the configs file or its command-argument
equivalents. Their preset defaults of
None
mean the settings inherit their values from the enclosing viewer
page as before, for backward compatibility (technically, None
selects
the corresponding viewer*Color
setting). If any of the three new
settings are not None
, their values are applied to popup dialogs.
The following build commands, for example, may improve text readability by specifying
muted white text to minimize glare on the default black background; off-white text on an
off-black background; and a black-on-grey popup scheme:
$ py3 .../thumbspage.py images/ popupFgColor=\'#dddddd\' $ py3 .../thumbspage.py images/ popupBgColor=\'#181818\' popupFgColor=\'#cccccc\' $ py3 .../thumbspage.py images/ popupBgColor=\'darkgrey\' popupFgColor=\'black\' popupBorderColor=\'white\'In these settings' names,
Fg
means foreground text, Bg
is the dialog's background, and Border
denotes borders around
both the dialog and its OK button.
When used, these new popup colors are applied to both Note and
info popups, but
not to viewer pages at large, whose toolbars, filenames, and image borders
still follow viewer*Color
viewer-page settings.
While graphic arts is a topic beyond this doc's scope, popup colors
as a rule should blend well with the viewer page. In accommodation,
popup OK buttons' background and foreground colors by design
still match the page, not the popup; only their borders respond to the new
popup color settings.
To sample the effects of these new configs, see the 2.3 gallery captures starting here, as well as the training-photos gallery. The former's notes give additional tips and command examples, and the latter's generate and publish scripts use command forms like those above to make popup text more readable. As part of this upgrade, OK buttons also morphed in appearance slightly, and gained rounded corners to match the enclosing popup box; this is a visual change, but the mod may vary by browser, and is too minor to warrant fanfare (or a new version number).
With 2.3, all console inputs can now be provided with config-file or config-argument settings, instead of actual inputs. This resolves an inconsistency that arose in version 2.2: config command-line arguments were allowed to override settings in the configs file, but a disjoint set of parameters was still always input at the console (technically, from stdin).
As of 2.3, these inputs can now be provided by either config-file or config-argument settings, but are still asked by default so existing builds are not affected. The net result provides a portable and uniform alternative to shell input techniques, which is backward compatible with earlier releases.
inputImagesFolderPath = None # string 'path' (iff not first argument) inputCleanThumbsFolder = None # True or False inputThumbsPerRow = None # integer (iff not dynamic layout) inputThumbMaxSize = None # 2-tuple of ints (or int x == x,x) inputUseViewerPages = None # True or False
Of these, inputImagesFolderPath
is never asked (and is
ignored) if the images folder path is provided in the command's first
argument
(per 1.7);
and inputThumbsPerRow
is never asked (and is
ignored) if dynamic index-page layout is used
(per 2.1).
Any or all of these settings may be provided in the config file itself,
or as setting=value
command-line arguments,
which were added in 2.2 and take precedence
over the file's settings. However provided, these settings may
override inputs as follows:
None
, the corresponding input is requested
in the console as before.
None
, its value is used for the
corresponding input, and the input is not requested in the console.
For backward compatibility, all input-override settings are preset to
None
in the config file, to ask by default. Hence, this requires no
changes in existing build systems that predate 2.3. When used, however,
input-override configs make command recall more useful, and can replace
shell < inputs
redirects and <<EOF
here
documents discussed earlier,
which are not generally portable across platforms and shells.
About precedence: with this change, build parameters may be provided in three different forms. When multiple setting forms are used, command-line arguments have the highest precedence, followed by the config file's variables, and then console inputs. That is, arguments are used first and inputs are tried last, and inputs are not requested if overridden by either argument or file. Special case: for the image-folder name, the first command-line argument, if provided, has higher precedence than all three other parameter sources.
C
is a shell variable
set to the folder where you unzipped thumbspage's source-code (extrapolate
as needed on non-Unix
platforms):
Folder by first arg, dynamic layout, inputs by config file or console $ python3 $C/thumbspage/thumbspage.py . useDynamicIndexLayout=True Folder by first arg, dynamic layout, all inputs by args $ python3 $C/thumbspage/thumbspage.py . \ useDynamicIndexLayout=True \ inputCleanThumbsFolder=True inputThumbMaxSize=128,128 inputUseViewerPages=True Folder by first arg, fixed layout, all inputs by args $ python3 $C/thumbspage/thumbspage.py trnpix \ inputCleanThumbsFolder=True inputThumbsPerRow=4 \ inputThumbMaxSize=100,100 inputUseViewerPages=True Folder by config arg, fixed layout, others by config file or console $ python3 $C/thumbspage/thumbspage.py inputImagesFolderPath=\'.\' Folder and all others by config file or console $ python3 $C/thumbspage/thumbspage.py
As a guideline, use the config file for settings that are defaults or universal, and either arguments or console inputs for settings that may vary from build to build. Console inputs can still be used by build scripts coded to provide them before 2.3, but their config overrides are handy for launching a build quickly with shell command recall, and are a natural extension to the config-argument model.
Nit: inputs provided by config file or argument are not error checked as much as actual console inputs. Configs are checked for valid Python syntax, but may yield invalid values or types which won't generally cause harm but may trigger exceptions and aborts during a build. thumbspage assumes that builders clever enough to edit the config file or pass command arguments are also clever enough to provide valid inputs. Still, lapses aren't out of scope for even the best of us; builders beware!
With 2.3, tooltips are now enabled by preset default, because viewer-page toolbars are now busier when Note is used. In addition, tooltips have been added for the Prev, Next, and Index buttons for consistency, and the former HTML hrefs used as a fallback for navigation buttons have been removed, because the mouseover URL popups they invoked are redundant with the buttons' now-default tooltips, and were noticeably more obtrusive. As before, tooltips do not work on mobile browsers, even with a stylus or mouse.
Caveat: the removal of hrefs now makes Prev, Next, and Index unusable without JavaScript in 2.3, because the hrefs formerly provided a fallback when JavaScript was disabled. This is a backward-incompatible change, but given that viewer pages were almost completely unusable without JavaScript already (e.g., images were not sized to the display, and other widgets did not work), this seems a small cost to pay for avoiding the former and distracting mouseover URL popups. It does, however, leave no usable widgets without scripting.
Hence, thumbspage viewer pages now officially require JavaScript. This was previously resisted, but JavaScript is now generally mandatory on the web at large, and is a necessary and reasonable tradeoff for enhanced utility of the sort provided by viewer pages. As before, a message is posted at top of page when JavaScript is disabled, to let users know that pages are not operative. The Note, Auto, and Full buttons are also omitted in this case, as another visual (if perhaps inconsistent) clue; Prev, Next, and Index remain for show and layout, but are no-ops sans JavaScript.
The usual basket of smaller enhancements found its way into 2.3:
popupOpacity
in the config
file or a command
argument.
This new setting gives the opacity of the
page overlay: a higher value makes the background dimmer (darker),
and a lower value makes the background brighter (lighter).
Strictly speaking, values may range from 0.0 (transparent) to 1.0 (full blackout).
The new config's preset 0.40 (40%) matches prior versions and is a fair compromise, but brighter may be better when notes refer to images (so images can be better seen), and darker may be better for long notes or bright images (so notes get more focus and images don't outshout note text). For a brief demo, see the screenshots starting here.
sed
config-file edits and
<<-EOF
here documents.
Most examples' build scripts don't use the new tools yet,
but one shows how,
and another
does.
The online training gallery's
generate and
publish scripts
provide another example
of the simpler new-style builds.
inputThumbMaxSize
command-argument equivalent. This value must still
be a 2-tuple when input from the console. When given as an integer value X
in configs, however, it is automatically treated as a 2-tuple (X, X)
,
with the same value for pixel width and height (i.e., with square bounds).
Because this is the normal case,
the new single-value config provides a convenient shorthand.
For the full story on this change, see the
2.3 update in generateDynamicThumbsLayout()
of file
thumbspage.py
,
as well as screenshots starting
here.
As before, configs can be used to fine-tune
cell layout out as desired: see the
file and
example.
This largely removes the "experimental" status of dynamic
layout—enough so that the training-gallery
demo
now comes in a dynamic
flavor.
Despite its mobile scrolls, though, fixed is still the default, because dynamic
may still show a single column on some phones and leave empty right-side space
in some windows.
Foldable mobiles might render some such
tradeoffs moot, but it could be a long wait.
Enable this option with new config omitIndexPageLabels
,
and tweak its cell spacing in dynamic layout with configs
dynamicLayoutPaddingH/V
per the config file's
docs; fixed layout sets spacing automatically.
Enabled or not, images are still sorted by filename for sequencing, and
filenames are still displayed at the top of viewer
pages for reference.
Thumbs-only index pages may be useful for galleries where thumbnail images suffice; filenames aren't descriptive or useful; space is a premium for mobile; or 2.3's image notes make filenames extraneous. Filenames generated by cameras, for example, might be pointless, especially if notes provide photo descriptions. On the other hand, this partly upends the paradigm of presenting a folder of images (thumbspage began as an alternative to browser views of image folders, and later, images themselves); and image notes may be less convenient (notes can say more, but informational filenames may be easier). As usual, your gallery is yours to craft.
\
characters in .note
files are now properly escaped
so they appear in viewer pages' Note popups. This cropped up only for notes containing
backslash characters (open the popups in this
example and
demo).
With the repair, \
and all other characters special to JavaScript and HTML
display verbatim and don't derail pages.
INDEX
config, instead of hardcoding index.html
.
This config can vary per build and has been present since thumbspage's
origin in 2016, but was never utilized in six years. The hardcoding
came to light when the training
gallery
added a second index page for
dynamic layout:
without using the config, Index taps in the dynamic
variant's viewer pages incorrectly returned to the fixed variant's index.
Atypical but true.
Bonus: info popups now use the first fix's code to escape their device/software name too. Special characters were never known to be an issue for info, but arbitrary text can be, well, arbitrary.
thumbspage 2.2, published on December 8, 2021, includes three major new features, plus smaller upgrades. This section provides overviews of its new swipe gestures, command-line config arguments, optional tooltips, and other changes. You can also find a demo of config arguments and galleries of tooltip screenshots in the 2.2 examples folder.
Version 2.2 adds support for swipe gestures on touch displays. This works in all dozen browsers tested on both smartphones and Windows PCs, except for Internet Explorer. Where supported, swipes originating in the image-display area of viewer pages move to next and previous images (for left and right swipes); open the image-info popup (for down swipes); and open image raw view (for up swipes, where supported per ahead).
Swipe usage is described in more detail earlier in the viewing section. These gestures are alternatives to existing tap actions—specifically, taps on Prev and Next buttons, filename widgets, and images—and the former taps (and their mouse-click equivalents) still work as before on all platforms. These gestures also do not preclude other touch gestures: long presses, pinch/spread zooms, and screen-edge back/forward swipes still work where supported earlier.
Because the direction of left/right gallery movement is somewhat arbitrary
(should it reflect Prev/Next toolbar button order or not?), this is
changeable in the configs
Fallback: thumbspage's swipes play well with default actions in all browsers
tested, and are too useful to relegate to optional feature.
If they ever prove to conflict with other gestures important to your use
cases, however, you can disable viewer-page touch gestures completely by
simply commenting out the JavaScript template
code that registers touch handlers; search for "DOMContentLoaded" in the template.
To sample swipes live, use a touch-screen device to open any of the demos listed
here.
Update: thumbspage 2.3 changes up-swipes to open
image notes instead of raw views, when its new Note feature is enabled. This
up-swipe parallels down-swipes for info popups. Unlike raw views, this also works
on all browsers, because it does not require a page switch. Hence, the convolution
of up-swipes—and the following notebox—go away completely in 2.3 when notes
are enabled. When notes are not used, however, up-swipe is still raw view as it was
before, and comes with all the same barbs.
See the 2.3 release note for more details.
The glitch:
in Chrome and its ilk, switching to the raw image's URL in touch events frequently
fails to push the calling page onto the history stack, despite using
code identical
to that which works well for image taps. The net result is that a later Back
does not return to the viewer page, which is both confusing and frustrating.
This
message
from Samsung Browser seems germane, though this browser allows Back to return
to viewer pages, and users are not returning to a page they did not visit.
This is clearly a misclassification; enabling popups and redirects in Chrome
does not fix this bug, and would be far too much to ask of users if it did.
The scope: this Back-after-swipe glitch occurs in Chrome and browsers
that share its code base (Edge and Opera) on Android and Windows.
It does not occur in Firefox on either platform, the Samsung browser on Android,
or any browsers on iOS (which all share the same code base per Apple edict).
Internet Explorer and other platforms are moot because they
don't support thumbspage touch events (IE implements a different
pointer-event API, and has been rendered nearly unusable on Windows as of late 2021).
The work-around:
rather than penalizing all browsers for a Chrome issue, up-swipe has been
coded to work on browsers known to support its action today, but may have to be
revised if Chrome's bug is ever repaired. In the interim, if a fix appears in
Chrome before thumbspage can be rereleased for it, you can set
For now, raw views are still available on all browsers via image taps,
which work flawlessly, and are arguably easier than swiping the same widget anyhow.
Sports fans: up-swipe was also tried as an equivalent to Index (but Back again
failed to return to the gallery); as well as Auto and Full to avoid
a page switch (but Auto was too subtle, and Full didn't
work most of the time—and was much too jarring when it did).
Fix please, Chrome.
Version 2.2 allows any number of config-file settings to be passed in as
command-line arguments. When used, these arguments override same-named settings
in the config file, and are generally more
convenient than config-file edits for customizations that vary per gallery.
Config-file arguments are coded in commands used to run thumbspage builds,
provide both a setting name and value separated by
As a Python expression, the
In the past, changing settings for a given gallery required either manual
edits to change and restore the configs file, or automated but perilous
edits in build scripts.
Some of this site's gallery build scripts, for example, resorted to
For instance, a gallery which wishes to opt in to 2.1 dynamic index
layout
and 2.2 tooltips,
change the index page's colors to a darker
theme, and mod
the order of 2.2 left/right touch swipes can now
be built with the following sort of command-line arguments instead of config-file edits,
from either the console or build scripts (some Windows
tools
use
If the images-folder path (e.g.,
Subtleties: command-line configs are applied after settings in the
configs file have been imported. This works well, but may lead to unexpected behavior
if the configs file assigns one setting name to another: if
Nit: with command-line configs, there are now three ways
to tailor generated galleries—configs files, configs arguments,
and console inputs. This is largely an artifact of this program's evolution.
Arguments are simply selective overrides for the file's complete settings.
Console inputs are disjoint from the others, and seem redundant with the new
arguments; while they could be made
file/argument settings to simplify this story, this would be a
backward-incompatible change whose costs are well over budget.
For a complete demo of config arguments in action, see the Bash script
here.
Update: thumbspage 2.3 added new config settings
that can be used to override console inputs. When provided, in the config
file or command-line arguments, input values are taken from the settings, and no input
is requested at the console's stdin. This leverages 2.2's config arguments, and
resolves the inconsistency nit noted above.
For more details, see the 2.3 release note and its command
examples.
Version 2.2 adds optional tooltip popups. When enabled, the tips appear
on mouse hovers over index-page thumbnails, as well as viewer-page image,
filename, and Auto and Full buttons. These use simple
Because they might also be distracting, and pointless (if not annoying) after initial
usage, tooltips can be turned on or off at build time via a new
Musing: HTML5
For a first-hand look at tooltips, see both the screenshots and live tips in
the gallery here.
Update: thumbspage 2.3 makes tooltips enabled
by default, because the viewer-page toolbar is busier with the new Note;
adds tooltips to the Prev, Next, and Index buttons for consistency; and drops
HTML hrefs on all viewer toolbar buttons to avoid URL popups, which were redundant
and obtrusive. See the 2.3 release note for more info.
In addition to swipe gestures, config arguments, and tooltips, 2.2 adds
the following enhancements:
lrSwipesPerButtons
is True
,
left/right swipes move to previous/next images, respectively; if False
,
left/right swipes move to next/previous images instead. Because the latter is more
common and "natural," it is also the default preset.
upSwipeOnAllBrowsers
in the configs file
(or its arguments, up next) to enable up-swipe for all browsers.
Setting this to True
allows you to both test for a
fix, and activate support in galleries.
Command-line Config Arguments
=
,
and appear after the folder name if one is used:
python3 thumbspage.py imagefolder? setting=value setting=value...
In this, setting
is the name of any variable assigned in
the config file; value
is any Python expression whose result is used for the config setting;
and there should be no spaces around the =
characters.
value
part should use
quotes around strings and escape them as needed (e.g.,
\'
or \"
), and any other characters special
to your shell should be similarly escaped as required (e.g., \&
).
Windows escaping rules naturally
differ,
unless you're using its Linux
subsystem.
Config arguments with invalid names or values that produce
exceptions on evaluation cause the gallery build to be aborted,
with a message in the console.
sed
editor scripts on Unix to change a handful of settings
in temporary file copies (here's an intentionally retained
example). With arguments,
settings that deviate from the norm are much easier and safer to tailor per gallery.
^
for line continuation instead of Unix's \
):
# single line
...thumbspage$ python3 thumbspage.py gallery/ useDynamicIndexLayout=True useToolTips=True thumbsBgColor=\"black\" thumbsFgColor=\"#ffffff\" thumbsBorderColor=\'white\' lrSwipesPerButtons=True
# multiline
...thumbspage$ python3 thumbspage.py gallery/ \
useDynamicIndexLayout=True \
useToolTips=True \
thumbsBgColor=\"black\" \
thumbsFgColor=\"#ffffff\" \
thumbsBorderColor=\'white\' \
lrSwipesPerButtons=True
gallery/
here) is omitted,
it will be requested in the console as
usual.
Any config-file setting can be overridden this way, and any setting not
listed in the command still uses its value in the file
as before. This is especially handy to enable dynamic layout, which is still not
a default in 2.2 despite its growing
usage
(but there's always the next release).
A=B
appears in the file and B=other
appears as an argument,
A
will still reflect B
's prior file value, not its
other
argument value. If this arises, simply reset
A
as an argument to the same value as argument B
.
Also note that setting values cannot name other settings when coded as arguments;
don't use variables in argument values.
Optional Tooltips
title
attributes with short text (e.g., "View image" and "View info"), and do
not appear on tested mobiles even with a stylus or mouse.
useToolTips
setting in the configs file and its equivalent
arguments (e.g., useToolTips=True
).
The default preset is False
for off (disabled), but this is nearly
too grey to call.
localStorage
might be used to display tooltips only
on first visits to a gallery, by checking a visits counter, and setting
title
s in JavaScript instead of HTML. This was rejected because
browser storage can be disabled or deleted by users,
may record differing values across browsers and devices,
and has quirks in some browsers that qualify as
nonstarters.
Etcetera: OK, UI, Local Views
alt
attributes to HTML image tags for completeness and
accessibility, though they are moot in many use cases. In both generated index and
viewer pages, these appear alongside title
attributes added for
the tooltips described earlier.
#f5f5f5
to the new
lightgrey
.
Though potentially impactful, this is just a preset default, and a partial
concession to the trend towards dark modes. You can change it arbitrarily
with file or argument configs,
and a true OLED dark mode matching viewer pages' preset
black
can be easily
had—see the example above.
Be sure to change any auto-edits to look for the config
file's new color, or use config arguments instead.
dynamicLayoutPaddingH
(which is applied regardless of label
and thumb widths).
The new spacing allows thumbnails with narrow labels to be squeezed closer together.
defaultFooterTagline
.
These messages—"Gallery built by thumbspage.py" and
"Page built by thumbspage.py" since 2.1—appear
by default at the end of such pages with a link to this program's website, like
this and
this.
As before, they can also be omitted with a
custom but simple FOOTER.html
file,
but the new config makes it easy to avoid the mild plugs when the default
footer is used.
Examples start
here.
thumbspage 2.1, published in final form on August 18, 2021, includes two major new features, plus a handful of smaller upgrades. This section provides overviews of its improved thumbnail images, dynamic thumbnail layout, and other changes.
As usual, you can also view the graphical effects of these changes in the new 2.1 galleries here and here, and explore their code in the files referenced ahead. Also as usual, 2.1 download packages newer than the publishing date above have only minor documentation and other edits which do not alter program behavior; the most recent of these non-functional updates was on August 31, 2021.
2.1's first major new feature is the addition of thumbnail-nail enhancements, both automatic and customizable. Eight new settings in the user-configurations file support arbitrary changes to thumbnail color, contrast, sharpness, and brightness, as well as the quality level used for saves.
Among these, two precoded enhancements are preset to be on by default:
The net effect of these two makes thumbnail images noticeably clearer on thumbspage index pages in all galleries and displays tested, at a trivial cost in extra storage requirements measured in just kilobytes per gallery in all test cases. Moreover, the improvement is visually identical for galleries generated on macOS, Windows, Linux, and Android, using Pillow 7 and 8.
In addition, another precoded thumbnail enhancement provides a black-and-white thumbnails mode, and users may tweak other enhancements' settings arbitrarily for more custom effects (e.g., to lighten thumbnails for darker photos).
For a brief look at the thumbnail presets and other settings in action,
see the new 2.1 upgrades
gallery.
For more details on this release's thumbnail changes, see both the
new configuration settings and their ample docs in
user_configs.py
,
as well as the implementation-level details in
viewer_thumbs.py
.
In the unlikely event that the two new preset enhancements won't work for your
use case, you can revert to 2.0 thumbnail appearance by simply changing
these presets' settings to False
in the first of these files.
2.1's second major new feature is the addition of a new layout mode for thumbnail index pages, which arranges thumbnail columns dynamically to match page size, and rearranges them on page resizes. With this new mode, galleries can now choose from two layouts for index pages:
Of these, the new dynamic layout works well on desktop displays, where pages can be freely expanded and shrunk to display more or fewer columns, and responds similarly to page resizes on popup and split-screen app windows on mobiles. This can make better use of available space, and avoid some horizontal scrolls. Because it adjusts to match display size, this layout is also arguably more in line with the responsive web-design paradigm.
On more rigid mobile displays, however, the new mode seems subpar: in some contexts, galleries may be displayed with just a single column in portrait mode (and perhaps more in landscape), which eliminates minor horizontal scrolling, but can make for radically more vertical scrolling in non-trivial galleries. The impact of this may vary per phone and content, but is typically glaring.
For example, compared to a fixed four-column table, the dynamic model can require four times more vertical scrolling on mobiles that display a single column. For a 100-image gallery, this may mean 20 pages to scroll through instead of 5, depending on display size. Even assuming one extra horizontal scroll per page of interest, the fixed scheme still saves substantial navigation work—and the user frustration it tends to invoke. Moreover, the new dynamic model:
Because of these tradeoffs, the new dynamic layout is provided as an
experimental alternative to the former fixed-column
tables,
which are still available, recommended, and default in 2.1.
To enable the new dynamic layout for your gallery instead, simply
change the new useDynamicIndexLayout
setting in
user_configs.py to True
prior to gallery generation.
If you do enable dynamic layout, you can also customize the amount of
padding between thumbs by setting either or both of
dynamicLayoutPaddingH
and dynamicLayoutPaddingV
to any CSS size in the configs
file.
The former is used for horizontal padding (both left and right) and is applied
whether thumbs or labels are wider; the latter is used for
vertical padding (both top and bottom). These settings provide more control over
layout, by expanding or collapsing
space.
Horizontal padding is also a fallback option if program choices prove unwanted,
though layout also depends on the widths of labels and thumbs.
To evaluate the new dynamic-layout mode for your use cases, you can catch it live at its new example gallery, or browse all its screenshots. To dig deeper into the implementation, you can view the source code of a demo page, or study the index-page code generation portions of thumbspage.py.
Related change: see also this doc's section on console inputs; dynamic layout removes the thumbs-per-row prompt, because this is automatically determined from page size when this layout is selected.
Update: dynamic layout is still not the default in version 2.2, but that version makes it easier to enable this option per gallery with config arguments in gallery-build command lines. Simply use a command like the following at a console or in a build script, and skip config-file edits:
python3 thumbspage.py imagefolder useDynamicIndexLayout=TrueFor more details, see the complete coverage of the new config arguments in 2.2 release notes.
Update: dynamic layout's horizontal spacing scheme was improved in version 2.3. It now packs columns tighter, which allows some galleries to render as more than one column on some mobiles. See the 2.3 release note for details; this mostly removes the "experimental" stigma of dynamic layout, though it's still not the default because it can leave some space empty on the right for some window sizes.
Beyond its new thumbnail-image and index-layout features, 2.1 also includes an assortment of smaller upgrades:
.
and ..
are not expanded because doing
so can be ambiguous (e.g., a subfolder of .
may
have the same name as .
).
subfolderSpacer
in the user-configs file.
This string setting can be changed to any CSS length-unit
value.
Its preset default was also increased from 6 CSS pixels to 7, for easier taps
on mobiles. To see the change live, view its 2.0-versus-2.1
screenshot,
or run the live
demo—ideally
on a mobile device, to gauge its improved tapping support.
For such imageless folders, 2.1 also drops the "Image" in default-header titles (it has no images); specializes the default-footer tagline to "Page built by thumbspage.py" (it's not a gallery); avoids issuing spurious console queries (no thumbs or viewer pages are created); and tailors the final build output line to say "imageless folder." For a typical use case for imageless-folder pages, see the 2.1 dynamic-layout gallery.
For parity, the Full button similarly changes its font as an indicator that it has invoked fullscreen mode and can be tapped to cancel it, but this isn't as useful as it is for Auto, because Full lasts for just one page, and its activation is obvious. Unlike Auto, Full font changes are also not supported by all browsers, and Full will not change for, and will not be able to cancel, a fullscreen mode invoked by a user outside the page (user-initiated and JavaScript-initiated fullscreens differ). Full does, however, change for cancellations invoked apart from the button (e.g., Escape keys).
Formerly, some such embedded thumbs could confuse some tools (e.g., file explorers) into displaying tilted thumbs, because they were not rotated along with the main image. Deleting these thumbs is a simple yet reasonable fix, given that rotated images are meant for use in thumbspage galleries; thumbspage makes and stores its own thumbnails separately; other tools will do the right thing without an embedded thumb, because they already must for other images; and the original image along with its embedded thumb is always saved to a backup copy prior to rotation.
For both more details and a demo of the new removal at work, see the new console log. As detailed there, this issue's scope was initially isolated to a single context (photos taken on a Galaxy Note 9 and viewed as thumbnails in macOS Finder post rotation), but was later seen to impact some other file explorers and images shot on other cameras, and may be aggravated by tools that add thumbnails covertly. The reorientation demo was also rebuilt with the 2.1 fix; see its image's before and after scene in macOS Finder.
As a fallback: if you're convinced that this isn't a
concern for your images and tools and wish to minimize image changes, you can disable
2.1's thumbnail deletions on rotations by changing the new setting
deleteEmbeddedThumbs
in the configs
file. Either way, you can always restore the
original versions of rotated images—including their embedded thumbnails—from
their .original
backups.
Subjective comment: embedded thumbnails in image files seem a very bad idea;
they aren't recorded by all devices, double the workload of both developers and
image-processing programs, and are prone to grow badly and arbitrarily out of sync
with main images over time. Some tools still naively use them anyhow, thereby
propagating a design flaw that makes redundancy a built-in. See also
getUpdatedExifData
in the code.
Second, because its examples folder passed the 200M mark, 2.1 now provides multiple zips for users to choose from. From smallest to largest, here are the new downloads and their sizes per macOS Finder (last updated for the 2.2 release; see the main web page for current sizes):
Beyond all this, 2.1 polished its code and docs as usual. Explore the package at large for more details.
RFC: if you have a suggestion regarding thumbspage 2.1's defaults for thumbnail quality, thumbnails layout, or subfolder links, please send feedback by email. The defaults suffice for thumbspage's host site, but your mileage may naturally vary. As a rule, thumbspage strives to avoid making choices for you, and all 2.1 defaults are configurable per gallery and builder by design; as always, tweak as desired.
thumbspage 2.0, released in final form on October 5, 2020, adds an automatic slideshow and optional one-page fullscreen on image-viewer pages; an optional floating Top button on index pages; a custom dialog and device line for info popups; improved overflow styling everywhere; and a resolution to an iOS Chrome history bug.
This version also changes the UI significantly (hence the 2.0): index pages grow an optional Top, and viewer pages replace Raw with Auto, sprout an optional Full, modify toolbar layout, and change info-popup appearance and content radically. To sample 2.0's new displays, see either the live demos on its hosting site, or the latest 2.0 screenshots. The latter supersede captures from older releases which have not yet been retaken (alas, this task may be preclusively monumental).
Version 2.0's enhancements were rolled out over four 2020 releases, from June 26 to October 5. The rest of this section drills down on each of 2.0's changes listed in the table below, from earliest to final.
In complete terms, 2.0 adds the following upgrades, with major items colored black for emphasis:
Though the new slideshow is largely automatic, the delay between image pages
can be configured on a per-gallery basis at build time: see the new
autoSlideShowDelayMS
setting in the
configs file. This delay setting is preset to
5 seconds, which seems a reasonable balance of user goals (in 2.1, it briefly
changed from 5 seconds to 4, but was later reset to 5 for usability, as explained
in the configs file).
To keep the interface simple, the delay is not changeable by gallery users,
though they can turn slideshows on and off at any time and on any viewer
page by tapping Auto.
In viewer-page toolbars, the new Auto button replaces the former Raw button, which grew redundant with the new image taps added in 1.7; tap the image for the former behavior of Raw (and mind any older screenshots that still display it). The new Auto button starts and stops the show's progression on demand, and image slideshows are paused when a gallery is exited; they may be resumed on both returns with a browser Back and new selections from the index page, and Next and Prev may be used to skip images.
A slideshow's scope spans all viewer pages visited in a browser tab, because it is based upon the tab's session-state storage. Technically (or at least conceptually), Next and Prev, like any page exit, cancel a scheduled timer event automatically, but the new page schedules a new timer event when loaded; the net effect is to continue the show from the new page. Slideshows also generally run in backgrounded tabs, but this is prone to vary per browser. This is all subject to the peculiarities of browser back-forward caches (and may be partially broken on Safari), but largely works well on all the many browsers tested.
For more slideshow usage details, read the earlier viewing coverage. For additional implementation details, see the initial proposal ahead, or browse the viewer-page template's code. And for concise demos of the sort of code used for cross-page state and slideshow timers, see and run the pages in the example folders here and here.
Update: version 2.1 later enhanced slideshows by changing the Auto button to underlined font as an indicator whenever a slideshow is on. This was a minor but crucial missing feature of the implementation, because the toggle's state was otherwise unclear, especially when first starting a slideshow, or returning to it from another page. See 2.1's release note for more info.
Constraint: the new Auto slideshow requires JavaScript. If JavaScript is disabled in the browser, the Auto button is still displayed but has no effect, and users will be shown the no-JavaScript warning which also apples to dynamic image scaling and more. Given that thumbspage galleries now require JavaScript for info popups, image scaling, Auto slideshows, Full one-page fullscreen, and the next section's floating Top, users can be reasonably expected to enable JavaScript for a more rewarding UI experience. See also the viewing note for another take on this requirement.
The new Top button is intended for larger indexes with useful top-of-page content. It won't appear in small indexes with little or no vertical scrolling, and is generally unrecommended clutter for indexes that have nothing worth scrolling to at the top of the page. It's enabled by default nonetheless, because some pages do have useful top content, large indexes require substantial scrolling on mobiles, and users may wish to start the new automatic slideshow at the first image in the gallery (see the prior section).
See the configs file for settings that customize the new Top button. In short, gallery builders can:
floatingTopEnabled
floatingTopAppearAt
floatingTopSpaceBelow
floatingTopFgColor
and floatingTopBgColor
template-floatingtop.html
which hosts its code
In addition: galleries that use a custom
FOOTER.html
file may need to
add space below a final non-fixed content element, to prevent Top
from overlaying and hiding it. This is accommodated automatically in default
index footers with a style of this sort, to leave blank space after the final
paragraph:
<P style="margin-bottom: 80px;">
The effect is captured
here.
Depending on its content, a custom footer may wish to do
similar.
Naturally, <br>
s and padding-bottom
can have
the same effect and a surrounding <div>
may help, and
be sure to also allow for horizontal scrollbars on desktop browsers that
display them if your index page needs to care. As
generated, Top's bottom-offset setting prevents it from encroaching on
lower fixed toolbars, but not end-of-page text; add space as needed to prevent
overlays at the bottom, especially on small mobile
displays.
Constraint: the new Top button requires JavaScript to be enabled in the browser. If JavaScript is disabled, index pages still work normally and as before 2.0, but do not display a Top even if one is generated. No warning is issued in this case, because the index page still functions in full with scrolls (where they are required at all).
showFullscreenButton
in file
user_configs.py.
Where supported, Full does expand and collapse the viewer page on taps, and its fullscreen may be collapsed by other means on some platforms (e.g., an Escape keypress). When on, fullscreen fills the whole screen, dropping system status and navigation bars, along with all browser bars (even in some contexts where manual options don't). This yields an enlarged image (plus an annoying startup message in some browsers). Crucially, though, the blowup lasts just for one image page: any navigation to a new page, including a Next and an Auto slideshow transition, cancels fullscreen jarringly.
There's no way to work around this, because fullscreen by spec must be requested only from a short-running, user-initiated event handler. It's usable only in response to user gestures, not page loads (read more about its API here). This is presumably to avoid phishing spoofs, but limits options for browser-based GUIs, and a complete redesign of thumbspage galleries to avoid page changes on image navigation is a nonstarter. (In short, a persistent fullscreen would require changing a single page in place, rather than opening new pages; depending on the design, it might also sacrifice per-image viewer-page URLs.)
Given Full's limitations, most users may be better served by browsers that have manual fullscreen options which persist across page changes and span Auto slideshows. See the earlier viewer note for manual tips, and the viewer-template file's code for implementation details. Manual fullscreen doesn't work everywhere, but it generally beats thumbspage's Full where it does. That said, Full may still be useful as a quick zoom, especially on browsers without a manual fullscreen option. Enable or disable as you like; it's your gallery.
Update: version 2.1 later enhanced the Full button to, like Auto, change its text to underlined font whenever fullscreen has been activated by the page's Full. This is for parity with Auto, but isn't quite as useful, because Full lasts just one page, and it's obvious that the mode is active. See 2.1's release note for more details.
Constraint: the new fullscreen display requires JavaScript. If JavaScript is disabled in the browser, the Full button will not be displayed, even if it is enabled in the configs file (its presence is decided by generated page code), and users will see the usual warning, which also applies to Auto buttons, filename-tap info popups, and dynamic image scaling on viewer pages. In fact, JavaScript is nearly required for thumbspage galleries today, and this might someday warrant a redesign that changes a single persistent page's content to support a true fullscreen mode (at the possible expense of per-image viewer-page URLs). This remains TBD, but users today can manually invoke a cross-page fullscreen on many browsers and platforms.
overflow-wrap: break-word
on the <h1>
title;
the result for pathologically long cases looks like
this
overflow-x: auto
(among others) on the toolbar's elements;
here's the result for long names and large fonts
before and
after scrolling
Shown in the last screenshots above, the filename at the top of viewer pages
was already set up to scroll on overflow too, and required no changes.
Also note that the first item above
applies to generated default headers only; index pages using custom
HEADER.html
files are
responsible for wrapping any title content manually if needed,
with similar techniques.
Though important, the two new stylings are mostly just for unusual border cases having either very large fonts, or very long filenames with no breakable characters, and generally apply only to very small windows or devices. On mobiles, for example, too-long filenames that exceed viewports make the entire page scroll horizontally, and too-big toolbar content can run together.
In typical usage, long filenames in HTML are automatically wrapped as needed
at embedded characters like -
, and the viewer-page's toolbar
content is too short to require a scroll. In unusual cases, though, the new
stylings ensure that long unbreakable content won't break viewports, and toolbars
won't lose their spacing.
For additional implementation details, see the code in the index-page generator and viewer-page template.
Because this was no worse that thumbspage's work-around, Version 2.0 initially changed the related config-file setting's default to skip the 1.6 work-around. Happily, as of Chrome 83 in June 2020, the history bug appears to have been fixed in full: thumbspage navigation pages are no longer stacked or retraced, and the initial 2.0 change to skip the temporary work-around suffices to adopt the fixed behavior. Hence, thumbspage history now works the same on iOS Chrome as on all other browsers; this issue is closed; and no further attention is required (unless and until it breaks again).
<head>
section of both default-header index pages, and all image-viewer pages,
as a target for search-and-replace:
<!-- Plus analytics code, custom styles, etc. (replace me) -->
This might be used by a site-publishing script to insert analytics code
or custom fonts, and may serve as a lighter-weight alternative to custom
HEADER.html
files for index pages,
and template-file edits for viewer pages. Automatic search-and-replace may
also require programming skills, but can be a simple customization
option for sites published with scripts.
Update: in version 2.1, this feature was finally leveraged by this script in the new build folder, to automatically insert analytics code prior to uploads. In hindsight, this may have been a bit late to the party; another manual replacement key also snuck into gallery indexes' custom headers over time.
alert()
dialog
with a custom modal
dialog—which
is really just a full-page
overlay with opacity, plus a display box within it, all shown and hidden on
demand. Most of the dialog's implementation code (CSS, HTML, and JavaScript)
is at end of the viewer-page template file;
the former callback handler for filename taps still formats the message using
Python replacements and JavaScript DOM data.
This naturally invalidates some former info screenshots, but 2.0's captures are all current. See especially its info-popup subfolder gallery for captures of the new dialog in action.
So why go to all the trouble of a custom dialog? For one thing, some browser
vendors discourage alert()
usage,
and a few browsers (e.g., Firefox) even treat it as a quasi threat, asking users if they
wish to
silence
it (odd, that, for a tool that's been a standard since the 90s).
Much worse, mobile browsers format alert()
text badly with wrapping,
and iOS 13's Hide Toolbars mode can even botch it altogether, displaying text outside
the dialog box.
All told, the new custom dialog offers a number of advantages, some of which work around browser-specific limitations. It:
The new dialog also scrolls its page vertically if needed for mobile landscape, and its use of viewer-page color settings makes for a more consistent appearance. Its only potential downside is that it won't pause Auto slideshows, but this may also be a bonus to some observers.
While the new dialog comes with added complexity, its UI improvements are well worth the cost—and better accommodate some of the following notes' content-expanding changes.
Update: as of version 2.3's May 2022 release, the background, foreground (text), and border colors of the info popup can now be configured to differ from the viewer page at large, and the opacity of the popup's overlay can be customized too. See the overviews here and here; the new color and opacity settings also apply to 2.3's Note popup, which borrows both code and design from info.
Depending on image content, the new info line may be present or not, and is formatted conditionally as follows:
Model
tag of an image's
metadata (part of the tag set collectively called Exif in this document; see the
standard); it's normally
present in shot photos and scans, but not in screenshots or drawn images.
Make
tag is appended if
it is present, just one-word long, and not redundant with a word already in the device
string; else it's judged extraneous and omitted.
Software
tag's value is reported instead
when available; it's recorded in some drawn images, but this varies by program.
In all cases, the device-or-software string is truncated if it exceeds a fixed length,
and characters special in JavaScript strings are replaced with ?
(until 2.3's more general escapes).
Here are examples of the new popup field for images with
device,
device and
brand,
no device but
software,
and neither type of maker
tag.
For additional examples, see the
info
gallery,
and images in this
live demo.
Along with the other curated info-popup fields already displayed, the new maker line seems primarily of interest to content creators. While additional Exif-tag display is nearly open ended (e.g., GPS data could spawn maps), thumbspage aims to strike a balance between image information and interface simplicity.
Here are examples of the three labels in action for photo, scan, and drawn images. 2.0 also considered omitting the date-of-origin line altogether when this date is unknown (similar to the new conditional Device/Software line of the prior note), but opted to retain it because the filesystem-based Modified line seems otherwise confusing: it might reflect origin or edit, but neither is implied.
Also note that some filesystems record a creation date explicitly (along with modification date), but it's generally unusable for thumbspage's cross-device info displays. Its support varies widely per filesystem (and not just platform), and even where present may not reflect actual creation date after copies and other changes. When in doubt, "unknown" is better than inaccurate. For more details, see Python's os.path.getctime() and the usual web searches.
w
and h
, respectively. For example,
the popup's new content for a scan looks like this, with dimension
lines bold here:
Digitized: 2013-04-14 @15:15:55 Modified: 2020-03-05 @15:46:57 File size: 482,451 bytes Image size: 2,944w x 2,088h Display size: 985w x 699h Device: CanoScan 8800F (Canon)
Trivial, perhaps, and English biased, certainly, but without the suffixes, it can be way too close to call in images that are nearly square.
window.name
JavaScript DOM variable to implement cross-page state. While this clearly
smacked of a hack, it was simple, could not be disabled by users, and was
supported on every one of the dozens of desktop and mobile browsers tested.
Unfortunately, this is now known to fail on Safari, only, when
viewing galleries offline on the local file system: the slideshow stops
after a single image flip, because state is not retained across pages.
This likely pertains to desktop Safari on macOS only, and was
seen on its version 13 in particular; mobile
Safari on iOS has no clear notion of local views
(and later evidence seems to exonerate mobile in full).
This issue wasn't detected in earlier testing, because slideshows work
correctly on Safari when viewing online galleries uploaded to remote servers.
In local-file mode, though, Safari apparently disables window.name
or reserves it for its own internal purposes—and
breaks any client code that wishes to make use of it in the process.
This may be yet another example of the
opinion-based rudeness which has grown sadly common in the software field
today, but that's a topic for another venue.
To make slideshows work on Safari too, thumbspage now uses HTML5 session
storage in all browsers to implement the cross-page state
required for the Auto toggle. As a fallback, it still uses the former
window.name
scheme when session storage is unavailable
or disabled; older IEs don't support it, and some browsers allow
users to manually disable it. Here are the highlights of the new
JavaScript template code; see its file's source
for the full, commented version
(and yes, this doc would rather show Python code, but vendors of web
browsers—and mobile devices—have rudely dictated otherwise):
var showDelayMS = %(SLIDESHOWDELAY)d; var toggleKey = 'thumbspageAuto'; var toggleOn = 'show-timer-on', toggleOff = 'show-timer-off'; function hasSessionStore() { try { sessionStorage.setItem('testkey', 'testval'); return (sessionStorage.getItem('testkey') == 'testval'); } catch (err) { return false; } } function autoClick() { if (hasSessionStore()) { if (sessionStorage.getItem(toggleKey) == toggleOn) { sessionStorage.setItem(toggleKey, toggleOff); clearTimeout(showTimerID); } else { sessionStorage.setItem(toggleKey, toggleOn); showTimerID = setTimeout(function() {navClick('%(NEXTPAGE)s');}, showDelayMS); } } else { if (window.name == toggleOn) { window.name = toggleOff; clearTimeout(showTimerID); } else if (window.name == toggleOff || window.name == '') { window.name = toggleOn; showTimerID = setTimeout(function() {navClick('%(NEXTPAGE)s');}, showDelayMS); } } } if (hasSessionStore()) { // on page load rescheduleAuto = (sessionStorage.getItem(toggleKey) == toggleOn); } else { rescheduleAuto = (window.name == toggleOn); } if (rescheduleAuto) { showTimerID = setTimeout(function() {navClick('%(NEXTPAGE)s');}, showDelayMS); }
With this change, Auto slideshows progress as expected on Safari and all others.
Session storage, available via both globals and window.
properties,
works the same as window.name
: it's unique per tab, applies to all pages
opened in a tab, and is removed on tab or browser closure. It differs between http
and https
accesses, but this is irrelevant for the relative URLs of Auto slideshows,
and sites can rewrite the former to the latter automatically. For more background
details, see the off-site coverage
here
and
here.
The lesson here may be that something which feels like a hack probably is one—and browser vendors may use it as such even if you don't. More fundamentally, this is a reminder that web development today requires supporting multiple, incompatible, constantly morphing, and opinion-driven implementations of what is supposed to be a reliable standard. That's not engineering, and it's about as far from developer friendly as it could be; fix, please!
In testing so far (using desktop Safari 12 and 13 on macOS Sierra and Catalina), these stops and crashes were seen to occur only for the following protocol-switching navigation sequence (spoiler: per the epilogue ahead, Safari Auto stops may also happen on domain switches in all-remote navigations, and this also seems limited to desktop Safari):
file://
)
https://
)
file://
)
Safari in this context wrongly clears the Auto toggle in HTML5 session storage to
JavaScript's falsy
null
, thereby turning the slideshow off.
Most of the time, simply tapping Auto suffices to restart the show.
Occasionally, though, and with no discernable cause, Safari botches the
gallery's viewer page completely in this event, displays the following
notorious
error in a popup
dialog,
and may need to be restarted in full:
"The operation couldn't be completed. Operation not permitted" (NSPOSIXErrorDomain:1)
Very oddly, none of this appears to happen when Safari's JavaScript console is open, which suggests a deep structural bug in the browser. Moreover, none of this occurs in any other tested browser—the Back stops and crashes were observed only on Safari. Unfortunately, it's unknown if this is limited to desktop Safari, because iOS's harsh access constraints discourage or preclude local-file galleries on mobile altogether (but watch for later clues in the epilogue).
In the possibly related department, it's worth noting that, both with and without the change to use session storage, Safari also produces an uncatchable exception on Auto image switch, which leaves the following suspicious error message in the JavaScript console:
Not allowed to load local resource: file:///favicon.ico
This happens only for local-file views just like the initial Auto-slideshow failures, but there appears to be no cause and effect relationship between the two: the exception also occurs on Next and Prev taps, and isn't remedied by disabling local-file restrictions in the Develop menu or pasting a valid favicon into the folder.
Just as strangely, a Back in Safari, only, does not run (or properly run) resize handlers in remote galleries: if the window is resized while away, the gallery doesn't catch up until the next page in the show or a forced reload. By contrast, a Safari Back does resize local-file galleries, but also kills their shows by clearing storage (and sometimes crashing). No other tested browser has these issues.
thumbspage tried multiple work-arounds to address the Safari failures—including
moving timer rescheduling from top-level code to pageshow
event handler;
forcibly cancelling timers on page unloads; and forcibly reloading the
page when reshown. Some of these were meant to address the theory (really,
wild guess) that failures stemmed from the browser's
back-forward cache.
For example, the first of these used top-level JavaScript code of this form:
function autoContinue(event) { // former top-level reschedule code of prior note } try { trace('run timer-reschedule code deferred'); window.addEventListener('pageshow', autoContinue); } catch (err) { trace('run timer-reschedule code immediately'); autoContinue(null); // run now in older browsers (or: window.onpageshow) }
None of these attempts helped as coded. Whether overzealous security constraints, cache optimization schemes gone bad, or outright bugs, something is clearly amiss in Safari local-file views that use session storage and timers.
As a more radical alternative,
HTML5 localStorage
, unlike sessionStorage
, might
not be erroneously cleared by Safari on Back, but this may not fix the crashes,
and is too broad and persistent to be used in any event: a site's slideshow toggle
would then be machine
global,
enduring to apply to later shows in ways users seem unlikely to expect—or
appreciate. Downgrading user experience this way for every browser
is not a valid fix for the foibles of one.
The upside here is that slideshows do work well in all other browsers, and work well in Safari most of the time too, after moving to session storage. They're currently known to fail only on switching from and to local galleries in specific Safaris; this is a rare—and even unlikely—use case. Nevertheless, given that this Safari glitch has all the hallmarks of a browser-vendor bug, and has stubbornly defied all work-around heroics to date, it will have to remain an out-of-scope caveat for thumbspage 2.0.
If this Safari quirk crops up in your thumbspage adventures, your best recourses are to tap Auto again to restart the show when possible; use an alternative browser that's less broken; or lobby Apple to fix its junk.
Epilogue:
after the above was written, further usage revealed that a Back in Safari sometimes
stops Auto slideshows for all-remote (i.e., online) navigation too, if users leave to another
domain (i.e., site) and return to the slideshow page.
That is, Safari's Back glitch may also be triggered by a domain switch. Just as
importantly, all-remote stops appear to happen only in the desktop Safari browser—not
in mobile Safari on iOS 13, and not in a dozen other browsers tested among Windows, macOS, Linux,
and Android.
In more detail, Safari, only, is now also known to stop an Auto slideshow running
in an https://
page, after navigating to another https://
page
at a different domain, and returning to the https://
slideshow page with
Back. This does not happen for all such remote switches, however. Curiously—and
perhaps entertainingly—a slideshow running at https://learning-python.com
rarely stops when switching to and from https://apple.com
, but reliably
does when the navigation target is https://google.com
.
HTML5 session storage should endure for all like-kind pages visited in a browser tab (or window).
It's specific to page protocol and domain by definition
(see
here,
here, and
here),
so it's expected to start fresh when navigating from a file://
page to https://
,
or to a new site in https://
.
But a prior page's session storage should also be restored and/or intact when a Back returns
in both of these contexts, and neither a protocol- nor domain-switching Back should ever
crash the browser. Only Safari bears these warts.
There's certainly more to this story, but forensics sans code is guesswork, and we'll
have to leave this bug tale here. Whatever the underlying cause, this is clearly
an issue for Apple to resolve;
until it does, the work-around for thumbspage users remains an Auto-press restart,
or an alternative-browser install.
Today's desktop Safari seems far too prone to toss its cookies
(or at least its session storage...).
DecompressionBombWarning
message now issued
senselessly by the underlying
Pillow library for all large
images.
Because the fix does not in any way alter the galleries which thumbspage builds,
none need be regenerated; this mutes one build-time message only.
Specifically, when building galleries with images larger than 89MP, the Pillow library by default prints a single DOS (denial of service) warning message in program output that looks like this (with line-breaks added here for marginal readability):
/Library/Frameworks/Python.framework/Versions/3.8/lib/python3.8/site-packages/PIL/Image.py:2797: DecompressionBombWarning: Image size (108000000 pixels) exceeds limit of 89478485 pixels, could be decompression bomb DOS attack. warnings.warn(
This baseless warning is completely harmless, and does not impact thumbspage results; large images work fine in thumbspage galleries whether this message appears or not. But it's also stupidly excessive, and needlessly confuses users of this and many other Pillow-based programs. The message's over-strong language may even scare some users over a threat that doesn't exist.
The warning was first seen for valid 108MP images shot on a Galaxy Note20 Ultra smartphone in 2020, and will crop up for all the increasingly common large images created by newer devices and tools. While the fix to silence the spurious warning may be trivial for program developers, program users have no recourse, and are left wondering if programs are buggy—or worse!
To see the fix's code for yourself, search for Sep-2020
in the
source.
Though simple, a "fix" is often far easier than a release.
At thumbspage's host site, multiple programs were impacted—including
thumbspage,
tagpix,
shrinkpix, and
PyPhoto—and each
required time-consuming retesting, repackaging, and redistribution.
This warning might be useful in some contexts, but it clearly should have
required an enable, not a disable.
Unfortunately, opt out is a regular by-product of opinion on overdrive.
Lesson: open-source agendas have consequences for others, and "batteries included" development entails substantial trade-offs. While this program could not easily exist without libraries like Pillow, it's also woefully dependent on them—even when they serve as platforms for personal preferences and subjective changes that stomp on the work of others.
So don't be rude out there.
Postscript:
though scantly documented, it turns out that Pillow later turned the
warning described here into a full error for images larger
than twice the warning's size limit. Though unlikely, this error takes
the form of an exception that will cause client programs to fail or terminate.
Despite this, its only mention seems to be in an obscure
release note;
whose details contradict an earlier obscure
release note;
and require studying Pillow's
source code
for full fidelity.
To avoid large-image Pillow errors, thumbspage's warning-silencing code has been
updated to use a new and broader fix
that sets Image.MAX_IMAGE_PIXELS
to None
—which will,
of course, suffice only until Pillow tightens the screws again. Large images are not
all attacks, and their creators are hardly ever criminals; unless it really
means to anger programmers and insult users, Pillow should really think
about making this check both opt in and better documented.
Summary: although thumbspage image rotations work well and as designed, recent findings suggest that it may be possible to implement thumbspage without rotations altogether, by propagating tags from sources to thumbnails. This hasn't yet been fully verified and no thumbspage changes have been implemented—mostly because programs that work don't generally need to be fixed—but this section provides background on non-rotation alternatives.
Details: to get started, we first need to understand the current rotation policy. Given a tilted source image coded with a non-"normal" Exif orientation tag (of the sort commonly had from smartphones held vertically), thumbspage today follows an evolved procedure to reorient images for in-page display that works like this:
.original
copy
The first three of these steps can be turned off on a per-build-run basis by a user configuration; if so disabled, thumbnails aren't rotated either, just because they're built from unrotated sources.
This scheme arose in phases over multiple versions and releases. thumbspage 1.5 displayed images too naively to count; 1.6 added auto-rotation of tilted sources, and their thumbnails by proxy; and 1.7 added Exif orientation-tag update and propagation for rotated source images (thumbs are simply built from already-straightened sources). The combination of these yields the procedure above, which is harmless for builders and viewers, and ensures that both sources and thumbs display right-side up in all contexts, regardless of their original orientation, and despite the vagaries of web-browser support.
Like much in computing, though, there may be alternative ways to do this. Per later research, it now appears that thumbspage may be able to skip rotations for both source images and their thumbnails altogether, as long as sources' orientation tags are propagated to their thumbnails. A non-rotation option is too involved to cover in full here, and is outside the scope of version 2.0. But the facts behind this alternative merit a few words.
In short, the JavaScript DOM seems to account for image orientation automatically in both its image properties and display. Specifically, the standard's doc here says this on the subject:
The IDL attributes naturalWidth and naturalHeight must return the density-corrected intrinsic width and height of the image, in CSS pixels, if the image has intrinsic dimensions and is available, or else 0. ... Since the intrinsic dimensions of an image take into account any orientation specified in its metadata, naturalWidth and naturalHeight reflect the dimensions after applying any rotation needed to correctly orient the image, regardless of the value of the 'image-orientation' property.
In other words, the DOM's "natural" sizes in JavaScript are already corrected for right-side-up display. The doc doesn't explicitly state that images themselves will be displayed this way when rendered as in-page elements, but this seems the case for a handful of major browsers in recent tests so far.
Because thumbspage has used the DOM's natural width and height in its dynamic image scaling since 1.6, it sizes the image for display in a viewer page with orientation already "baked in." By contrast, version 1.5 used the Pillow library's build-time width and height, which are not corrected for orientation the way that the DOM properties are; the result is an unscaled image auto-rotated by the same browsers, but stretched into pre-rotation dimensions.
Given the DOM's definition—and assuming every browser of interest implements an arguably implicit reorientation for in-page display—thumbspage's own reorientation of the source image is not required. Reorientation of thumbnails may similarly be unnecessary, but only as long as the source's orientation tag is propagated unchanged to the thumb to force the browser's adjustment. In sum, the alternative non-rotating procedure would:
This seems simpler, and may be, but it still must propagate updated tags using the piexif library (and fix tags known to make that library fail); requires the image-info popup to get dimensions from the JavaScript DOM instead of build-time Pillow (else they may appear swapped); and adds a browser-support dependency that the current scheme evades (and these are almost always best evaded). On the other hand, the source image would remain unchanged.
The required tag propagation of step 2 in the alternative scheme wasn't considered in 1.6 because it wasn't implemented until thumbspage 1.7. Prior to that, thumbnails made from unrotated sources were askew just because they had no orientation tag (and rotated source images were askew if their tags were copied but not updated). Rotating the source solved the issue in 1.6, but may now seem overkill.
All of which sounds simpler in hindsight, but thumbspage's scheme arose
in piecemeal fashion over years, and cannot easily be modified today.
While it's possible to disable thumbspage's auto-rotations process by setting
autoRotateImages
to False in
user_configs.py
, this leaves thumbs
tilted: because they're currently built from source images without copying the
source's orientation tag, they will display askew. Hence, either the source
must be rotated first (as done currently), or code must be changed to skip rotations
and propagate source tags to thumbs. And any change must also be predicated on
universal browser support for reorientations.
For now, thumbspage's reorientation scheme works as intended, has been used
successfully for years, and incurs no penalties apart from the minor overhead
of .original
backups.
While rotations might be rendered superfluous with redesign and recoding,
it's difficult to justify fixing a program with such a positive marketing story.
In its role as example for learners, however, thumbspage follows a
full-disclosure policy—even when that means pointing out its own rooms
for possible improvement.
Stop the presses: the non-rotating alternative proposed in the preceding section proved to be unusable after testing revealed that browser support for reorientation of images displayed as in-page elements is still coming online, and nowhere near the universal level required even on the latest platforms. Hence, in the name of inclusiveness, thumbspage will continue to rotate source and thumbnail images itself, because some browsers don't.
Details: although Exif orientation support has appeared in some browsers by fall 2020, a more complete analysis reveals that this is a very recent feature that has been appearing slowly. Along the way, its course included a CSS property that aimed to force the issue, but had weak support and was deprecated in favor of automatic adjustment. While automatic reorientation seems on track to becoming the norm in years ahead, it won't be present in versions of some browsers released just one or two years ago, and may never be adopted by others.
As a sample of where this support lies today, the following table summarizes auto-rotation findings for desktop and mobile browsers, as compiled empirically by thumbspage's testing department in October 2020; a "No" in the leftmost column means the browser doesn't adjust for orientation when displaying images as in-page elements:
Rotates | Device | OS | Browser | Version | Released |
---|---|---|---|---|---|
No | Desktop | macOS 10.15 | Firefox | 70 | Oct-2019 |
No | Desktop | macOS 10.15 | Safari | 13 | Sep-2019 |
Yes | Desktop | macOS 10.15 | Firefox | 78 | Jul-2020 |
Yes | Desktop | macOS 10.15 | Chrome | 85 | Aug-2020 |
No | Desktop | Windows 7 | Chrome | 77 | Sep-2019 |
No | Desktop | Windows 7 | Firefox | 56 | Sep-2017 |
No | Desktop | Windows 7 | IE | 9 | (unknown) |
No | Desktop | Windows 10 | IE | 11 | (unknown) |
Yes | Desktop | Windows 10 | Firefox | 78 | Jun-2020 |
Yes | Desktop | Windows 10 | Chrome | 85 | Aug-2020 |
Yes | Desktop | Windows 10 | Edge | 85 | 2020 |
No | Mobile | Android 7 | Firefox | 65 | Jan-2019 |
Yes | Mobile | Android 7 | Chrome | 81 | Apr-2020 |
No | Mobile | Android 10 | Samsung | 12 | Jun-2020 |
Yes | Mobile | Android 10 | Chrome | 85 | Aug-2020 |
Yes | Mobile | Android 10 | Firefox | 81 | Oct-2020 |
Yes | Mobile | iOS 13 | Safari | 13 | Sep-2019 |
Yes | Mobile | iOS 13 | Chrome | 84 | Jul-2019 |
For a more graphical look at these results, open the examples' thumbspage-screenshots gallery. As you can see, auto-rotation has come to some browsers in just the last year, and others still don't support it at all in fall 2020. On both desktop and mobile devices, current and recent versions of widely used web browsers still show orientation-coded images askew.
Which nicely captures a dilemma constantly facing engineers who work in a field as dynamic as software. Dropping its own reorientation might make sense if thumbspage could restrict its scope to just the latest-and-greatest browsers and versions. As a program designed to build galleries inclusive of all viewers, though, this is not an option. Even for browsers that have added the required support recently, versions just a few years old will still be common for years to come (despite the best efforts of their vendors).
Because we don't live in a world that restarts afresh each release cycle, thumbspage cannot be changed to rely on browsers to reorient tilted images any time soon. While this might trim some program complexity, it wouldn't work for many thumbspage-gallery viewers today, and requiring browser updates or replacements just to view thumbspage galleries would both be too extreme, and almost certainly qualify as rude (see the prior note).
For hard-core readers: the following Python session demos one way to do
the source-to-thumb tag propagation which was tested but ruled out;
Pillow also has some support for reading tags and processing them as
a bytes blob (see reorientImage
in
viewer_thumbs.py
), but
little to match
piexif
when it comes to updating tags:
>>> from PIL import Image >>> import piexif >>> it = Image.open('2020-09-17__180904.jpg') # unreoriented thumb >>> es = piexif.load('../2020-09-17__180904.jpg') # source tags (unreoriented) >>> et = piexif.load('2020-09-17__180904.jpg') # thumb tags (none yet) >>> es['0th'][piexif.ImageIFD.Orientation] 6 >>> et {'0th': {}, 'Exif': {}, 'GPS': {}, 'Interop': {}, '1st': {}, 'thumbnail': None} >>> >>> et['0th'][piexif.ImageIFD.Orientation] = es['0th'][piexif.ImageIFD.Orientation] >>> et['0th'][piexif.ImageIFD.Orientation] 6 >>> saveexifs = piexif.dump(et) >>> saveexifs[:20] b'Exif\x00\x00MM\x00*\x00\x00\x00\x08\x00\x01\x01\x12\x00\x03' >>> it.save('2020-09-17__180904.jpg', exif=saveexifs)
It's also worth noting that Pillow's ImageOps
module
has a call exif_transpose(image)
that straightens images
and simply deletes their Exif orientation tags, but this call
also discards all other Exif tags if its result is saved directly;
per warnings, seems to have even more issues parsing tags than
piexif;
and wouldn't make logistics any easier here in any event—regardless
of how rotation is coded, Exif tags must be propagated to sources for display,
and thumbs and sources must be saved to files in reoriented form because
both are displayed in thumbspage galleries as in-page elements that
aren't automatically righted by many browsers:
>>> from PIL import ImageOps >>> it = Image.open('../2020-09-17__180904.jpg') # unreoriented source >>> ir = ImageOps.exif_transpose(it) /Library/Frameworks/Python.framework/Versions/3.8/lib/python3.8/site-packages/PIL/TiffImagePlugin.py:588: UserWarning: Metadata Warning, tag 282 had too many entries: 2, expected 1 warnings.warn( /Library/Frameworks/Python.framework/Versions/3.8/lib/python3.8/site-packages/PIL/TiffImagePlugin.py:588: UserWarning: Metadata Warning, tag 283 had too many entries: 2, expected 1 warnings.warn( /Library/Frameworks/Python.framework/Versions/3.8/lib/python3.8/site-packages/PIL/TiffImagePlugin.py:588: UserWarning: Metadata Warning, tag 34853 had too many entries: 6, expected 1 warnings.warn( >>> ir.save('rotated.jpg') # simple saves drop Exif tags >>> piexif.load('rotated.jpg') {'0th': {}, 'Exif': {}, 'GPS': {}, 'Interop': {}, '1st': {}, 'thumbnail': None} >>> piexif.load('../2020-09-17__180904.jpg') {'0th': {256: 4000, 257: 3000, 271: b'samsung', 272: b'SM-N986U1'... # and a lot more
In the end—and with all the evidence in—thumbspage did the right rotations thing in 2018 and still does in 2020. Its universally portable image reorientation still beats the partial support among browsers today (even if readers 10 years down the pike may not think so).
Version 1.7 was initially released in February 2020, and repackaged as point releases until June of the same year. Here are the highlights of the 1.7 changes introduced by each release package:
Most 1.7 changes appear on generated image-viewer pages, and all can be found by
searching for [1.7]
in the system's code files as usual.
For examples of all the visual changes in 1.7, see its
screenshots page.
This section provides more detailed reviews of changes introduced across 1.7 releases. Because development of this version spanned half a year, its upgrades are many. For readers in a hurry, the later and longer reviews here come with summaries, and the following index provides fast access to 1.7 changes, from first to last, with major items colored black for emphasis (and time-strapped readers):
To fix, 1.7 now calculates that actual space used by non-image page elements, instead of using a constant. See the results here, here, and here. As a bonus, this allows viewer pages to use all available space for the image, yielding a larger image display in most contexts. There is no downside to this change, and thumbspage users are encouraged to regenerate their pages with 1.7.
1.25em
in CSS).
Though very rare, this could cause the buttons to
run together
badly on some browsers
if the user applied a very large font setting. This was observed on only one
of dozens of mobile and desktop browsers tested, and at a font setting large
enough to break the design of many a site on the web, but was still subpar.
To fix this for the sole known offending context, and make it less likely to occur in general, 1.7 abandons the prior font scale up for toolbar buttons altogether. See the result here. As a bonus, this allows more space for the image, yielding a larger image display on all mobile browsers tested. The only potential downside to this change is slightly smaller toolbar buttons on mobile devices, but the new button size is more symmetric with the filename's font (see the next note), and users can still scale up text as desired.
Note that both this and the preceding change invalidate some examples screenshots in trivial ways: toolbar buttons are now slightly smaller on mobiles, and the image display is slightly larger in general (e.g., see the befores and afters starting here). Older screenshots will be retaken if and as time allows, but are unlikely to be updated soon; to sample thumbspage's current behavior, be sure to try the examples in the thumbspage package, and the live demos at thumbspage's hosting website.
Update:
version 2.0 later resolved viewer-page toolbar button run-together in full
by a new CSS min-width
setting, which both enabled scrolls for
very large fonts in small displays, and reserved space between buttons
by limiting the crunch. See the 2.0 note as well as
the source-code of the viewer
template file for more details.
This new dialog gives the image's date/time taken (if any), last modified date/time, file size, original dimensions, and display dimensions. Taken details reflect Exif tags embedded in the image (see the standard); original and display dimensions are via the Pillow library and JavaScript DOM, respectively; and modified and file-size data reflect the image file itself. On devices that support hovering, the cursor also changes to a pointer over the filename to make the new action more obvious.
Note that, apart from display dimensions, the info displayed by the new popup is static: it reflects the image at the time that thumbspage generated the viewer page, not the time that the image is viewed. For accurate info, be sure to rerun thumbspage if images are modified—just as you would if images are added or deleted. Also note that the new dialog is simplistic and filename taps are somewhat subtle, but both pass as usable without overly convoluting the display or its code.
Update:
1.7 later specialized the creation-date label to "Digitized" for scans
(see ahead).
More strikingly, version 2.0 replaced the info popup's former alert()
call
with a
custom dialog
that's a complete solution to alert()
's
limitations described in the 2.0 release note.
2.0 also extended the info popup's content to include a maker line
that gives device (e.g., camera or scanner) or else software (e.g., drawing program),
if either is present, and changed "Taken" to "Created" when date is unknown
to accommodate drawn images.
Update: Later still, versions 2.2 and 2.3 further enhanced info popups with touch swipes, customizable colors and opacity, and more; see later releases for more info.
Update: version 2.0 replaced the former Raw button with an Auto for automatic slideshows (and later added a Full for fullscreen). Raw image display is still available via image taps—which made Raw fully redundant anyhow.
<!-- Generated
. They may be
useful in auditing or revision-tracking roles.
Tip: to strip these comments out of files to make results more comparable,
see this included
utility script.
y
(yes), instead of its former n
(no),
because viewer pages are now mature enough to broadly recommend over raw browser displays.
The default for the clean-thumbs prompt is now also y
, because this is the
normal and recommended response. These are minor usage enhancements,
but invalidate some console logs created under prior versions (two prompts differ
trivially); these examples
are being updated for the 1.7 release, but please pardon any remaining dust.
When provided on the command line, the images-folder
name isn't requested interactively, but other parameters still are; if you wish
to avoid inputs altogether, use the usual <
shell syntax to pipe in a
file of precoded replies, one per line, from a text file. To better reflect the new
argument, 1.7 also changes the order of inputs, moving file name to be asked first.
Like the prior note's change, this invalidates some older session logs (where folder
name is asked later); also like the prior note, the difference is negligible,
and examples
are being updated in the 1.7 package as they are found.
Update:
versions 2.2 and 2.3 generalized thumbspage
command lines further, with both config-file and console-input overrides coded as
setting=value
command-line arguments, though the
1.7 folder argument still works as noted here in its unadorned form.
See later releases for details.
.*
hidden names in subfolders bullet lists
.
and thus
hidden per Unix convention no longer appear in subfolder bullet lists on index
pages. Formerly, folders with leading _
s were skipped as gallery
private, but hidden folders are now ignored too; use either to omit from bullet lists.
__main__
test). This change does not impact the
content or behavior of the pages that thumbspage generates, which work on
any browser as before; but it allows thumbspage itself to be run to create
galleries in contexts that support Pillow but not tkinter.
While generally helpful, this upgrade's main impetus was Python Android apps. For instance, thumbspage is now usable on Android devices in both the tkinter-less Termux, as well as the tkinter-aware Pydroid 3, after installing Pillow. Especially when using a keyboard or tablet, this makes running thumbspage on Android a reasonable goal.
For more details on running thumbspage to build galleries on Android devices, see the notebox earlier in this doc.
Summary: thumbspage 1.7 discards its former hover effects on toolbar-button mouseover, because they are supported unevenly and poorly on mobile browsers. URL popups and cursor changes still provide indicators where available (i.e., on desktops, and a renegade mobile or two).
Version 1.7 grudgingly abandons the font effects formerly applied to viewer pages' toolbar buttons on mouseover (a.k.a. cursor hovers). These effects work well as a visual indicator on devices with a mouse (e.g., desktop browsers), but are just plain buggy on devices with touch screens (e.g., mobile browsers).
For example, hover effects commonly get stuck on after a tap on Android mobile browsers, and aren't cleared until the next user event; this is especially poor if the tap does not switch to a next page—like the new filename tap in 1.7 above. Hover effects may also require a double tap to activate the main action on iOS devices (one tap for the hover, one to activate). In thumbspage, hovering was formerly disabled for smaller screens via CSS media queries, but its problems could still crop up in landscape mode on mobile devices. Hover effects may be superfluous for toolbar buttons which also invoke a URL popup; but the filename does not, and neither do some hoverable touch screens.
Unfortunately, the only full remedy for this today is to avoid hover effects altogether on pages that wish to support both desktop and mobile devices—including thumbspage viewer pages as of 1.7. This is an unfair penalty on desktop browsers, where hovers work correctly, and are a common and even expected behavior. But it's also typical of the quality issues that mire the mobile realm, and one of far too many interoperability issues that plague the web domain. Alas, both seem to have been more accumulated than designed to date. For more on hover problems, see the note on this MDN page, and the iOS results for this search.
As a consolation, thumbspage viewer pages still provide some visual cues when a pointing device is used, by changing the cursor to a pointer whenever it covers any widget with a clickable action (though not when the action won't fire because JavaScript is disabled). This works well on all desktop browsers tested, but has no effect on mobile browsers even when a stylus is present—except for Samsung's browser on a stylus-equipped Galaxy smartphone, a device which otherwise inhabits a strange browser purgatory between mouse and touch, and further muddles a hover story that sorely needs a happier ending.
Summary: in thumbspage 1.7, thumbnails of non-transparent GIF images have been much improved, thanks to both a fix initially installed to do the same for thumbnails of shrunken images, and the generalized utility of the underlying Pillow library.
Per the usage note above, the site hosting thumbspage has started globally downscaling image filesize with the shrinkpix program. This makes online images both faster to view, and friendlier to limited-quota users. As a consequence of the conversions, though, thumbnails generated by thumbspage from shrunken images have potential to lose visual quality; in worst cases, the effect is a nearly complete munge.
To fix this, version 1.7 now internally and temporarily converts some images to "RGBA" color mode when creating their thumbnails. This produces thumbnails that look the same and are at least as good as those made from original, unshrunk images by prior versions. As a bonus, using "RGBA" for some unshrunk GIFs renders their thumbnails much better than before; to show how, here are two pages with GIF-image thumbs before and after the change.
For more technical details on this change, see the [1.7]
notes in
viewer-thumbs.py.
This fix is not applied to JPEG images (which yield the same thumb quality
when shrunk), or GIFs with transparencies (whose transparent parts would render
black). An alternative "RGB" conversion was less constrained, but rendered
macOS screenshot shadows fully black, which matched shadows on
full-size viewer-page displays only coincidentally when the default black
background color was used.
These shadows may be best avoided as a rule, but this lesson comes too
late for the hundreds of captures on thumbspage's host website.
Update: version 2.1 more broadly improved thumbnails by sharpening images and boosting save quality. The net effect removes both resize blur and compression noise in earlier releases' results. See 2.1's release note.
Summary: as of 1.7, thumbspage propagates Exif metadata tags to auto-rotated images, and updates dimension tags for the new right-side-up orientation using a third-party library included in the package. It's important to preserve metadata in general, but this is also required for proper display and info popups.
Version 1.6 added auto-rotation of original images and their thumbnails in thumbspage galleries. With version 1.7, rotated and saved JPEG images now retain their original Exif tags (see the standard). This grew more important with the introduction of image-info dialogs in 1.7 described above. To see why this matters, click the filename on this page; this rotated image's date taken is now displayed in the info dialog, because this data comes from the Exif tags propagated from original to rotated images by 1.7.
One catch to this process: if Exif tags were propagated directly and unchanged, they would reflect the original image, not the rotated copy. Among the possible consequences, width/height dimension tags wouldn't reflect the new swapped dimensions of the rotation. This isn't crucial, because rotated images are meant for use in thumbspage galleries only, and prior values document originally recorded size (though they may confuse other tools).
The orientation tag, however, cannot be propagated to saved rotations unchanged; if it's not updated to the normal-rotation value, other viewers will incorrectly reorient already-reoriented images, resulting in flipped displays. This happens even in the underlying web browsers that display such images directly in thumbspage's own Raw mode (a.k.a. image tap). Perhaps worse, if the orientation tag isn't adjusted, thumbspage itself will reorient the image and its thumbnail on every new gallery-build run—with erroneous, if comical, results.
To fix this, thumbspage updates both orientation and dimension values in the Exif data while propagating it from original to rotation. This both saves date taken, and rights the Raw apes. To perform these updates, thumbspage now uses the third-party piexif library, and ships this library's code in its download package. The much broader Pillow image-processing library used everywhere else in thumbspage must already be separately installed, but has almost no support for Exif tags, apart from fetching and saving their raw bytes data. By contrast, the pure-Python piexif both parses and composes Exif data, and provides read/write access to individual tags.
All of which is substantial added complexity. Without Exif-tag propagation, though, date-taken would not be displayed for rotated images alone; without Exif-tag updates, rotated images would be flipped by browsers and reoriented on each thumbspage run. The former would probably qualify as a bug, and the latter would be worse.
Also note that Exif propagation is currently applied only to JPEG images. PNGs have some notion of Exifs too, but it's just recently been standardized, was largely ad-hoc in the past, and the priority here is on commonly available photo data that thumbspage galleries and their info popups display. If present at all, PNG tags seem unlikely to be as widely useful as the information slavishly stamped onto JPEG photos by our cameras and smartphones.
Update: see also the June 2020 work-around ahead, applied after piexif was seen to fail on some uncommon Exif tags; updating tags makes thumbspage only as robust as a third-party library.
Update: version 2.0 explored the idea of dropping rotations in thumbspage itself, by propagating sources' Exif orientation tags to their thumbnails and relying on browsers to rotate. This was abandoned because browsers don't support rotations universally; thumbspage must rotate because some browsers don't. See the 2.0 note here.
Update: on a related note, version 2.1 and later now delete embedded thumbnail images, if present, so that their former Exif tags don't confuse some file explorers. See the update.
-webkit-text-size-adjust
style in default thumbnail-index pages, to disable iOS Safari's upscaling
(a.k.a. boosting) of some text sizes in landscape orientation. Upscaling is
arguably an iOS Safari misfeature: the zoom prevents users from seeing more
text in the wider format after rotating to landscape. The style—arguably
a hack, but common on the web—is new in index-page default headers only;
it was already used in generated image-viewer pages,
and added manually by most galleries with custom index headers. It's generally
recommended, but you can restore prior behavior by either changing the new
noiOSIndexTextBoost
setting in the configs
file, or using a custom HEADER.html
file sans the new hack.
Summary: in popup info dialogs triggered on filename taps, thumbspage 1.7 now labels capture dates for scanned images as "Digitized" instead of "Taken," as in this. This avoids potential confusion, but relies on device manufacturers (and programming libraries) to use Exif tags consistently.
The Exif standard's use of
multiple date/time fields leaves roles somewhat vague, and relies on consistent
interpretation by device manufacturers.
In particular, the two main date/time tags—DateTimeOriginal
and DateTimeDigitized
—seem prone to interoperability skew.
In practice, though, images captured on digital cameras normally have
both their original and digitized tags set to capture
(i.e., taken) date/time. While comparatively rare, scanned images
instead generally have a blank original tag, and record scan
date/time in the digitized tag.
Exif defines an additional DateTime
tag, but it's meant
only for recording time changed, and is considered out of scope by thumbspage
(and likely redundant with the file's modification time already shown).
In prior 1.7 releases, thumbspage's info popups used the original tag if present, else the digitized tag, and labeled the result "Taken" for either. This logic was borrowed from tagpix, which must select a filename prefix for uniqueness. It makes sense in thumbspage too if "Taken" is generalized to mean any digital origin, but might be mildly confusing for photo scans: the scan date/time doesn't reflect that of the actual scene, and most people are unlikely to manually add an Exif capture-date/time tag to older scanned images (when known at all). Here's the prior release's potentially puzzling display for a scan.
To do better, thumbspage's info screens now display either a "Taken" date/time, for images with an original-tag value; a "Digitized" date/time, for images with only a digitized-tag value; or "Taken: (unknown)" as before if neither tag is set (this is the normal case in thumbspage for PNG screenshots). Here's the new, improved results for a camera photo, a scanned photo, and a screenshot. This should better reflect photo scans in galleries—especially for photos with years in their filenames, like some at this site. This also accommodates screen-size limits in the info popup: displaying all date-related tags might run off screen on small mobiles, and date of origin is usually the main item of interest.
Beyond all this, it's also worth noting that Exif date/times generally have no time-zone information (a new standard does, but it's not widely followed); they reflect the recording device's time, which is presumably that of the capture location. Moreover, metadata can appear in multiple redundant-but-differing formats in digital images, and thumbspage is wholly dependent on the Pillow library to collect it, and devices to supply it. Although results have been good to date, digital image metadata, like much in the computing world, seems to have been as much accumulated as designed. For more on Exif issues, try this.
Summary:
as of 1.7, thumbspage now also uses JavaScript scaling for all iOS browsers in landscape
mode—including Safari.
This works well in all four non-Safari iOS browsers
tested,
and matches their Android
behavior.
It's also arguably better in Safari on iOS < 13: users must still scroll
once,
but it's easier to view the bottom of an
image
than
before.
Best of all, the new scheme works
ideally
as of iOS 13 in Safari, if the hide-toolbars option is enabled in its new
"aA"
menu.
As a legacy option, the former 1.5 CSS scaling can still be used
for iOS Safari's landscape
views
where preferred; see the configs file's
iOSSafariLandscapeCSS
.
Per this guide's pre-1.7 coverage here and here, thumbspage 1.6 reverted to 1.5's CSS-based scrolled scaling for landscape (i.e., vertical) orientation display on iOS devices. This was an intentional compromise, because page sizes were returned inconsistently in this context; no good work-around could be found for usability issues; and iOS's low traffic share at this technically focused website didn't merit additional efforts at the time. iOS's overall share was 3%, and the most popular alternative to Safari on iOS weighed in at just 0.4% (and lower if analytics blockers were factored in; details here). When time is limited, it's tough to justify investing it on outliers, especially when the only hope seems brittle kludges.
In retrospect, this policy may have been too broad. For one thing, not all sites' iOS percentages will be as low as those here. For another, most iOS issues were present only in Safari: only iOS Safari reported viewport sizes poorly, and only iOS Safari cropped landscape pages with mandatory toolbars. While Safari likely accounts for most iOS traffic on the web, 1.6's blanket policy unfairly penalized other iOS browsers whose landscape displays work well. The real clincher here, though, was a new option in iOS 13 which works around Safari's issues in full, and easily justifies a new approach.
To do better, thumbspage 1.7 by default now uses 1.6's JavaScript scaling for landscape views in all browsers on iOS, instead of the falling back on the former CSS-based scaling in this context. This is a clear win for all non-Safari iOS browsers tested; here are the results on a 4" test-device screen for Chrome, Firefox, the latest Edge, and UC Browser. These displays now better match those on Android, which has always scaled well in both orientations.
For iOS Safari, results are improved across the board, but vary by iOS release and user action. The best news is that thumbspage landscape views now work in full for users running iOS 13 and later. As of that version, iOS Safari has gained an option that hides its bottom toolbar altogether, and shrinks its top toolbar substantially. This option lives in the new "aA" menu at the upper left of Safari's window, and is applied to all pages opened in an enabling tab. For more usage details we'll skip here, try a web search; iOS 13 may also hide toolbars automatically when rotated to landscape, but the menu setting makes the effect permanent, and provides additional UI fixes. Specifically, when the new iOS 13 option is enabled:
All that being said, the thing about options is that they're optional. Although the new iOS 13 work-around can indeed solve multiple thumbspage issues in full, it's important to keep in mind that it won't help users who don't know about the new option; choose not to apply the new option; or use devices running iOS 12 and earlier that don't have the new option (yes, it's still possible). That's far short of an all-inclusive fix. Where the iOS 13 option isn't available or enabled, the new JavaScript scaling results in 1.7 are still obscured by toolbars and require a scroll, but they at least allow users to view image top and bottom more easily (the former CSS display revealed image bottom only during a down-scroll).
Browser flux aside, thumbspage's new scaling rules are very simple: it now uses JavaScript
image scaling everywhere, with just two exceptions. First, CSS-based scaling
is still used as a fallback when JavaScript is
disabled.
Second, because this change
differs from prior releases, and because some users may very well prefer the former
results in Safaris not using the iOS 13 work-around, you can reinstate CSS-based landscape
displays
by setting the iOSSafariLandscapeCSS
variable in the
configs file. This legacy option works for iOS Safari
only, because all other browsers handle JavaScript scaling without issue.
To summarize, thumbspage's complete image-page user story on mobile is now as follows:
At least that's the story until the next browser change. If web browsers are a kind of GUI, then web development is like having to make your code run on dozens of incompatible versions of a GUI toolkit, and on dozens of disparate platforms, all of which morph regularly and frequently and are naturally inclined to view interoperability as an obstacle to control. This has wrought a programming and maintenance nightmare—and ironically so, given the web's original portability ambitions. Please enjoy thumbspage today, but bear in mind that longevity cannot be counted among the assets of web-based systems.
Footnote: to be fair, iOS Safari isn't alone in harboring usability "features" that break websites. Edge for years had a URL-mouseover defect which rendered toolbar buttons unusable at the bottom left of a desktop page; and Chrome's duet adventures on Android regularly threaten to steal a sizable chunk of the display for browser agendas. Edge's overlay was fixed, and Chrome's duet is currently not a default, but both are typical of the sorts of vendor quirks that developers must battle constantly. A decent conspiracy theorist might point out here that vendors have vested interests in driving content to fullscreen web apps which generate store revenue. More likely, these are just artifacts of the opinion-based morph that plagues software at large, and renders web development in particular as much hacking as engineering. Improve me...
Summary: The piexif third-party library raises build-time exceptions for some uncommon Exif tag types with values coded unexpectedly by some cameras. thumbspage 1.7 now tries to correct specific known instances, and skips auto-rotation for others with a message instead of terminating the run; manually rotate the offenders and rerun if desired. This patch doesn't change the code of any generated pages, so no existing galleries need be rebuilt for its point release.
As described earlier, thumbspage 1.7 adopts and bundles the piexif third-party library, for updating Exif orientation and dimension tags in auto-rotated images. The Pillow library used by thumbspage for all other image processing can extract and set Exif data as a single blob of bytes, but cannot readily change individual tags; piexif adds parsing to and generation from an easily changed tags dictionary.
Unfortunately, piexif also appears to have a bug which causes it to raise exceptions at gallery-build time for some less common Exif tags, when trying to convert its changeable dictionary back into a storable blob. This happens even for tags parsed by piexif itself and unchanged by thumbspage. Whether or not the offending data uses type coding that deviates from the Exif standard, this seems a clear design flaw: piexif should be more forgiving, and should not abort when reformatting data which it created.
For a general review of the issue, see the formal report of this piexif bug on GitHub. In thumbspage specifically, the following cropped up for photos shot on a Samsung Galaxy Note 9 smartphone which were part of a folder being built into a gallery:
Making thumbnail: test/_thumbspage/2018-11-15__172646.jpg --Reorienting tilted image --Source rotate failed: skipped Exception: "dump" got wrong type of exif value. 41729 in Exif IFD. Got as <class 'int'>.
This photo was shot in 2018 and is hardly ancient in 2020, but Galaxy cameras have historically had issues conforming to the Exif tag standard (a different tags-related issue arose in the tagpix program for photos from the same device category). Still, piexif should do better than failing with the message above. In this case, tag 41729 maps to SceneType—an uncommon Exif tag that piexif correctly defines as a byte but happily parses as an integer, which in turn triggers exceptions in its unparsing method.
Also unfortunately, although thumbspage did catch and recover from the piexif exception, it failed later when querying image size, because the offending image file was not present—its original was moved to its backup copy, but never rewritten in rotated form. This terminated the thumbspage run in full, and left the gallery unbuilt:
Generating view page for: 2018-11-15__172303.jpg Traceback (most recent call last): ...details cut... FileNotFoundError: [Errno 2] No such file or directory: 'test/2018-11-15__172646.jpg'
To work around this, thumbspage now both fixes known offending tags, and restores originals on failing rotations so builds can proceed. For example, thumbspage special-cases tag 41729 to accommodate piexif's expectations, with code like this:
origexif = imgstart.info.get('exif', b'') # original raw bytes from Pillow parsedexif = piexif.load(origexif) # parse to dict for changes here ... parsedExif = parsedexif['Exif'] # parsed tags: dict of dicts if 41729 in parsedExif: tagval = parsedExif[41729] # miscoded on some Galaxy if type(tagval) is int: # munge from int to byte if 0 <= tagval <= 255: parsedExif[41729] = bytes([tagval]) print('--Note: bad SceneType Exif tag type was corrected') else: del parsedExif[41729] print('--Note: bad SceneType Exif tag type was dropped') ... saveexif = piexif.dump(parsedexif) # back to raw bytes for Pillow
Per another piexif bug report, a similar fix is applied for tag 37121 to convert integer tuples to bytes, though this was never seen to crash thumbspage. When corrected, tags leave a note in the run's output:
Making thumbnail: test/_thumbspage/2018-11-15__172646.jpg --Reorienting tilted image --Note: bad SceneType Exif tag type was corrected
Additionally, on piexif failures, thumbspage now automatically restores the failing image's backed-up original, to avoid terminating altogether. The failing image won't be rotated, but the gallery build will run to completion successfully, and a message in the logs directs users to try rotating the offending image manually and rerunning; here's the scene if the 41729-fixer code is disabled:
Making thumbnail: test/_thumbspage/2018-11-15__172646.jpg --Reorienting tilted image --Source rotate failed: skipped Exception: "dump" got wrong type of exif value. 41729 in Exif IFD. Got as. --Rotate manually and rerun if desired ...and finish the build...
To see all the code changed by this update, search for Jun-2020
in
viewer_thumbs.py. Again, no generated
page code is modified by this patch; it's an improvement to
build-time mechanics only. thumbspage also considered rotating without
Exif updates on piexif failures (as before 1.7), but the consequences for
this are worse than requiring manual rotations for skips.
As a reminder, you can always disable image rotations altogether in the configs file,
if the failure messages grow too grievous; see the 1.6 note.
Like many, this work-around is a stopgap measure. The real solution is a fix for piexif which better handles oddball tags, and type mismatches in general. As is, this seems a general and broad issue in the library, which may not be addressable by one-off tag patches in clients. Until a better piexif becomes available, the best thumbspage can sometimes do is skip auto-rotations for photos that crash piexif's otherwise highly useful utilities. Alas, third-party dependencies make programs, well, dependent.
Version 1.6 was finalized on October 28, 2018. This version was released multiple times with new features in each release; release dates identify point releases. Its main extensions are dynamic image scaling and image auto-rotation, but it introduces numerous enhancements to the program—including this HTML guide.
This version was last repackaged as an interim release in February 2019 with a minor change to avoid a tkinter dependency for contexts that have Pillow but not tkinter (e.g., some Android Python apps). The tkinter module is now required only when running the simple GUI viewer. For more details, see the update note above for version 1.7, which formally adopted this change.
Despite the prior version's hand-waving, thumbspage's image-viewer pages now use JavaScript in their generated HTML to dynamically scale images to available display size, while preserving original aspect ratio and avoiding content screen overflow. The net effect emulates browser-native scaling in viewer pages, while allowing for extra control widgets. The former CSS scaling scheme is still used, but only for no-JavaScript views and iOS landscape mode. In more detail:
Exception: on iOS only, images are always fully scaled in portrait device orientation, but for implementation reasons fall back on version 1.5's CSS scrollable scaling result in landscape device mode (in short, display size returned by iOS in this mode is unusable). Android fully scales images always, which yields smaller but complete landscape images—which is arguably better, but open to feedback.
Update: thumbspage 1.7 now also uses JavaScript scaling for images in landscape mode on iOS, in all browsers. This includes iOS Safari, which still has landscape issues in earlier versions, but gained a new toolbar-hiding option in iOS 13 which can fully solve former thumbspage problems. Chrome, among others, now works the same in landscape on Android and iOS. Read more about this change here.
On both device types, a "Loading..." message is also posted as a visual status indicator. Version 1.5 deemed JavaScript a "nonstarter" partly because of the added complexity, but mostly because locking out users who don't wish to run JavaScript is a rude non-option. Here, though, pages still work when JavaScript is disabled—they use the former 1.5 CSS scaling, with a note recommending JavaScript results. Turn off JS in your browser to see how 1.5 CSS scaling compares, and see the code for more details.
thumbspage now automatically reorients (rotates) tilted source images and their thumbspage-generated thumbnails to be right-side up (top edge on top), if they have valid "Orientation" Exif tags (see the standard). This is really just an automatic alternative to manually rotating tilted images before making thumbs, but is especially useful for photos shot on smartphones that commonly tilt photos shot in the natural portrait (vertical) device orientation.
Less positively, this feature applies only to images with reliable Exif tags (e.g., JPEG and TIFF) from cameras and tools that tag as expected. More intrusively, this feature must rotate source-image files too (not just their thumbnails), because not all viewers will rotate images when opened from thumbnails. In thumbspage specifically, web browsers will not rotate tilted source images (except on "Raw" clicks), because its viewer pages display source images as in-page elements. The related PyPhoto program now also adjusts for orientation in memory only and does not require image copies, but uses forked thumbnail-generation code (see its website).
Because this feature modifies source images in-place, it by default saves
them to backup copies with a .original
extension before making
changes. Users may also control the feature with two new settings in
user_configs.py
:
autoRotateImages
can turn the feature off,
and backupRotatedImages
can skip its .original
backups. Disable rotation if needed or desired, and manually rotate
tilted images before running thumbspage as preferred.
See examples/reorientation
for a test case's results; that folder's
restore-prerotate-originals.py
utility to restore from backups; and module
viewer_thumbs.py
for implementation code
(rotation is neither an automatic nor an easy option for Pillow/PIL thumbnails).
Caveat: rotated source images drop the originals' Exif tags,
but rotated images are not the same as the original (some tags might not apply);
they are meant to be viewed in HTML galleries only; and some other image-processing
tools drop the tags too.
Update: version 1.7 now retains and updates Exif tags in JPEG images rotated and saved (read the details). This Exifs propagation grew more important with 1.7's new info dialogs which display date-taken tag data.
Update: though rare, it's possible for rotations to fail due to permission errors, miscoded Exif tags, and other reasons. Check the output for messages if a photo remains tilted in a gallery, and see this 1.7 note for one cause of rotation failures.
Update: as noted at version 1.7, 2.0 explored the idea of dropping rotations in thumbspage itself, by propagating sources' Exif orientation tags to their thumbnails and relying on browsers to rotate. This was abandoned because browsers don't support rotations universally, especially for in-page elements like those in thumbspage viewer pages; thumbspage must rotate because some browsers don't. See the 2.0 note and demo.
Update: as of version 2.1, embedded thumbnails in images automatically rotated for in-page display are automatically deleted, to avoid issues with tools like file explorers that may grow confused if the main image is rotated but its embedded thumbnail is not. See the 2.1 release note; deleting is substantially simpler than updating the nested thumb.
In addition to the main items above, version 1.6 also:
Adds the HTML user/developer guide you are reading, as a replacement for the former in-code and text-based documentation. Text is easier to code, but HTML can be much nicer to read.
Moves configuration options and viewer-page HTML to separate files for easier viewing and edits. See Customization above for links and story.
Uses _thumbspage
as the default name of its thumbnails +
viewer-pages subfolder, to avoid clashing with other content.
The former "thumbs" name default may still appear in some docs and screenshots.
Prior-Version users: your "thumbs" folder will be unused; manually
delete or rename, or set THUMBS='thumbs'
in
user_configs.py
.
Expands image-folder name .
to its true basename in generated
default-header text, and appends a /
to subfolder hyperlinks
so they will not trigger redirects or clash with files if and where
it matters. Because the .
fix uses Python's os.path.abspath()
to expand the dot, it also properly names image-folder paths with any "."
or ".." (e.g., ".", "..", "../..", "../Desktop/trnpix/../trnpix/.", and other oddities).
Adds
a new configuration setting, templateEncoding
, which
allows the Unicode encoding of the viewer page template file to be easily configured,
and differ from that of index-page header and footer files. Its preset default and
generally recommended setting is the broad UTF-8;
edit user_configs.py
to tailor.
Note that this new setting is used for loading the template file only;
generated viewer pages instead use the same Unicode setting for saves as
index pages—outputEncoding
. The two settings may be the same or
differ, depending on your usage. The Unicode encoding declared in viewer-page
<meta>
tags also uses outputEncoding
automatically so that it
agrees with page content; if you manually edit this tag, it should similarly agree.
Update:
as of version 2.0, templateEncoding
is now also used to load the
floating Top button's template-floatingtop.html
file, whose code
is added to index pages per outputEncoding
. See the 2.0
note.
Adds an expandSmallImages
option in
user_configs.py
.
If False, the maximum viewer-page scale ratio is 1.0, which
constrains images to an actual-size maximum, and thereby
avoids stretching and possibly blurring small images. If
True, smaller images are always expanded to display size.
The preset default is False, because this is generally better when smaller images are present (e.g., icons, and small-window screenshots). Use True for the prior expanding behavior which may be preferred in some contexts. Note that this setting applies only to small images; most digital photos and scans are far larger than display areas, and will only be scaled down.
Uses JavaScript viewer-page code to avoid adding viewer pages to destack browser history where supported, so that N image views don't require N browser Back clicks to get back to pre-gallery context; a single Back returns to the gallery entry page. This destacking works everywhere but Chrome on iOS, where the feature is disabled (i.e., viewer pages are stacked on history and retraced by Back clicks for this browser only). The JS template file hosts most of the implementation's code (view its source).
This feature works and is used in Chrome on Android+Windows+Mac;
in Safari+Firefox on iOS; and in all other 20+ browsers tested
(including Internet Explorer 9, and other Android browsers). The failure
in Chrome on iOS is clearly a bug in its location.replace()
of versions 64 and 68 tested; if used in this browser, Back clicks
redisplay the same page N times for N image views, which is
worse than stacking pages.
Because this might be fixed in the future,
user_configs.py
's new
chromeiOSBackFixed
controls the iOS Chrome disabling.
At present, though, Chrome on iOS is just 0.40% of the audience at the
site hosting this program (just 10.96% of iOS, which is itself just 3.67%
overall); it makes no sense to omit an enhancement for 99.6% of users,
for the sake of just 0.40%. For more of this site's analytics,
visit this page.
Update: history destacking is still broken in
iOS Chrome, as of its version 75 in May 2020. Its history-cropping
location.replace()
method runs without error, but silently
stacks all pages. This requires Back clicks to retrace all images
viewed—marginally better than displaying the same page N times,
but still nonfunctional. The disable is still on by default in the
configs file, though its effect is currently moot (pages are stacked either way).
Update:
because this iOS Chrome work-around is no different than the browser's
current buggy behavior, version 2.0 makes the work-around bypass switch's default
True
. This skips the work-around, and stacks navigation pages
in this browser's history today, but will also pick up a true Chrome fix if
one ever appears.
Update: as of June 2020, this iOS Chrome history bug appears to now be fixed in Chrome 83 (its actual fix version is unknown, but Chrome 75 was still broken). No changes are required, because the former update's config-file change sufficed to adopt the new, fixed behavior. For more details see the 2.0 note.
Works around a Pillow library bug that could occur only in limited usage contexts for folders having very many images. In brief, Pillow's auto-close of loaded image files does not work as documented, which can lead to "Too many open files" errors after many thumbnails have been generated. Here, this meant that results could reflect partial source content for very large folders, though only on macOS in general.
The best and applied fix is to manually open and close image files,
instead of passing filenames to Pillow. With this change,
arbitrarily large folders are supported in all contexts.
For more details on both the bug and its work-around, see
module viewer_thumbs.py
,
where the fix is coded.
Works
around a Chrome issue on Windows and Linux, by using
auto-scroll: hidden
CSS for the body, to forcibly
hide the vertical scrollbar. Else, Chrome (and possibly
older Firefox) flash a scrollbar momentarily during viewer-page
loads. This setting doesn't impact displays in any other
way, and scrollbars are never required on viewer pages (images
are scaled, not scrolled).
Caveat: Chrome on Android (only) may still sometimes very briefly flash a vertical scrollbar anyhow. This may be an indicator or other normal behavior, but seems more likely a browser bug with no known work-around—applying the hidden setting to "html" in CSS doesn't help. This is minor and cosmetic, but like much web experience, has to be chalked up to browser idiosyncrasy.
Works
around a Chrome desktop peculiarity, by using
thin
instead of 1px
for CSS <img>
border-width in both index
and viewer pages (for thumbs and images). On this browser only,
1px
can cause some image borders to not be
drawn at zoom levels < 100% due to fractional pixel math.
thin
is equivalent to 1px
in size today (not
2px
, as once rumored on the web), but does not suffer from
pixel-math cloaking, and doesn't impact displays otherwise.
This works through Chrome desktop zoom level 50%; below that
Chrome (again, only) may drop viewer-page bottom borders for some
window sizes, but that's a reasonable usage cutoff (pages requiring
a microscope are off-table). Using a 1.5px
almost works,
but can add empty space between border and content. To test borders, see
this site.
As part of this work-around, image border color was also made
configurable for both page types; set border color to background color
to omit borders altogether. Note that index-page table top/bottom borders work
unchanged as 1px
, and this 1.6 change is disjoint from 1.5's
<hr>
Chrome fix noted ahead.
Version 1.5 was finalized on August 12, 2018. This version's main feature is the introduction of image viewer pages, which were further improved in version 1.6 (see its notes). A set of extra enhancements rounds out the release.
In addition to its former thumbnail-index pages, thumbspage now generates a styled viewer page for each image in the folder, instead of relying on each browser's native image display.
Viewer pages are opened on index-page thumbnail clicks, and have filename, the image scaled per CSS, view-native and go-to-index links, and previous/next-image links that cycle through all images without having to return to the index page. Viewer pages center images horizontally, but not vertically; the latter is too jarring during next/previous navigation.
Viewer pages are generated in the thumbs folder, along with the prior version's thumbnail-index image files. They can also be suppressed via console input prompts; when omitted, images open in browsers directly and natively, as before. To view an example client live, visit this site.
Caveat: because image scaling is weak with CSS alone (and JavaScript seems both overkill and non-starter for this project), 1.5 image-viewer pages should be considered an optional feature. See "1.5 Usage Note" for deployment suggestions, and in-code documentation in the main script for more details.
Update: version 1.6 later replaced 1.5's CSS scaling with much better JavaScript dynamic image scaling for most use cases, and deleted the now-moot "1.5 Usage Note" referenced above. See the 1.6's release notes above.
In addition to the main items above, version 1.5 also:
Formalizes index-page and navigation ordering. It's now
case sensitive by default everywhere, but can be changed in
user_configs.py
if case-neutral (Windows-like) ordering is preferred;
see that file's setting caseSensOrder
.
See also orderedListing()
in
thumbspage.py
for more details.
Formalizes
URL-escapes' Unicode encoding, which determines the content
of %xx
-formatted bytes. It's now always UTF-8,
regardless of the encoding used for whole HTML pages, because this
seems to be required by both standards and browsers.
See also url_escape()
in
thumbspage.py
for more details.
Sets body font to Arial (sans serif) for default-header index
pages. This is cosmetically nicer, and matches the new viewer pages' precoded font.
This font is set for the body in a default header <style>
block only and not
inlined in generated page components, so that global font can differ in a custom
HEADER.html
file.
Works around a desktop Chrome <hr>
bug
which botches the separator line (it's much lighter at some zoom levels than
others). To fix, the thumbs table was restyled to use table top and bottom
borders instead of <hr>
s. As a consequence, the thumbs table now
always stretches to 100% window width, to extend the border
lines (this was formerly a configuration,
off by default). See also the 1.6 Chrome fix for vanishing
<img>
borders.
Sets thumbs-table background color to light grey as part of
the former note's <hr>
restyling, and allows it to be changed
in user_configs.py
.
The new viewer pages' colors (and others) can be similarly tailored
in that file.
Refactors its code to functions (it's now large enough that top-level code is difficult to navigate), and cleans up its page output (HTML/CSS is tough to read as it is).
Still uses all-inline styles for the thumbnails tables on index pages,
not <style>
blocks, so that custom
HEADER.html
files
need not include or link to styles for the generated table. Conversely,
viewer pages use a <style>
block, as they are not customizable
without HTML edits (yet?).
Version 1.4 was finalized on March 4, 2018. In this release, this script's output page better supports browsers on smaller screens (e.g., mobile phones and tablets), and looks nicer in general. Its new generated CSS code:
Autoscrolls the thumbs table to appease mobile browsers.
Adds padding to thumb cells to reduce run-together.
Center-aligns thumbs images for a more even look; this helps overall, but especially for narrow/portrait images.
Uses nowrap paragraphs for image labels; the former
<br>
+wrap scheme looked jumbled, and made Chrome (only!)
arrange thumbs-table columns unevenly in small windows
(the leftmost was narrower).
This version also adds a mobile-friendly viewport <meta> tag to
default headers if useViewPort
is True in
user_configs.py
(this is its
preset default). This may impact other page components; use a custom
HEADER.html
file for more
options and control where needed.
Tip: because filenames used on labels are now not wrapped, their width largely determines column spacing in the index page; use shorter filenames for less space between (i.e., narrower) columns.
Version 1.3 was finalized on August 8, 2016. This version makes several changes to better accommodate arbitrary non-ASCII Unicode filenames and page content (see the live demo). Specifically, this release:
HTML-escapes all added text—image, folder, subfolder names.
URL-escapes all added links—to thumbs, images, subfolders, and viewer pages in 1.5.
Outputs the generated index file in UTF-8 Unicode encoding by
default, with a content-type <meta>
tag. ASCII content is
unchanged, as it is a subset of UTF-8. Other encodings may be used for
the output file via setting outputEncoding
in
file user_configs.py
.
This setting is also used to save viewer pages, per the 1.5 update below.
Loads any header and footer inserts per UTF-8 Unicode encoding
by default, as it is general and supports ASCII directly.
Other encodings, including the platform's default, may be
used for inserts via setting insertEncoding
in
file user_configs.py
.
Note that this setting is used for insert-file loads only; generated
pages are always saved per outputEncoding
, which may or
may not differ in your usage.
Assumes any inserts are both HTML-safe and compatible with the default or configured inserts encoding. Be sure to verify your inserts before using this program's results.
See examples/unicode/images for the results of a comprehensive Unicode-content test case.
Note: if you use a custom
HEADER.html
file,
make sure that the Unicode type declared in its content-type <meta>
tag matches the setting for outputEncoding
(#3 above).
thumbspage uses the latter to save the whole index-page (including the content
of its thumbnails table), so these two encodings must be the same or compatible.
Update:
version 1.5 further refines URL escapes to use UTF-8 encoding for escape
byte values,
and loads its new viewer pages as UTF-8 by default
(configurable as of 1.6 by setting
templateEncoding
;
viewer pages are still output per outputEncoding
).
The Unicode test's folder also moved to the new path named above, and
thumbspage development switched from Windows ("\") to macOS ("/")
along the way, though some docs may still retain their former Windows bias.
Version 1.2 was finalized on August 1, 2016.
This version adds assorted cosmetic tweaks, mainly in the name of
better thumbnails-table styling, and subject to settings now in
file user_configs.py
.
Primarily, this release:
Uses uniform-width columns, instead of sizing columns
per content (on by default; see setting uniformColumns
).
Stretches the thumbs table to fill the whole window or display
(off by default; see setting spanFullWindow
).
Update: this is now always on to
expand the table borders added for the 1.5 <hr>
fix.
Adds a scrollbar if the window is too small? (prototyped but skipped in 1.2).
Update: this was obsoleted by version 1.4's auto-scrolls.
Version 1.1 was finalized on July 27, 2016.
This version adds an automatic
subfolder-links list
to the index page just before the thumbnails table, if enabled by
setting listSubfolders
in file
user_configs.py
.
This feature can be used to build nested galleries, or
folders with supplemental gallery information.
This feature is on by default. It creates a links
list with one entry for every subfolder nested in the images
folder. The links in turn open the subfolder, with a result that
varies per usage mode. For example, online views may automatically
open the subfolder's index.html
page or web-server
index, while offline views open a simple directory listing.
See your web server for online configurations, and click
an index.html
manually when offline.
As a convenience, automatic subfolder-link lists skip (do not display)
folders whose names are prefixed with a _
or .
character, as well as the generated thumbnails folder (whether it's named
with a _
or .
prefix or not). You can use
folder names that start with a _
, for instance, for related
content that you do not wish to publish in your gallery.
Note: if enabled, the subfolder-links list is generated whether you
use a custom HEADER.html
file or
not. When a custom header is used, the list's code appears after
the header's text. Hence, pages that use a custom header may wish to either
accommodate the list in the header's opening narrative, or disable this feature
and code the list manually as desired. Naturally, this may vary per gallery,
and matters only if there are subfolders in your images folder.
Tip: thumbspage must be run on each image subfolder in a tree separately,
and subfolder index.html
files must be clicked manually offline as
noted above.
The tree is not walked automatically because its folders may have arbitrary content,
and subfolder links do not automatically open index.html
files
because some subfolders may not have them (e.g., a docs-only
subfolder).
If you really want subfolders' index files to open automatically when
offline, you might consider running a web server on the local
machine.
Update: version 2.1 now allows the space between
subfolder links to be customized with subfolderSpacer
in
configs, and increases
this setting's default from 6 CSS pixels to 7 to make taps easier on mobile devices.
See 2.1's release note.
Version 1.0 was finalized on July 24, 2016. It provides a basic thumbnails-index page, with generated thumbnail images that open browser-native views. Its functionality is limited, but its results already beat default folder pages.
As normal, later thumbspage releases were spurred by using this program over time. Where it goes next is limited only by developer availability and user experience. The next and final section sketches a few TBDs for future consideration—and argues against many of them.
Like all software, thumbspage is an open-ended project.
In closing, this section collects implementation-related issues
and proposals, some of which hail from early releases and have
been resolved over time.
They're likely of most interest to developers, though some usage
context might also be gleaned along the way.
A few of these items are also outstanding questions, open to feedback;
if you have preferences or suggestions, please pass them along with the
Input
link on this doc's bottom toolbar.
Also note that this list is not necessarily complete; search for "caveat" in code files for additional code-related items, and see Version History above for more on changes both noted and not noted here. This section hosts the following content:
Less positively, thumbspage also does not update an embedded thumbnail when it rotates a full image (it rotates only the thumbnail it generates itself). This, however, seems a trivial concern: thumbspage gallery images are largely intended for browser-based display, and other tools should continue to automatically accommodate any lingering thumbnail orientation. That said, it's not impossible that some file explorers might become confused, and display a still-flipped embedded thumbnail; improvements TBD.
Update: as of version 2.1, embedded thumbnails in images automatically rotated for in-page display are automatically deleted, to avoid issues with tools like file explorers that may grow confused if the main image is rotated but its embedded thumbnail is not. See the 2.1 release note.
Update: 1.7 now retains and updates Exif tags in JPEG images rotated and saved (read the details). This grew more important with 1.7's new info dialogs which display date-taken tag data.
Update: 1.7 shows Android landscape images slightly larger after a fix for large-font image clipping, and makes it more natural to view images full size by making an image tap the same as the Raw button (see the details here). This helps, but the Android landscape scaling question is still open.
Update: 1.7's final release adopted JavaScript scaling for images in landscape mode on iOS, in all browsers. This includes Safari, which still has issues in older versions, but gained a toolbar-hiding option in iOS 13 which can solve its problems, and justifies the new uniform approach. Chrome and others now work the same in landscape on Android and iOS. Read more about this change here. CSS scaling is still available for Safari on iOS, but only as a legacy option.
location.replace()
bug that required
disabling a Back feature on that browser, per version 1.6 notes (and their
May 2020 update) above. Watch for a browser
fix and change the configuration if and when it appears.
Update: Per the 1,6 note above, iOS Chrome fixed this bug as of version 83 (and perhaps earlier), and the thumbspage config file setting was changed in 2.0 to skip the temporary work-around; unless this resurfaces in the future, no other action is required.
user_configs.py
, and custom index-page fonts can be
had via CSS in a HEADER.html
(see the configurations file for pointers).
Still, user customization is open
ended—and pending usage feedback. For example, default-header
title and text could be configurable too, though they naturally
vary per folder and might have to be changed often.
Update: tables are fixed grids that don't support automatic wrapping on resizes, but it's possible to layout image links with cells that do wrap to display size dynamically by generating CSS and HTML code of this sort:
#thumbslinks>div {display: inline-block; vertical-align: top} <div id="thumbslinks"> <div>...image link and filename label...</div> <div>...image link and filename label...</div> </div>Unfortunately, this doesn't retain fixed-sized columns like tables do; cells are scattered randomly across the page. There may be ways to get this to work better (e.g., force all cell sizes to the longest filename label), but scrolled tables seem just as reasonable, and this remains TBD.
Update: release 2.1 implemented dynamically sized/resized index-link tables, using the layout model above, along with fixed cell sizes and CSS styling that tries to simulate columns with the maximum of thumbnail-image and filename-label widths. The new dynamic layout is available as an alternative to the former fixed-column tables, but not enabled by default, because it typically yields just one or two columns on mobile: this avoids some minor horizontal scrolling, but makes for much more vertical scrolling in non-trivial galleries. Fixed tables seem a better scrolling tradeoff for mobile, but your needs may naturally vary. PC-only galleries, for instance, might prefer the dynamic model; on desktop, the new model shows more or fewer thumbs per row on window resizes, but doesn't stretch or shrink the space between them. See 2.1's release note for details.
Update: 1.7 added an image-info
dialog, but kept the program simple by
using a JavaScript alert()
call, and avoided GUI clutter by invoking the
dialog with a tap on the existing filename field. This dialog's information
is mostly build-time static, and could be more grandiose, but it probably shouldn't be.
Update: 2.0 expanded info-popup content, and replaced
its alert()
call with a custom dialog (a page overlay with opacity
and scrollable content box). This is more complex, but is a much better solution,
given the limitations of alert()
described in the 2.0
release note.
Update: 2.0 implements a fullscreen mode toggled by a new Full button, but makes it optional because of its downsides. Besides being unsupported on some platforms, JavaScript-induced fullscreen lasts for just one page, which makes it much less useful than manual but persistent fullscreen options available in many browsers. Hence, Full is shown by default but can be easily disabled per gallery if it seems just a browser parlor trick; your gallery's users may, however, find it useful as a quick zoom. If used, Full doesn't crowd or break viewer toolbars because they now scroll (but normally don't have to, even on small devices). For more details, see the 2.0 release note.
Implementation notes:
thumbspage's fullscreen toggle uses the JavaScript requestFullscreen()
call in document.documentElement
, along with its cohorts. Because this API
is not universally
available,
prefixed variants are also tried on browsers that don't support the standard; even then,
some browsers ignore fullscreen requests altogether today. Hence, thumbspage's Full isn't
supported where it isn't supported. Such is life in the splintered world of web development.
Update: 2.0 added automatic slideshows and redesigned viewer-page toolbars to accommodate them: instead of adding widgets, slideshows are toggled by a new Auto button that replaces the former Raw, which grew redundant with 1.7's addition of filename taps. The slideshow's button is a no-op without JavaScript, but so are dynamic image scaling, info popups, and Top; users are warned if JavaScript is off, and thumbspage now generally assumes that it will be on. The toolbar now also scrolls to alleviate UI cramping altogether, though this is more important for the preceding note's Full. For more details, see the related 2.0 notes here and here.
Implementation notes:
cross-page state retention in JavaScript turned out to be
trivial: HTML5 session storage, client-side cookies, and the window.name
attribute all persist between pages, and can be used to record a slideshow toggle.
Of these, the latter was used initially because it's simple and won't be disabled,
but HTML5 session storage was later adopted due to a flaw in Safari: see above.
Note that other JavaScript variables won't work—unlike the DOM-defined
window.name
, page-defined attributes (e.g., window.autoToggle
)
do not persist across pages; if used, Auto switches pages once, but the attribute is
undefined
in the next page.
Handling JavaScript timer events is also easy: they're cleared manually
by the page on toggle off and automatically by browsers on gallery/page exit,
and are rescheduled on each viewer-page load while the tab-global slideshow toggle is on
(some of which may be more conceptual than literal: browser back-forward caches and
quirks may muddle this story).
Subtle to be sure, but broadly functional nonetheless.
Update: 2.0 added a floating Top to index pages as an optional-but-default widget, because some pages do have useful top content, users may wish to start a slideshow at the first image, and large indexes can make for lots of scrolling on mobiles. Still, Top can be omitted via a setting in the configs file if it seems useless clutter for your gallery's use case. Where used, its colors, scroll threshold, and page location can also be tailored for custom needs (e.g., for larger or smaller headers and toolbars). For more details, see the 2.0 release note.
Update: no, it cannot—per the 2.0 note, browsers do not support rotations widely enough in fall 2020 to rely on this. thumbspage must rotate images itself because some browser's don't.
filename.ext
, if a text file
filename.txt
is also present in the images folder,
thumbspage could generate an index-page link which opens the text file when
tapped. This would support documentation more specific than the general
opening narrative available in a custom header file. This hasn't been
pursued, because there seems no good way to do this without cluttering
what is an intentionally simple UI: hover events won't apply to mobile,
links on filename labels just below images would be difficult to tap,
and adding new widgets would be too much. In the end, thumbspage
galleries are simple on purpose, and generally prefer to remain that way.
Update:
version 2.3 added per-image notes anyhow, as described in full
here. For the UI, this took the form of a single
Note button on viewer pages (with line-throughs when notes are absent),
along with an equivalent up-swipe gesture on touch. Notes themselves
were limited to plain text in .note
files in the images folder,
with content popped up in-page to be less jarring than separate files.
This is arguably simple, and provides a new narration component
to galleries that's sorely needed in many use cases; it's impossible to
describe photos in an index-page preamble, once the number of images
passes the trivial.
This hasn't been pursued, in part because it seems overkill: thumbspage is a utility that likely won't be run on a frequent basis, and its target audience is not technically naive—its users are content creators who either already know about command lines and folders, or are willing to learn enough to run the script. Moreover, some thumbspage customization requires writing code that's beyond the scope of a GUI (see its header and footer inserts); and GUIs, especially the portable kind, have a way of being much more effort than you may think. In short, a GUI seems an unwarranted can of worms that would consume resources better invested elsewhere. On the other hand...
In the end, thumbspage is really about trying to do something useful in the stunningly ad-hoc and convoluted domain that is today's web development. As usual, fork with care. Just because you can save the village, doesn't mean you should.
You've reached the end of this guide, but there's more to the thumbspage story:
docetc/more-docs-trimmed
.
And may all your monkeys be right-side up.