o
    .wi                     @   s   d dl mZ d dlmZmZmZ d dlmZmZ d dl	m
Z
 d dlmZ d dlmZmZ d dlmZmZ dd	giZes>d
gZG dd deZdS )    )Sequence)AnyOptionalUnion)Tensortensor)$short_time_objective_intelligibility)Metric)_MATPLOTLIB_AVAILABLE_PYSTOI_AVAILABLE)_AX_TYPE_PLOT_OUT_TYPE!ShortTimeObjectiveIntelligibilitypystoi&ShortTimeObjectiveIntelligibility.plotc                	       s   e Zd ZU dZeed< eed< dZeed< dZeed< dZ	eed< d	Z
eed
< dZeed< 	ddedededdf fddZdededdfddZdefddZddeeee df dee defddZ  ZS ) r   a  Calculate STOI (Short-Time Objective Intelligibility) metric for evaluating speech signals.

    Intelligibility measure which is highly correlated with the intelligibility of degraded speech signals, e.g., due
    to additive noise, single-/multi-channel noise reduction, binary masking and vocoded speech as in CI simulations.
    The STOI-measure is intrusive, i.e., a function of the clean and degraded speech signals. STOI may be a good
    alternative to the speech intelligibility index (SII) or the speech transmission index (STI), when you are
    interested in the effect of nonlinear processing to noisy speech, e.g., noise reduction, binary masking algorithms,
    on speech intelligibility. Description taken from  `Cees Taal's website`_ and for further details see `STOI ref1`_
    and `STOI ref2`_.

    This metric is a wrapper for the `pystoi package`_. As the implementation backend implementation only supports
    calculations on CPU, all input will automatically be moved to CPU to perform the metric calculation before being
    moved back to the original device.

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

    - ``preds`` (:class:`~torch.Tensor`): float tensor with shape ``(...,time)``
    - ``target`` (:class:`~torch.Tensor`): float tensor with shape ``(...,time)``

    As output of `forward` and `compute` the metric returns the following output

    - ``stoi`` (:class:`~torch.Tensor`): float scalar tensor

    .. hint::
        Using this metrics requires you to have ``pystoi`` install. Either install as ``pip install
        torchmetrics[audio]`` or ``pip install pystoi``.

    Args:
        fs: sampling frequency (Hz)
        extended: whether to use the extended STOI described in `STOI ref3`_.
        kwargs: Additional keyword arguments, see :ref:`Metric kwargs` for more info.

    Raises:
        ModuleNotFoundError:
            If ``pystoi`` package is not installed

    Example:
        >>> from torch import randn
        >>> from torchmetrics.audio import ShortTimeObjectiveIntelligibility
        >>> preds = randn(8000)
        >>> target = randn(8000)
        >>> stoi = ShortTimeObjectiveIntelligibility(8000, False)
        >>> stoi(preds, target)
        tensor(-0.084...)

    sum_stoitotalFfull_state_updateis_differentiableThigher_is_better        plot_lower_boundg      ?plot_upper_boundfsextendedkwargsreturnNc                    sV   t  jdi | tstd|| _|| _| jdtddd | jdtddd d S )	Nz}STOI metric requires that `pystoi` is installed. Either install as `pip install torchmetrics[audio]` or `pip install pystoi`.r   r   sum)defaultdist_reduce_fxr   r    )super__init__r   ModuleNotFoundErrorr   r   	add_stater   )selfr   r   r   	__class__r    T/home/ubuntu/sommelier/.venv/lib/python3.10/site-packages/torchmetrics/audio/stoi.pyr"   V   s   z*ShortTimeObjectiveIntelligibility.__init__predstargetc                 C   sF   t ||| j| jd| jj}|  j| 7  _|  j| 7  _dS )z*Update state with predictions and targets.FN)	r   r   r   tor   devicer   r   numel)r%   r)   r*   
stoi_batchr    r    r(   updateh   s
   z(ShortTimeObjectiveIntelligibility.updatec                 C   s   | j | j S )zCompute metric.)r   r   )r%   r    r    r(   computeq   s   z)ShortTimeObjectiveIntelligibility.computevalaxc                 C   s   |  ||S )a  Plot a single or multiple values from the metric.

        Args:
            val: Either a single result from calling `metric.forward` or `metric.compute` or a list of these results.
                If no value is provided, will automatically call `metric.compute` and plot that result.
            ax: An matplotlib axis object. If provided will add plot to that axis

        Returns:
            Figure and Axes object

        Raises:
            ModuleNotFoundError:
                If `matplotlib` is not installed

        .. plot::
            :scale: 75

            >>> # Example plotting a single value
            >>> from torch import randn
            >>> from torchmetrics.audio import ShortTimeObjectiveIntelligibility
            >>> preds = randn(8000)
            >>> target = randn(8000)
            >>> metric = ShortTimeObjectiveIntelligibility(8000, False)
            >>> metric.update(preds, target)
            >>> fig_, ax_ = metric.plot()

        .. plot::
            :scale: 75

            >>> # Example plotting multiple values
            >>> from torch import randn
            >>> from torchmetrics.audio import ShortTimeObjectiveIntelligibility
            >>> metric = ShortTimeObjectiveIntelligibility(8000, False)
            >>> preds = randn(8000)
            >>> target = randn(8000)
            >>> values = [ ]
            >>> for _ in range(10):
            ...     values.append(metric(preds, target))
            >>> fig_, ax_ = metric.plot(values)

        )_plot)r%   r1   r2   r    r    r(   plotu   s   *r   )F)NN)__name__
__module____qualname____doc__r   __annotations__r   boolr   r   r   floatr   intr   r"   r/   r0   r   r   r   r   r   r4   __classcell__r    r    r&   r(   r      s,   
 /	2N)collections.abcr   typingr   r   r   torchr   r   "torchmetrics.functional.audio.stoir   torchmetrics.metricr	   torchmetrics.utilities.importsr
   r   torchmetrics.utilities.plotr   r   __doctest_requires____doctest_skip__r   r    r    r    r(   <module>   s   
