-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathindex.html
243 lines (160 loc) · 26.6 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
<!doctype html>
<!-- The Time Machine GitHub pages theme was designed and developed by Jon Rohan, on Feb 7, 2012. -->
<!-- Follow him for fun. http://twitter.com/jonrohan. Tail his code on http://github.com/jonrohan -->
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<link rel="stylesheet" href="stylesheets/stylesheet.css" media="screen"/>
<link rel="stylesheet" href="stylesheets/pygment_trac.css"/>
<script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js"></script>
<script type="text/javascript" src="javascripts/script.js"></script>
<title>Fractal-mosaics</title>
<meta name="description" content="Photomosaics 3.0">
<meta name="viewport" content="width=device-width,initial-scale=1">
<script type="text/javascript">
var _gaq = _gaq || [];
_gaq.push(['_setAccount', 'UA-40518979-1']);
_gaq.push(['_trackPageview']);
(function() {
var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s);
})();
</script>
</head>
<body>
<div class="wrapper">
<header>
<h1 class="title">Fractal-mosaics</h1>
</header>
<div id="container">
<p class="tagline">Explorations in Photomosaics</p>
<div id="main" role="main">
<div class="download-bar">
<div class="inner">
<a href="https://github.com/s-ben/Fractal-Mosaics/raw/gh-pages/code/fractal_mosaics.tgz" class="download-button tar"><span>Download</span></a>
<a href="https://github.com/s-ben/Fractal-Mosaics/raw/gh-pages/code/fractal_mosaics.zip" class="download-button zip"><span>Download</span></a>
<a href="https://github.com/s-ben/Fractal-Mosaics" class="code">View Fractal-mosaics on GitHub</a>
</div>
<span class="blc"></span><span class="trc"></span>
</div>
<article class="markdown-body">
<h1>Abstract</h1>
<p>The Fractal Mosaic algorithm is a rotation, scale, and translation invariant photomosaic algorithm. While traditional photomosaics create a larger image out of smaller square images on a rectangular grid, Fractal Mosaics creates a larger image out of rectangular images that can be placed at arbitrary locations, can be scaled to any size, and can be rotated by any angle. </p>
<p>This is accomplished with a combination of image search (i.e. image retrieval) algorithms and an image registration algorithm developed with NASA funding by Siavash Zokai and George Wolberg at the City College of New York (<a href="http://www-cs.engr.ccny.cuny.edu/~wolberg/pub/icip00.pdf">pdf</a>).</p>
<p align="center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/eye_mosaic_flickr_800pix.jpg?login=s-ben&token=d045d2a20e53d9009700d6829157669f" alt="Alt text">
<p style="text-align:center">Figure 1: Fractal Mosaic created with graffiti image set (best viewed <a href="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/eye_mosaic_flickr_large.jpg?login=s-ben&token=87b09134a5f66d314547e419299a65b7">large</a>)</p>
<p>For more Fractal Mosaics, check out my <a href="http://www.flickr.com/photos/travelingseth/sets/72157623789223762/" target="_blank">Flickr page</a>.</p>
<p>Figure 2 shows three traditional photomosaics created with <a href="http://www.andreaplanet.com/andreamosaic/" target="_blank">AndreaMosaic </a>using the same image set and target image. </p>
<p align="center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/traditional_mosaic_different_resolutions_800pix.png?login=s-ben&token=5dc052c2ee602a1b7bfd08d525e9d00a" alt="Alt text">
<p style="text-align:center">Figure 2: Traditional Photomosaics using large tiles (left), medium sized tiles (middle), and small tiles (right)</p>
<p>As can be seen, using large tiles (left), finer details are lost. Using small tiles (right), we capture target image details, but can’t see individual image detail. Using medium size tiles (middle), we get a compromise between the two. Fractal Mosaics aims to create mosaics that capture both target image details and individual image details, in an aesthetically interesting way.</p>
<p>Many people have, of course, created photomosaic variants with similar aims. Too many to list here. Some of my favorite photomosaic pioneers that I’m aware of: <a href="http://www.flickr.com/photos/tsevis/sets/72157594536252686/" target="_blank">Tsevis</a>, <a href="http://www.flickr.com/photos/village9991/sets/72157603327275992/" target="_blank">Village9991</a>, and <a href="http://www.flickr.com/photos/stellame/" target="_blank">StellaMe</a>. If I’m unaware of your work, <a href="http://www.sethbenton.com/sb_contact.html">contact me</a>!</p>
<p>By allowing for arbitrary placement, rotation, and scaling of the “library images” (images making up the mosaic), there are an many more possible matches to evaluate. This allows for larger, often more aesthetically interesting matches than possible with traditional photomosaics. However, searching such a large solution space on a personal computer presents a challenging problem. This “paper” presents my solution to this problem (two years worth of unpaid work...). It explains the algorithm in depth for those wishing to hack the Matlab code. It should also aid those using the code to create mosaics, as understanding how the algorithm works will help tweak parameters that shape mosaic output.</p>
<p>The following sections will walk you through the main pieces of the Fractal Mosaic algorithm. </p>
<ul>
<li><a href="#introduction">Introduction/Problem Statement</a></li>
<li><a href="#image_registration">Image Registration</a></li>
<li><a href="#image_search">Image Search (Retrieval)</a></li>
<li><a href="#mosaic_rendering">Mosaic Rendering</a></li>
<li><a href="#conclusion">Conclusions / Future Directions</a></li>
</ul>
<h1><a id="introduction"></a>Introduction / Problem Statement</h1>
<p>In a traditional photomosaic, the number of possible matches is large but still easily manageable on a modern PC. The number of possible matches is simply the number of “tiles” (square areas in the target image) times the number of library images. For each tile, a similarity metric is calculated (root mean squared (rms) error, typically) for every library image. The library image with the smallest rms error is placed in the mosaic.</p>
<p>In a Fractal Mosaic, the number of possible possible matches is effectively infinite. For example, let’s say we use 5,000 library images, a 3072 x 1024 pixel target image, and do the following fairly rough quantization of the possible translations, rotations, and scalings: </p>
<ul>
<li>2 degree rotations (180 possible rotations)</li>
<li>Every fifth pixel is a possible placement location (629,145 possible translations)</li>
<li>50 possible scaling values</li>
</ul><p>This gives us (180 rotations) x (629,145 translations) x (50 scalings) x (5,000 library images) = 28,311,525,000,000 possible matches to search. Or ~28 trillion matches. Obviously, a brute force search is not feasible (I tried).</p>
<p>Fractal Mosaics tackles this complexity with three main strategies. One strategy is inherent in the “Log-polar” image registration algorithm at the core of Fractal Mosaics. This algorithm uses a coarse-to-fine multi-resolution framework, whereby estimates computed in lower resolution images are used as initial guesses in higher resolution images. An in-depth explanation of how log polar registration is implemented in Fractal Mosaics is given in the <a href="#logpolar">Log-polar Registration</a> section. </p>
<p>The second main strategy is to use an image search algorithm to filter out library images that are statistically unlikely to produce a match. Details on this can be found in the <a href="#image_search">Image Search (Retrieval)</a> section. </p>
<p>The third main strategy for reducing the number of matches evaluated is to look for large matches first, then not look for smaller matches “under” those matches (i.e., don’t look for matches in areas where we’ve already placed an image). This saves considerable computation. However, it will cause the algorithm to miss aesthetically better matches under the placed matches. For this reason, Fractal Mosaics can be configured to also look for alternate matches under placed images. All matches can optionally be output to PNG files that can be loaded into Photoshop for later compositing. </p>
<h1><a id="image_registration"></a>Image Registration</h1>
<p>Image registration is generally defined as the lining up of images misaligned due to rotation, scale, and position (translation). The field of image registration is mature, constituting of several decades of research. The key insight of Fractal Mosaics is that image registration algorithms can be employed to efficiently search the photomosaic solution space. Instead of evaluating every possible rotation, scaling, and translation for every library image (computationally impossible), we can use image registration to efficiently estimate these parameters. Once these parameters are estimated, they are used to rotate, scale, and translate the library image in question to best match the target image. A similarity metric (rms error in this case) is applied. If the rotated, scaled, and translated library image scores high enough on the similarity metric, it is placed in the mosaic. </p>
<h2>Registration Algorithm Choice</h2>
<p>After an initial survey of the image registration literature, a frequency domain image registration algorithm was evaluated; specifically, Fourier-Mellin-based image registration, as implemented in this <a href="http://www.mathworks.com/matlabcentral/fileexchange/3000-fourier-mellin-based-image-registration-with-gui" target="_blank">Matlab code</a>. This would have been much more computationally efficient than the spatial domain algorithm eventually chosen, as rotation, scale, and translation are efficiently recovered in one step. However, this technique proved not robust enough for the photomosaic application. This is likely because it relies more heavily on the assumption made by all image registration algorithms: that the two images being matched are of the same object/scene. In photomosaics however, we are matching parts of the target image to random library images. The assumption that the library images are rotated, scaled, and translated versions of the target image is weak. </p>
<p>An image registration algorithm that operates in the spatial domain was tried; specifically, the “Log-polar registration” algorithm found in the paper “Robust Image Registration Using Log-Polar Transform” (<a href="http://www-cs.engr.ccny.cuny.edu/~wolberg/pub/icip00.pdf">pdf</a>). This algorithm proved very robust. And most importantly, it met the main aesthetic criteria: for any two images presented to the algorithm, it aligned them the way a human would. To test this claim out for yourself, check out the video below showing the algorithm at work matching library images to a target image. </p>
<p style="text-align:center"><a href="http://www.youtube.com/watch?v=fw7deJSLVE4" target="new"><img src="images/video_screenshot_200rms_500pix.png" width="500" height="427" longdesc="http://www.youtube.com/watch?v=fw7deJSLVE4"></a></p>
<p>The downside to Log-polar registration is increased computation. Because Log-polar registration only recovers rotation and scale, to recover translation, it must be applied to every location (pixel) in the target image. The location resulting in the lowest rms error is then chosen as the estimate. This creates a lot of extra computation. To mitigate this, the authors of the algorithm use a coarse-to-fine multi-resolution framework, whereby estimates computed in lower resolution images are used as initial guesses in higher resolution images. The following section reviews Log-polar registration in Fractal Mosaics.</p>
<h2><a id="logpolar"></a>Log Polar Image Registration</h2>
<p>In “Robust Image Registration Using Log-Polar Transform”, the authors lay out a “two module” approach. First, a “Log-polar registration” module estimates rotation, scale, and translation. Then, a second module uses a nonlinear least squares optimization method to refine this estimation and obtain sub-pixel accuracy. Since Fractal Mosacis does not require sub-pixel accuracy, and the log polar registration module was found to be accurate enough for this application, just the log-polar registration module is implemented.</p>
<p>The following sections overview how Log-polar registration works in general. For nitty gritty details on how the multi-resolution framework was adapted to the photomosaic problem, see the <a href="log_polar_registration_framework.html" target="_blank">Image Registration Algorithm Steps</a> page.</p>
<h3>Polar Coordinates</h3>
<p>To understand Log-polar registration, we briefly review <a href="http://en.wikipedia.org/wiki/Polar_coordinate_system" target="_blank">polar coordinates</a>. </p>
<p>When transforming from Cartesian to polar coordinates, each pixel location (x,y) is represented as a radius and angle (r,a), where <span style="font-style:italic;">r</span> is the distance from the center of the image (x<span style="font-size:xx-small;">C</span>,y<span style="font-size:xx-small;">C</span>), and <span style="font-style:italic;">a</span> is the angle in degrees from the x-axis.</p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/radius_eq_polar.gif?login=s-ben&token=05e5efcd7cad361572bbf941f86ad1f2" alt="equation"></p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/angle_equation_polar.gif?login=s-ben&token=ee4778d991bfe42ea68e377ed9c59f5f" alt="equation"></p>
<p>Figure 3 illustrates the polar transform using an image of Abraham Lincoln. Here we see a crop of the Lincoln image (a), the Lincoln image rotated by 90° (c), and the log-polar transforms of both ((b) and (d)).</p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/Polar_transform_500pix.png?login=s-ben&token=2297c64c8a54efd5b9fe4e7f3a62f311" alt="Alt text"></p>
<p style="text-align:center">Figure 3: Polar transform of Lincoln image rotated 90°</p>
<p>As can be seen, in the polar space, the 90 degree rotation manifests as a circular shift along the x-axis of the polar image (i.e. the angle axis). If we take the cross-correlation of the polar transformed images ((b) and (d)), the maximum of the cross-correlation gives us the rotation. However, scale cannot be recovered from the cross-correlation.</p>
<h3>Log-polar Transform</h3>
<p>Here’s where we take advantage of some math magic. According to the product rule of logarithms, </p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/log_general_equation.gif?login=s-ben&token=daccda0e863fce8c925880b950d70545" alt="equation"></p>
<p>Assuming we scale an image up by a factor of 3, in the polar space, this is equivalent to multiplying the radius values by 3.</p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/polar_scaling_eq.gif?login=s-ben&token=20de8f9d649563130ea241a3573fd5b6" alt="equation"></p>
<p>If we apply the log function to the radius values (i.e. the Log-polar transform),</p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/log_polar_scale_eq.gif?login=s-ben&token=cc3546e0e40606f5ffbcff3586f42c92" alt="equation"></p>
<p>Scale now manifests as a scalar phase shift, <span style="font-style:italic;">log(3)</span>, in the radius axis (i.e. x-axis). Rotation still manifests as a phase shift in the angle axis (i.e. y-axis), as it did in the straight polar transform. We can now recover both scale and rotation from the cross-correlation of the log-polar transformed images. Magic!</p>
<p>Figure 4 shows a crop of the Lincoln image (a), the Lincoln image rotated by 45<span style="text-align:center">°</span> and scaled up by a factor of 2 (c), and the log-polar transforms of both ((b) and (d)). </p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/LogPolar_transform_500pix.png?login=s-ben&token=e72f28a8e8855f7484d116120e03fe1b" alt="Alt text"></p>
<p style="text-align:center">Figure 4: Log-polar transform of Lincoln image rotated 45° and scaled by 2X</p>
<p>As can be seen, the 45<span style="text-align:center">°</span> rotation shows up as a circular shift along the x-axis (i.e. angle axis) and the 2X scaling shows up as a phase shift along the y-axis (i.e. angle axis).</p>
<h3>Rotation and Scale Estimation</h3>
<p>To estimate rotation and scale, we take the cross-correlation of the log-polar transformed images ((b) and (d) in Figure 4). The maximum of the cross-correlation (dark red peak) corresponds to rotation and scale (equations for calculating rotation and scale from peak can be found in the code).</p>
<p align="center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/Lincoln_cc.png?login=s-ben&token=22024bc7d3e71e2a15e9fb2d16e9e80b" alt="Alt text"></p>
<p style="text-align:center">Figure 5: Cross-correlation of the log-polar transformed images ((b) and (d) in Figure 4)</p>
<p>For computational efficiency, the cross-correlation is computed in the Fourier domain using 2D FFTs,</p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/cc_eq.gif?login=s-ben&token=dac60d00b66a927d84272a207994b202" alt="equation"></p>
<p>where L1 and L2 are the log-polar transformed library and target images, respectively.</p>
<h3>Recovering Translation</h3>
<p>As mentioned previously, Log-polar registration does not recover translation. It assumes the two images presented to it line up, except for rotation and scale differences. To recover translation, the algorithm must be applied to every location (pixel). The translation resulting in the best match (i.e. lowest rms error) is then chosen as the translation estimate. </p>
<p>To reduce computation, the authors of Log-polar registration implement a multi-resolution framework, whereby estimates computed in lower resolution images are used as initial guesses in higher resolution images. At each resolution, rotation and scale estimates are computed for every location. The location(s) that result in an rms error below a threshold are sent to the next higher resolution for refining.</p>
<p>The video below shows the algorithm at work evaluating library images. As can be seen, most library images aren't evaluated beyond the lowest resolution (8x8). This is because most library images, even when rotated and scaled to best match the target image, aren't good enough matches to evaluate at higher resolutions. For details on error thresholds and other implementation details, see the <a href="log_polar_registration_framework.html" target="_blank">Image Registration Algorithm Steps</a> page.</p>
<p style="text-align:center"><a href="http://www.youtube.com/watch?v=oGCi5TPagJo" target="_blank"><img src="images/video_screenshot_30rms_500pix.png" width="500" height="447"></a></p>
<p> </p>
<h1><a id="image_search"></a>Image Search (Retrieval) Filtering</h1>
<p>An image search (i.e. retrieval) algorithm is used to filter out statistically unlikely matches before the computationally intensive image registration step (i.e., don’t waste cycles trying to match a white part of the target image with an image of the night sky). Specifically, Fractal Mosaics performs Content-Based Image Retrieval (<a href="http://en.wikipedia.org/wiki/Content-based_image_retrieval" target="_blank">CBIR</a>), similar to Google’s <a href="http://support.google.com/images/answer/1325808?hl=en" target="_blank">search by image</a> feature. The “search image” (the part of the target image we’re looking to match), is compared to every library image using a similarity metric. Then only the most similar library images are evaluated using image registration.</p>
<p>Numerous similarity metrics were evaluated, including histograms, standard deviation, Hu moments, General Fourier Descriptors (GFD), and Fourier Mellin (FM) shape descriptors. The metrics which performed best were histograms, GFDs, and standard deviation. These metrics were combined to create a custom similarity metric that slightly outperformed any individual similarity metric. This metric is currently only used in black and white mode, as I ran out of time to extend it to color. For color, a relatively simple yet effective similarity metric (histogram distance in the HSV color space) is used. Implementation details for black and white and color retrieval are given in the sections below.</p>
<h2>Black and White</h2>
<p>For black and white mosaics, a custom similarity metric was developed. This metric combines three commonly used similarity metrics: histogram distance, General Fourier Descriptor (GFD) distance, and difference in standard deviation. For a given pixel location, the histogram, GFD, and standard deviation of the “search image”, or target tile (piece of the target image we’re matching) are calculated. These are then compared to the histogram, GFD, and standard deviation of every library image, which have been precomputed. For the histogram and GFD, euclidean distance is computed. For standard deviation, the absolute difference is used. </p>
<p>The histogram distance, GFD distance, and standard deviation difference are then input into respective Probability Density Functions (PDFs), which were calculated empirically using 430,500 pseudo-random samples of the solution space. The sum of the output of these PDFs is the “Match PDF”. </p>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/Match_PDF.gif?login=s-ben&token=43c084e1129c2e6a5fd517bd3410cfa3" alt="equation"></p>
<p>All library images are then sorted by their Match PDF scores, and only the images with the highest probability of being a match are sent to the image registration step.</p>
<h2>Color</h2>
<p>When creating a color mosaic, the similarity metric is the distance between 3D histograms in the HSV color space. The code used to implement this (and implementation details) can be found <a href="http://www.mathworks.com/matlabcentral/fileexchange/22030-image-retrieval-query-by-example-demo" target="_blank">here</a>. Thanks to <a href="http://www.mathworks.com/matlabcentral/fileexchange/authors/30223" target="_blank">Theodoros Giannakopoulos</a> for sharing!</p>
<h1><a id="mosaic_rendering"></a>Mosaic Rendering</h1>
<p>The main Fractal Mosaic script renders a version of the mosaic as it goes. Alternately, after running the main script, a separate rendering script can be run, which gives the user the ability to play with different rendering parameters. Notably, you can set:</p>
<ul>
<li style="font-style:normal">Whether the images will be placed on top of the target image or on a black background.</li>
<li fstyle="font-style:normal">How much blending to do between images. Blending is currently done with a simple linear gradient. The user sets a variable which sets the percentage of the image that will be blended. Figure 6 shows a blending window that uses the default, 25%.</li>
<p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/blending_window.png?login=s-ben&token=a0227807701da25bdbebd0b297376baf" alt="Alt text"></p>
<p style="text-align:center">Figure 6: Default blending window (25% of image edge blended)</p>
<li style="font-style:normal">Whether to use color images or not. For color mosaics, this is typically a no brainer. However, you can render a black and white mosaic using the color versions of the black and white images used for matching. This can often lead to interesting, accidental results, as illustrated in Figure 7.</li>
</ul><p style="text-align:center"><img src="https://raw.github.com/s-ben/Fractal-Mosaics/gh-pages/images/graffiti_SF_color_render_ex.png?login=s-ben&token=7dd819d3aeae99b287ad3ae610342b0e" alt="Alt text"></p>
<p style="text-align:center">Figure 7: Example of using color images when rendering a black and white mosaic</p>
<h1><a id="conclusion"></a>Conclusion / Next Steps</h1>
<p>Considerable effort has been put into automating the entire mosaic creation process, and making beautiful mosaics. However, Fractal Mosaics is not programmed to make artistic or conceptual choices on individual images. Therefore, its most promising use in some use cases will be generating input into a another process (e.g. a human using Photoshop or other software). </p>
<p>Considerable effort has been put into optimizing the code, both on the algorithmic and code level. However, it still takes a considerable amount of time to run (e.g. 6-8 hrs for a black and white mosaic with medium complexity, 2-3 days for a equivalent color mosaic). Promising optimization strategies include translation from Matlab to another language such as C/C++, accelerating key functions on the GPU (graphics card), and cloud computing (e.g. run on Amazon's EC2, etc.). </p>
<p>Considerable effort has been put into creating art with the code. However, Fractal Mosaics constitutes a new tool. One that exposes many new possibilities. These possibilities, as with any "new medium", will take time to flesh out. It is my hope that by putting this code out there, artists will take up the challenge and create art I can't even imagine. Have at it!</p>
</article>
</div>
</div>
<footer>
<div class="owner">
<p><a href="https://github.com/s-ben" class="avatar"><img src="https://secure.gravatar.com/avatar/3d1ddae23883c53b38057c6c8db71e6d?s=30&d=https://a248.e.akamai.net/assets.github.com%2Fimages%2Fgravatars%2Fgravatar-user-420.png" width="48" height="48"/></a> <a href="https://github.com/s-ben">s-ben</a> maintains <a href="https://github.com/s-ben/Fractal-Mosaics">Fractal-mosaics</a></p>
</div>
<div class="creds">
<small>This page generated using <a href="https://pages.github.com/">GitHub Pages</a><br/>theme by <a href="http://twitter.com/jonrohan/">Jon Rohan</a></small>
</div>
</footer>
</div>
<div class="current-section">
<a href="#top">Scroll to top</a>
<a href="https://github.com/s-ben/Fractal-Mosaics/tarball/master" class="tar">tar</a><a href="https://github.com/s-ben/Fractal-Mosaics/zipball/master" class="zip">zip</a><a href="" class="code">source code</a>
<p class="name"></p>
</div>
</body>
</html>