_util.py 4.12 KB
Newer Older
1
2
3
import numpy as np
from scipy.interpolate import interp1d

Wei Chengliang's avatar
Wei Chengliang committed
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
def binningPSF(img, ngg):
    imgX = img.reshape(ngg, img.shape[0]//ngg, ngg, img.shape[1]//ngg).mean(-1).mean(1)
    return imgX


def radial_average_at_pixel(image, center_x, center_y, dr=10):
    # Get coordinates relative to the specified center pixel (x, y)
    y, x = np.indices(image.shape)
    r = np.sqrt((x - center_x)**2 + (y - center_y)**2)

    # Set up bins
    max_radius = int(r.max())  # Maximum distance from the center pixel
    radial_bins = np.arange(0, max_radius, dr)

    # Compute average value in each bin
    radial_means = []
    for i in range(len(radial_bins) - 1):
        mask = (r >= radial_bins[i]) & (r < radial_bins[i + 1])
        if np.any(mask):
            radial_means.append(image[mask].mean())
        else:
            radial_means.append(0)  # In case no pixels are in the bin
    return radial_bins[:-1], radial_means  # Exclude last bin since no mean calculated


def psf_extrapolate(psf, rr_trim=64, ngg=256):
    # ngg = 256
    # extrapolate PSF
    if True:
        xim = np.arange(256)-128
Wei Chengliang's avatar
Wei Chengliang committed
35
36
        xim, yim = np.meshgrid(xim, xim)
        rim = np.sqrt(xim**2 + yim**2)
37
38
39

        # rr_trim  = 96
        psf_temp = psf
Wei Chengliang's avatar
Wei Chengliang committed
40
        psf_temp[rim > rr_trim] = 0
41
42
43
44
45
46
47
48
49
50
        radii, means = radial_average_at_pixel(psf_temp, 128, 128, dr=4)

        radii_log = np.log(radii[1:])
        means_log = np.log(means[1:])

        finite_mask = np.isfinite(means_log)
        f_interp = interp1d(radii_log[finite_mask][:-1], means_log[finite_mask][:-1], kind='linear', fill_value="extrapolate")

        # ngg = 1024
        xim = np.arange(ngg)-int(ngg/2)
Wei Chengliang's avatar
Wei Chengliang committed
51
52
53
        xim, yim = np.meshgrid(xim, xim)
        rim = np.sqrt(xim**2 + yim**2)
        rim[int(ngg/2), int(ngg/2)] = np.finfo(float).eps  # 1e-7
54
55
56
57
58
        rim_log = np.log(rim)
        y_new = f_interp(rim_log)

        arr = np.zeros([ngg, ngg])
        arr[int(ngg/2-128):int(ngg/2+128), int(ngg/2-128):int(ngg/2+128)] = np.log(psf_temp + np.finfo(float).eps)
Wei Chengliang's avatar
Wei Chengliang committed
59
60
        arr[rim > rr_trim] = 0
        arr[arr == 0] = y_new[arr == 0]
61
        psf = np.exp(arr)
Wei Chengliang's avatar
Wei Chengliang committed
62
        psf[rim > int(ngg/2)] = 0
63
64
65
    imPSF = psf  # binningPSF(psf, int(ngg/2))
    imPSF = imPSF/np.nansum(imPSF)
    return imPSF
66

Wei Chengliang's avatar
Wei Chengliang committed
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
def psf_extrapolate1(psf, rr_trim=64, ngg=256):
    # ngg = 256
    # extrapolate PSF
    if True:
        psf_enlar = np.zeros([ngg, ngg])
        psf_enlar[int(ngg/2-128):int(ngg/2+128), int(ngg/2-128):int(ngg/2+128)] = psf
        xim = np.arange(ngg)-ngg/2
        xim, yim = np.meshgrid(xim, xim)
        rim = np.sqrt(xim**2 + yim**2)

        psf_temp = psf_enlar
        # psf_temp[rim >= rr_trim] = 0
        psf_temp[rim >= ngg/2-2] = np.finfo(float).eps
        radii, means = radial_average_at_pixel(psf_temp, ngg/2, ngg/2, dr=2)

        radii_log = np.log(radii[1:])
        # radii_log = radii[1:]
        means_log = np.log(means[1:])

        # xim = np.arange(256)-128
        # xim, yim = np.meshgrid(xim, xim)
        # rim = np.sqrt(xim**2 + yim**2)

        # # rr_trim  = 96
        # psf_temp = psf
        # psf_temp[rim > rr_trim] = 0
        # radii, means = radial_average_at_pixel(psf_temp, 128, 128, dr=4)

        # radii_log = np.log10(radii[1:])
        # # radii_log = radii[1:]
        # means_log = np.log10(means[1:])

        finite_mask = np.isfinite(means_log)
        f_interp = interp1d(radii_log[finite_mask][:-1], means_log[finite_mask][:-1], kind='linear', fill_value="extrapolate")

        # ngg = 1024
        # xim = np.arange(ngg)-int(ngg/2)
        # xim, yim = np.meshgrid(xim, xim)
        # rim = np.sqrt(xim**2 + yim**2)
        # rim[int(ngg/2), int(ngg/2)] = np.finfo(float).eps  # 1e-7
        rim_log = np.log(rim)
        y_new = f_interp(rim_log)

        arr = np.zeros([ngg, ngg])
        arr[int(ngg/2-128):int(ngg/2+128), int(ngg/2-128):int(ngg/2+128)] = np.log(psf + np.finfo(float).eps)
        arr[rim > 128-2] = 0
        arr[arr == 0] = y_new[arr == 0]
        psf_n = np.exp(arr)
        # psf_n[int(ngg/2-128):int(ngg/2+128), int(ngg/2-128):int(ngg/2+128)] = psf
        # psf[rim > int(ngg/2)] = 0
    imPSF = psf_n  # binningPSF(psf, int(ngg/2))
    # imPSF = imPSF/np.nansum(imPSF)
    return imPSF