o
    yi)                     @   s   d dl mZmZmZmZmZ d dlZd dlmZ d dlm	Z	 d dl
mZ d dlmZ d dlmZ d dlmZmZ G d	d
 d
eZdS )    )AnyCallableOptionalTupleno_type_checkN)Tensor)Literal)_dice_compute)_stat_scores_update)Metric)AverageMethodMDMCAverageMethodc                       s   e Zd ZU dZdZeed< dZeed< dZeed< e						
				d!de
dee
 dedeed  dee dee
 dee
 dee deddf fddZe	dededdfddZe	deeeeef fddZe	defdd Z  ZS )"Dicea  Computes `Dice`_:

    .. math:: \text{Dice} = \frac{\text{2 * TP}}{\text{2 * TP} + \text{FP} + \text{FN}}

    Where :math:`\text{TP}` and :math:`\text{FP}` represent the number of true positives and
    false positives respecitively.

    It is recommend set `ignore_index` to index of background class.

    The reduction method (how the precision scores are aggregated) is controlled by the
    ``average`` parameter, and additionally by the ``mdmc_average`` parameter in the
    multi-dimensional multi-class case.

    As input to ``forward`` and ``update`` the metric accepts the following input:

    - ``preds`` (:class:`~torch.Tensor`): Predictions from model (probabilities, logits or labels)
    - ``target`` (:class:`~torch.Tensor`): Ground truth values

    As output to ``forward`` and ``compute`` the metric returns the following output:

    - ``dice`` (:class:`~torch.Tensor`): A tensor containing the dice score.

        - If ``average in ['micro', 'macro', 'weighted', 'samples']``, a one-element tensor will be returned
        - If ``average in ['none', None]``, the shape will be ``(C,)``, where ``C`` stands  for the number of classes

    Args:
        num_classes:
            Number of classes. Necessary for ``'macro'``, ``'weighted'`` and ``None`` average methods.
        threshold:
            Threshold for transforming probability or logit predictions to binary (0,1) predictions, in the case
            of binary or multi-label inputs. Default value of 0.5 corresponds to input being probabilities.
        zero_division:
            The value to use for the score if denominator equals zero.
        average:
            Defines the reduction that is applied. Should be one of the following:

            - ``'micro'`` [default]: Calculate the metric globally, across all samples and classes.
            - ``'macro'``: Calculate the metric for each class separately, and average the
              metrics across classes (with equal weights for each class).
            - ``'weighted'``: Calculate the metric for each class separately, and average the
              metrics across classes, weighting each class by its support (``tp + fn``).
            - ``'none'`` or ``None``: Calculate the metric for each class separately, and return
              the metric for every class.
            - ``'samples'``: Calculate the metric for each sample, and average the metrics
              across samples (with equal weights for each sample).

            .. note::
               What is considered a sample in the multi-dimensional multi-class case
               depends on the value of ``mdmc_average``.

        mdmc_average:
            Defines how averaging is done for multi-dimensional multi-class inputs (on top of the
            ``average`` parameter). Should be one of the following:

            - ``None`` [default]: Should be left unchanged if your data is not multi-dimensional
              multi-class.

            - ``'samplewise'``: In this case, the statistics are computed separately for each
              sample on the ``N`` axis, and then averaged over samples.
              The computation for each sample is done by treating the flattened extra axes ``...``
              as the ``N`` dimension within the sample,
              and computing the metric for the sample based on that.

            - ``'global'``: In this case the ``N`` and ``...`` dimensions of the inputs
              are flattened into a new ``N_X`` sample axis, i.e.
              the inputs are treated as if they were ``(N_X, C)``.
              From here on the ``average`` parameter applies as usual.

        ignore_index:
            Integer specifying a target class to ignore. If given, this class index does not contribute
            to the returned score, regardless of reduction method. If an index is ignored, and ``average=None``
            or ``'none'``, the score for the ignored class will be returned as ``nan``.

        top_k:
            Number of the highest probability or logit score predictions considered finding the correct label,
            relevant only for (multi-dimensional) multi-class inputs. The
            default value (``None``) will be interpreted as 1 for these inputs.
            Should be left at default (``None``) for all other types of inputs.

        multiclass:
            Used only in certain special cases, where you want to treat inputs as a different type
            than what they appear to be.

        kwargs: Additional keyword arguments, see :ref:`Metric kwargs` for more info.

    Raises:
        ValueError:
            If ``average`` is none of ``"micro"``, ``"macro"``, ``"weighted"``, ``"samples"``, ``"none"``, ``None``.
        ValueError:
            If ``mdmc_average`` is not one of ``None``, ``"samplewise"``, ``"global"``.
        ValueError:
            If ``average`` is set but ``num_classes`` is not provided.
        ValueError:
            If ``num_classes`` is set and ``ignore_index`` is not in the range ``[0, num_classes)``.

    Example:
        >>> import torch
        >>> from torchmetrics import Dice
        >>> preds  = torch.tensor([2, 0, 2, 1])
        >>> target = torch.tensor([1, 1, 2, 0])
        >>> dice = Dice(average='micro')
        >>> dice(preds, target)
        tensor(0.2500)
    Fis_differentiableThigher_is_betterfull_state_updater   N      ?microglobalzero_divisionnum_classes	thresholdaverage)r   macroweightednonemdmc_averageignore_indextop_k
multiclasskwargsreturnc	                    s  t  jdi |	 d}
||
vrtd|
 d| dtjtjd f}d|	vr0||v r,tjn||	d< d|	vr8||	d< || _|| _|| _	|| _
|| _|| _|| _|dvrYtd| d	|d
vretd| d	|dkrs|ro|dk rstd|r|d ur||k r|dkrtd| d| ddd }d}|dkr|dkr|dkrg  n|dkr|g ntd| d fdd}d}dD ]}| j|| |d q|| _|| _d S )N)r   r   r   samplesr   NzThe `average` has to be one of z, got .reducemdmc_reduce)r   r   r"   zThe `reduce` z is not valid.)N
samplewiser   zThe `mdmc_reduce` r      zMWhen you set `average` as 'macro', you have to provide the number of classes.zThe `ignore_index` z is not valid for inputs with z classesc                   S   s   g S )N r(   r(   r(   T/home/ubuntu/.local/lib/python3.10/site-packages/torchmetrics/classification/dice.py<lambda>   s    zDice.__init__.<locals>.<lambda>catr&   r"   r   zWrong reduce=""c                      s   t j t jdS )N)dtype)torchzeroslongr(   zeros_shaper(   r)   r*      s    sum)tpfptnfn)defaultdist_reduce_fxr(   )super__init__
ValueErrorr   WEIGHTEDNONEMACROr$   r%   r   r   r   r   r   	add_stater   r   )selfr   r   r   r   r   r   r   r   r    allowed_average_reduce_optionsr8   	reduce_fns	__class__r1   r)   r;      sL   
zDice.__init__predstargetc                 C   s   t ||| j| j| j| j| j| j| jd	\}}}}| jtj	krB| jt
jkrB|  j|7  _|  j|7  _|  j|7  _|  j|7  _dS | j| | j| | j| | j| dS )z*Update state with predictions and targets.)r$   r%   r   r   r   r   r   N)r
   r$   r%   r   r   r   r   r   r   SAMPLESr   
SAMPLEWISEr4   r5   r6   r7   append)rA   rH   rI   r4   r5   r6   r7   r(   r(   r)   update   s(   zDice.updatec                 C   s   t | jtrt| jn| j}t | jtrt| jn| j}t | jtr*t| jn| j}t | jtr9t| jn| j}||||fS )zbPerforms concatenation on the stat scores if neccesary, before passing them to a compute function.)
isinstancer4   listr.   r+   r5   r6   r7   )rA   r4   r5   r6   r7   r(   r(   r)   _get_final_stats   s
   zDice._get_final_statsc                 C   s(   |   \}}}}t|||| j| j| jS )zComputes metric.)rP   r	   r   r%   r   )rA   r4   r5   _r7   r(   r(   r)   compute   s   zDice.compute)r   Nr   r   r   NNN)__name__
__module____qualname____doc__r   bool__annotations__r   r   r   intr   floatr   strr   r;   r   rM   r   rP   rR   __classcell__r(   r(   rF   r)   r      sR   
 h
	
=r   )typingr   r   r   r   r   r.   r   typing_extensionsr   +torchmetrics.functional.classification.dicer	   2torchmetrics.functional.classification.stat_scoresr
   torchmetrics.metricr   torchmetrics.utilities.enumsr   r   r   r(   r(   r(   r)   <module>   s   