forked from pixray/pixray
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfftdrawer.py
executable file
·109 lines (88 loc) · 3.66 KB
/
fftdrawer.py
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
from DrawingInterface import DrawingInterface
from torchvision.utils import save_image
from aphantasia.image import to_valid_rgb, fft_image, dwt_image, pixel_image
import torch
from util import str2bool
# canonical interpolation function, like https://p5js.org/reference/#/p5/map
def map_number(n, start1, stop1, start2, stop2):
return ((n-start1)/(stop1-start1))*(stop2-start2)+start2;
class FftDrawer(DrawingInterface):
@staticmethod
def add_settings(parser):
parser.add_argument("--fft_use", type=str, help="use fft or dwt or pixel", default="fft", dest='fft_use')
parser.add_argument('--fft_decay', default=1.5, type=float, dest='fft_decay')
parser.add_argument('--fft_wave', default='coif2', help='wavelets: db[1..], coif[1..], haar, dmey', dest='fft_wave')
parser.add_argument('--fft_sharp', default=0.3, type=float, dest='fft_sharp')
parser.add_argument('--fft_colors', default=1.5, type=float, dest='fft_colors')
parser.add_argument('--fft_lrate', default=0.3, type=float, help='Learning rate', dest='fft_lrate')
return parser
def __init__(self, settings):
super(DrawingInterface, self).__init__()
self.canvas_width = settings.size[0]
self.canvas_height = settings.size[1]
self.fft_use = settings.fft_use
self.decay = settings.fft_decay
self.wave = settings.fft_wave
self.sharp = settings.fft_sharp
self.colors = settings.fft_colors
self.lrate = settings.fft_lrate
self.img = None
def load_model(self, settings, device):
self.device = device
def get_opts(self):
return self.opts
def rand_init(self, toksX, toksY):
self.init_from_tensor(None)
def init_from_tensor(self, init_tensor):
shape = [1, 3, self.canvas_height, self.canvas_width]
resume = None
if init_tensor is not None:
save_image(init_tensor, "res_init.png")
resume = "res_init.png"
if self.fft_use == "dwt":
print("Using DWT instead of FFT")
params, image_f, sz = dwt_image(shape, self.wave, self.sharp, self.colors, resume=resume)
elif self.fft_use == "pixel":
params, image_f, sz = pixel_image(shape, sd=1, resume=resume)
elif self.fft_use == "fft":
params, image_f, sz = fft_image(shape, sd=0.01, decay_power=self.decay, resume=resume)
else:
raise ValueError(f"fft drawer does not know how to apply fft_use={self.fft_use}")
self.params = params
self.image_f = to_valid_rgb(image_f, colors=1.5)
def get_opts(self, decay_divisor=1):
# Optimizers
optimizer = torch.optim.Adam(self.params, self.lrate / decay_divisor)
self.opts = [optimizer]
return self.opts
def reapply_from_tensor(self, new_tensor):
self.init_from_tensor(new_tensor)
def get_z_from_tensor(self, ref_tensor):
return None
def get_num_resolutions(self):
return None
def synth(self, cur_iteration):
if cur_iteration < 0:
return self.img
img = self.image_f(contrast=0.9)
self.img = img
return img
@torch.no_grad()
def to_image(self):
img = self.img.detach().cpu().numpy()[0]
img = np.transpose(img, (1, 2, 0))
img = np.clip(img, 0, 1)
img = np.uint8(img * 255)
pimg = PIL.Image.fromarray(img, mode="RGB")
return pimg
def clip_z(self):
pass
def get_z(self):
return None
def get_z_copy(self):
return None
def set_z(self, new_z):
return None
@torch.no_grad()
def to_svg(self):
pass