В данной статье я покажу и расскажу, как можно сгенерировать аватарки как на Github.

Результат генерации для ника "test1"
Результат генерации для ника "test1"

Для начала нужно понять, как устроена аватарка с Github'а. На первый взгляд, это просто случайный набор закрашенных квадратов (далее, блоков) в удачном порядке на сером фоне.

Сколько квадратов в аватарке
Сколько квадратов в аватарке

В каждой аватарке 12 на 12 блоков.

Случайная автарка с просторов Github
Случайная автарка с просторов Github

Взглянув на следующую картинку, думаю, вы поняли что изображения симметричны, поэтому будем генерировать матрицу блоков 6 на 12, а затем отразим и сконкатенируем две матрицы, получим матрицу 12 на 12.

Ну что ж, похоже, пора кодировать. Я буду делать это на python.

Подключим библиотеки

from PIL import ImageDraw, Image
import numpy as np
import hashlib

Инициализируем переменные

background_color = '#f2f1f2'
s = 'test1'

Получаем набор псевдослучайных байт. Я буду использовать хеш-функцию для того, чтобы получать картинки от конкретной строки, так результат получится интереснее.

bytes = hashlib.md5(s.encode('utf-8')).digest()

Получаем цвет из хеша

main_color = bytes[:3]
main_color = tuple(channel // 2 + 128 for channel in main_color) # rgb

Генерируем матрицу заполнения блоков, для этого берем следующие байты. Так как матрица 6 на 12, а информации на каждый блок у нас один бит, то нам понадобится:

6 \cdot 12 \cdot 1\text{бит} = 72 \text{бит} = 9 \text{байт}
# матрица блоков 6 на 12
need_color = np.array([bit == '1' for byte in bytes[3:3+9] for bit in bin(byte)[2:].zfill(8)]).reshape(6, 12)

# получаем матрицу 12 на 12 сконкатенировав оригинальную и отраженную матрицу
need_color = np.concatenate((need_color, need_color[::-1]), axis=0)

Рисуем изображения по матрице заполнения

img_size = (avatar_size, avatar_size)
block_size = avatar_size // 12 # размер квадрата

img = Image.new('RGB', img_size, background_color)
draw = ImageDraw.Draw(img)

for x in range(avatar_size):
    for y in range(avatar_size):
        need_to_paint = need_color[x // block_size, y // block_size]
        if need_to_paint:
            draw.point((x, y), main_color)

Отобразим то, что получилось

img.show()

И хоба

Результат
Результат

Хммм, что-то не то. Ах, да, забыл, самые крайние блоки всегда не цветные.

Исправим это, добавив рамку из пустых блоков.

for i in range(12):
    need_color[0, i] = 0
    need_color[11, i] = 0
    need_color[i, 0] = 0
    need_color[i, 11] = 0

Вуаля. Давайте теперь посмотрим на сгенерированные аватарки для других никнеймов.

test2
test2
test3
test3
test4
test4
test5
test5

И напоследок, специально для Хабра.

habr
habr
ufo
ufo

На этом все. Спасибо тем, кто дочитал, а тех, кто хочет экспериментировать, отправляю в свой репозиторий со всем кодом.