import torch
import torch.nn as nn
from torch.nn import Parameter

def conv3x3(in_planes, out_planes, stride=1):
    return nn.Conv2d(in_planes, out_planes, kernel_size=3, stride=stride, padding=1, bias=False)

class SEBlock(nn.Module):
    def __init__(self, channel, reduction=16):
        super(SEBlock, self).__init__()
        self.avg_pool = nn.AdaptiveAvgPool2d(1)
        self.fc = nn.Sequential(
            nn.Linear(channel, channel // reduction), nn.PReLU(),
            nn.Linear(channel // reduction, channel), nn.Sigmoid()
        )
    def forward(self, x):
        b, c, _, _ = x.size(); y = self.avg_pool(x).view(b, c); y = self.fc(y).view(b, c, 1, 1)
        return x * y

class IRBlock(nn.Module):
    expansion = 1
    def __init__(self, inplanes, planes, stride=1, downsample=None, use_se=True):
        super(IRBlock, self).__init__()
        self.bn0 = nn.BatchNorm2d(inplanes); self.conv1 = conv3x3(inplanes, inplanes)
        self.bn1 = nn.BatchNorm2d(inplanes); self.prelu = nn.PReLU()
        self.conv2 = conv3x3(inplanes, planes, stride); self.bn2 = nn.BatchNorm2d(planes)
        self.downsample = downsample; self.use_se = use_se
        if self.use_se: self.se = SEBlock(planes)
    def forward(self, x):
        residual = x; out = self.bn0(x); out = self.conv1(out); out = self.bn1(out); out = self.prelu(out)
        out = self.conv2(out); out = self.bn2(out)
        if self.use_se: out = self.se(out)
        if self.downsample: residual = self.downsample(x)
        out += residual; out = self.prelu(out); return out

class ResNet(nn.Module):
    def __init__(self, block, layers, use_se=True):
        self.inplanes = 64; self.use_se = use_se
        super(ResNet, self).__init__()
        self.conv1 = nn.Conv2d(3, 64, 3, 1, bias=False); self.bn1 = nn.BatchNorm2d(64); self.prelu = nn.PReLU()
        self.maxpool = nn.MaxPool2d(2, 2)
        self.layer1 = self._make_layer(block, 64, layers[0])
        self.layer2 = self._make_layer(block, 128, layers[1], 2)
        self.layer3 = self._make_layer(block, 256, layers[2], 2)
        self.layer4 = self._make_layer(block, 512, layers[3], 2)
        self.bn2 = nn.BatchNorm2d(512); self.dropout = nn.Dropout()
        self.fc = nn.Linear(512 * 7 * 7, 512); self.bn3 = nn.BatchNorm1d(512)
        for m in self.modules():
            if isinstance(m, nn.Conv2d): nn.init.xavier_normal_(m.weight)
            elif isinstance(m, (nn.BatchNorm2d, nn.BatchNorm1d)): nn.init.constant_(m.weight, 1); nn.init.constant_(m.bias, 0)
            elif isinstance(m, nn.Linear): nn.init.xavier_normal_(m.weight); nn.init.constant_(m.bias, 0)
    def _make_layer(self, block, planes, blocks, stride=1):
        downsample = None
        if stride != 1 or self.inplanes != planes * block.expansion:
            downsample = nn.Sequential(nn.Conv2d(self.inplanes, planes * block.expansion, 1, stride, bias=False), nn.BatchNorm2d(planes * block.expansion))
        layers = []; layers.append(block(self.inplanes, planes, stride, downsample, use_se=self.use_se)); self.inplanes = planes
        for _ in range(1, blocks): layers.append(block(self.inplanes, planes, use_se=self.use_se))
        return nn.Sequential(*layers)
    def forward(self, x):
        x = self.conv1(x); x = self.bn1(x); x = self.prelu(x); x = self.maxpool(x)
        x = self.layer1(x); x = self.layer2(x); x = self.layer3(x); x = self.layer4(x)
        x = self.bn2(x); x = self.dropout(x); x = x.view(x.size(0), -1); x = self.fc(x); x = self.bn3(x); return x