"""
Objects to manage values on an Open Sound Control port.
OscSend takes the first value of each buffersize and send it on an
OSC port.
OscReceive creates and returns audio streams from the value in its
input port.
The audio streams of these objects are essentially intended to be
controls and can't be sent to the output soundcard.
These objects are available only if pyo is built with OSC (Open Sound
Control) support.
"""
"""
Copyright 2009-2015 Olivier Belanger
This file is part of pyo, a python module to help digital signal
processing script creation.
pyo is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
pyo is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with pyo.  If not, see <http://www.gnu.org/licenses/>.
"""
from ._core import *
from ._maps import *
def assertOSCSupport(obj):
    if not withOSC():
        raise Exception("Pyo built without OSC support! '%s' objects is not available." % obj.__class__.__name__)
### TODO - Know bugs:
### OscListReceive.setValue(address, value) make the program segfault on quit (python 2.7 and 3.5).
### OscSend/OscReceive don't work with unicode on python 2.7 (ok on python 3.5)
###
######################################################################
### Open Sound Control
######################################################################
[docs]class OscSend(PyoObject):
    """
    Sends values over a network via the Open Sound Control protocol.
    Uses the OSC protocol to share values to other softwares or other
    computers. Only the first value of each input buffersize will be
    sent on the OSC port.
    :Parent: :py:class:`PyoObject`
    :Args:
        input: PyoObject
            Input signal.
        port: int
            Port on which values are sent. Receiver should listen on the
            same port.
        address: string
            Address used on the port to identify values. Address is in
            the form of a Unix path (ex.: '/pitch').
        host: string, optional
            IP address of the target computer. The default, '127.0.0.1',
            is the localhost.
    .. note::
        The out() method is bypassed. OscSend's signal can not be sent
        to audio outs.
        OscSend has no `mul` and `add` attributes.
    >>> s = Server().boot()
    >>> s.start()
    >>> a = Sine(freq=[1,1.5], mul=[100,.1], add=[600, .1])
    >>> b = OscSend(a, port=10001, address=['/pitch','/amp'])
    """
    def __init__(self, input, port, address, host="127.0.0.1"):
        assertOSCSupport(self)
        pyoArgsAssert(self, "oiss", input, port, address, host)
        PyoObject.__init__(self)
        self._input = input
        self._in_fader = InputFader(input)
        in_fader, port, address, host, lmax = convertArgsToLists(self._in_fader, port, address, host)
        self._base_objs = [
            OscSend_base(wrap(in_fader, i), wrap(port, i), wrap(address, i), wrap(host, i)) for i in range(lmax)
        ]
        self._init_play()
[docs]    def out(self, chnl=0, inc=1, dur=0, delay=0):
        return self.play(dur, delay) 
[docs]    def setMul(self, x):
        pass 
[docs]    def setAdd(self, x):
        pass 
[docs]    def setBufferRate(self, x):
        """
        Sets how many buffers to wait before sending a new value.
        :Args:
            x: int
                Changes the data output frequency in multiples of the buffer size.
                Should be greater or equal to 1.
        """
        pyoArgsAssert(self, "I", x)
        [obj.setBufferRate(x) for obj in self._base_objs] 
    @property
    def input(self):
        """PyoObject. Input signal."""
        return self._input
    @input.setter
    def input(self, x):
        self.setInput(x) 
[docs]class OscReceive(PyoObject):
    """
    Receives values over a network via the Open Sound Control protocol.
    Uses the OSC protocol to receive values from other softwares or
    other computers. Get a value at the beginning of each buffersize
    and fill its buffer with it.
    :Parent: :py:class:`PyoObject`
    :Args:
        port: int
            Port on which values are received. Sender should output on
            the same port.
            Unlike OscSend object, there can be only one port per OscReceive
            object.
            Available at initialization time only.
        address: string
            Address used on the port to identify values. Address is in
            the form of a Unix path (ex.: '/pitch').
    .. note::
        Audio streams are accessed with the `address` string parameter.
        The user should call :
        OscReceive['/pitch'] to retreive stream named '/pitch'.
        The out() method is bypassed. OscReceive's signal can not be sent
        to audio outs.
    >>> s = Server().boot()
    >>> s.start()
    >>> a = OscReceive(port=10001, address=['/pitch', '/amp'])
    >>> b = Sine(freq=a['/pitch'], mul=a['/amp']).mix(2).out()
    """
    def __init__(self, port, address, mul=1, add=0):
        assertOSCSupport(self)
        pyoArgsAssert(self, "IsOO", port, address, mul, add)
        PyoObject.__init__(self, mul, add)
        address, mul, add, lmax = convertArgsToLists(address, mul, add)
        self._address = address
        self._mainReceiver = OscReceiver_base(port, address)
        self._base_objs = [
            OscReceive_base(self._mainReceiver, wrap(address, i), wrap(mul, i), wrap(add, i)) for i in range(lmax)
        ]
        self._init_play()
    def __getitem__(self, i):
        if type(i) in [bytes, str]:
            return self._base_objs[self._address.index(i)]
        elif i < len(self._base_objs):
            return self._base_objs[i]
        else:
            print("'i' too large!")
[docs]    def getAddresses(self):
        """
        Returns the addresses managed by the object.
        """
        return self._address 
[docs]    def addAddress(self, path, mul=1, add=0):
        """
        Adds new address(es) to the object's handler.
        :Args:
            path: string or list of strings
                New path(s) to receive from.
            mul: float or PyoObject
                Multiplication factor. Defaults to 1.
            add: float or PyoObject
                Addition factor. Defaults to 0.
        """
        pyoArgsAssert(self, "sOO", path, mul, add)
        path, lmax = convertArgsToLists(path)
        mul, add, lmax2 = convertArgsToLists(mul, add)
        for i, p in enumerate(path):
            if p not in self._address:
                self._mainReceiver.addAddress(p)
                self._address.append(p)
                self._base_objs.append(OscReceive_base(self._mainReceiver, p, wrap(mul, i), wrap(add, i)))
                self._base_objs[-1].play() 
[docs]    def delAddress(self, path):
        """
        Removes address(es) from the object's handler.
        :Args:
            path: string or list of strings
                Path(s) to remove.
        """
        pyoArgsAssert(self, "s", path)
        path, lmax = convertArgsToLists(path)
        self._mainReceiver.delAddress(path)
        indexes = [self._address.index(p) for p in path]
        for ind in reversed(indexes):
            self._address.pop(ind)
            obj = self._base_objs.pop(ind) 
[docs]    def setInterpolation(self, x):
        """
        Activate/Deactivate interpolation. Activated by default.
        :Args:
            x: boolean
                True activates the interpolation, False deactivates it.
        """
        pyoArgsAssert(self, "B", x)
        [obj.setInterpolation(x) for obj in self._base_objs] 
[docs]    def setValue(self, path, value):
        """
        Sets value for a given address.
        :Args:
            path: string
                Address to which the value should be attributed.
            value: float
                Value to attribute to the given address.
        """
        pyoArgsAssert(self, "sn", path, value)
        path, value, lmax = convertArgsToLists(path, value)
        for i in range(lmax):
            p = wrap(path, i)
            if p in self._address:
                self._mainReceiver.setValue(p, wrap(value, i))
            else:
                print('Error: OscReceive.setValue, Illegal address "%s"' % p) 
[docs]    def get(self, identifier=None, all=False):
        """
        Return the first sample of the current buffer as a float.
        Can be used to convert audio stream to usable Python data.
        Address as string must be given to `identifier` to specify
        which stream to get value from.
        :Args:
            identifier: string
                Address string parameter identifying audio stream.
                Defaults to None, useful when `all` is True to
                retreive all streams values.
            all: boolean, optional
                If True, the first value of each object's stream
                will be returned as a list. Otherwise, only the value
                of the first object's stream will be returned as a float.
                Defaults to False.
        """
        if not all:
            return self._base_objs[self._address.index(identifier)]._getStream().getValue()
        else:
            return [obj._getStream().getValue() for obj in self._base_objs] 
[docs]    def play(self, dur=0, delay=0):
        if type(dur) is list:
            maindur = dur[0]
        else:
            maindur = dur
        if type(delay) is list:
            maindelay = delay[0]
        else:
            maindelay = delay
        tmp = self._mainReceiver.play(maindur, maindelay)
        return PyoObject.play(self, dur, delay) 
[docs]    def out(self, chnl=0, inc=1, dur=0, delay=0):
        return self.play(dur, delay) 
[docs]    def stop(self, wait=0):
        self._mainReceiver.stop(wait)
        return PyoObject.stop(self, wait)  
[docs]class OscDataSend(PyoObject):
    """
    Sends data values over a network via the Open Sound Control protocol.
    Uses the OSC protocol to share values to other softwares or other
    computers. Values are sent on the form of a list containing `types`
    elements.
    :Parent: :py:class:`PyoObject`
    :Args:
        types: str
            String specifying the types sequence of the message to be sent.
            Possible values are:
            - "i": integer
            - "h": long integer
            - "f": float
            - "d": double
            - "s" ; string
            - "b": blob (list of chars)
            - "m": MIDI packet (list of 4 bytes: [midi port, status, data1, data2])
            - "c": char
            - "T": True
            - "F": False
            - "N": None (nil)
            The string "ssfi" indicates that the value to send will be a list
            containing two strings followed by a float and an integer.
        port: int
            Port on which values are sent. Receiver should listen on the
            same port.
        address: string
            Address used on the port to identify values. Address is in
            the form of a Unix path (ex.: '/pitch').
        host: string, optional
            IP address of the target computer. The default, '127.0.0.1',
            is the localhost.
    .. note::
        The out() method is bypassed. OscDataSend has no audio signal.
        OscDataSend has no `mul` and `add` attributes.
    >>> s = Server().boot()
    >>> s.start()
    >>> def pp(address, *args):
    ...     print(address)
    ...     print(args)
    >>> r = OscDataReceive(9900, "/data/test", pp)
    >>> # Send various types
    >>> a = OscDataSend("fissif", 9900, "/data/test")
    >>> msg = [3.14159, 1, "Hello", "world!", 2, 6.18]
    >>> a.send(msg)
    >>> # Send a blob
    >>> b = OscDataSend("b", 9900, "/data/test")
    >>> msg = [[chr(i) for i in range(10)]]
    >>> b.send(msg)
    >>> # Send a MIDI noteon on port 0
    >>> c = OscDataSend("m", 9900, "/data/test")
    >>> msg = [[0, 144, 60, 100]]
    >>> c.send(msg)
    """
    def __init__(self, types, port, address, host="127.0.0.1"):
        assertOSCSupport(self)
        pyoArgsAssert(self, "siss", types, port, address, host)
        PyoObject.__init__(self)
        types, port, address, host, lmax = convertArgsToLists(types, port, address, host)
        self._base_objs = [
            OscDataSend_base(wrap(types, i), wrap(port, i), wrap(address, i), wrap(host, i)) for i in range(lmax)
        ]
        self._addresses = {}
        for i, adr in enumerate(address):
            self._addresses[adr] = self._base_objs[i]
        self._init_play()
[docs]    def out(self, chnl=0, inc=1, dur=0, delay=0):
        return self.play(dur, delay) 
[docs]    def setMul(self, x):
        pass 
[docs]    def setAdd(self, x):
        pass 
[docs]    def getAddresses(self):
        """
        Returns the addresses managed by the object.
        """
        return list(self._addresses.keys()) 
[docs]    def addAddress(self, types, port, address, host="127.0.0.1"):
        """
        Adds new address(es) to the object's handler.
        :Args:
            types: str
                String specifying the types sequence of the message to be sent.
                Possible values are:
                - "i": integer
                - "h": long integer
                - "f": float
                - "d": double
                - "s" ; string
                - "b": blob (list of chars)
                - "m": MIDI packet (list of 4 bytes: [midi port, status, data1, data2])
                - "c": char
                - "T": True
                - "F": False
                - "N": None (nil)
                The string "ssfi" indicates that the value to send will be a list
                containing two strings followed by a float and an integer.
            port: int
                Port on which values are sent. Receiver should listen on the
                same port.
            address: string
                Address used on the port to identify values. Address is in
                the form of a Unix path (ex.: '/pitch').
            host: string, optional
                IP address of the target computer. The default, '127.0.0.1',
                is the localhost.
        """
        pyoArgsAssert(self, "siss", types, port, address, host)
        types, port, address, host, lmax = convertArgsToLists(types, port, address, host)
        objs = [OscDataSend_base(wrap(types, i), wrap(port, i), wrap(address, i), wrap(host, i)) for i in range(lmax)]
        self._base_objs.extend(objs)
        for i, adr in enumerate(address):
            self._addresses[adr] = objs[i]
        self.play() 
[docs]    def delAddress(self, path):
        """
        Removes address(es) from the object's handler.
        :Args:
            path: string or list of strings
                Path(s) to remove.
        """
        pyoArgsAssert(self, "s", path)
        path, lmax = convertArgsToLists(path)
        for p in path:
            if p in self._addresses:
                self._base_objs.remove(self._addresses[p])
                del self._addresses[p] 
[docs]    def send(self, msg, address=None):
        """
        Method used to send `msg` values as a list.
        :Args:
            msg: list
                List of values to send. Types of values in list
                must be of the kind defined of `types` argument
                given at the object's initialization.
            address: string, optional
                Address destination to send values. If None, values
                will be sent to all addresses managed by the object.
        """
        if address is None:
            pyoArgsAssert(self, "l", msg)
            [obj.send(msg) for obj in self._base_objs]
        else:
            pyoArgsAssert(self, "lS", msg, address)
            self._addresses[address].send(msg)  
[docs]class OscDataReceive(PyoObject):
    """
    Receives data values over a network via the Open Sound Control protocol.
    Uses the OSC protocol to receive data values from other softwares or
    other computers. When a message is received, the function given at the
    argument `function` is called with the current address destination in
    argument followed by a tuple of values.
    :Parent: :py:class:`PyoObject`
    :Args:
        port: int
            Port on which values are received. Sender should output on
            the same port. Unlike OscDataSend object, there can be only
            one port per OscDataReceive object. Available at initialization
            time only.
        address: string
            Address used on the port to identify values. Address is in
            the form of a Unix path (ex.: "/pitch"). There can be as many
            addresses as needed on a single port.
        function: callable (can't be a list)
            This function will be called whenever a message with a known
            address is received. there can be only one function per
            OscDataReceive object. Available at initialization time only.
    .. note::
        The header of the callable given at `function` argument must be in this form::
            def my_func(address, *args):
                ...
        The out() method is bypassed. OscDataReceive has no audio signal.
        OscDataReceive has no `mul` and `add` attributes.
    >>> s = Server().boot()
    >>> s.start()
    >>> def pp(address, *args):
    ...     print(address)
    ...     print(args)
    >>> r = OscDataReceive(9900, "/data/test", pp)
    >>> # Send various types
    >>> a = OscDataSend("fissif", 9900, "/data/test")
    >>> msg = [3.14159, 1, "Hello", "world!", 2, 6.18]
    >>> a.send(msg)
    >>> # Send a blob
    >>> b = OscDataSend("b", 9900, "/data/test")
    >>> msg = [[chr(i) for i in range(10)]]
    >>> b.send(msg)
    >>> # Send a MIDI noteon on port 0
    >>> c = OscDataSend("m", 9900, "/data/test")
    >>> msg = [[0, 144, 60, 100]]
    >>> c.send(msg)
    """
    def __init__(self, port, address, function):
        assertOSCSupport(self)
        pyoArgsAssert(self, "IsC", port, address, function)
        PyoObject.__init__(self)
        self._port = port
        self._function = WeakMethod(function)
        self._address, lmax = convertArgsToLists(address)
        # self._address is linked with list at C level
        self._base_objs = [OscDataReceive_base(port, self._address, self._function)]
        self._init_play()
[docs]    def setMul(self, x):
        pass 
[docs]    def setAdd(self, x):
        pass 
[docs]    def out(self, chnl=0, inc=1, dur=0, delay=0):
        return self.play(dur, delay) 
[docs]    def getAddresses(self):
        """
        Returns the addresses managed by the object.
        """
        return self._address 
[docs]    def addAddress(self, path):
        """
        Adds new address(es) to the object's handler.
        :Args:
            path: string or list of strings
                New path(s) to receive from.
        """
        pyoArgsAssert(self, "s", path)
        path, lmax = convertArgsToLists(path)
        for p in path:
            if p not in self._address:
                self._address.append(p)
                self._base_objs[0].addAddress(p) 
[docs]    def delAddress(self, path):
        """
        Removes address(es) from the object's handler.
        :Args:
            path: string or list of strings
                Path(s) to remove.
        """
        pyoArgsAssert(self, "s", path)
        path, lmax = convertArgsToLists(path)
        for p in path:
            if p in self._address:
                index = self._address.index(p)
                self._base_objs[0].delAddress(index)
                self._address.remove(p)  
[docs]class OscListReceive(PyoObject):
    """
    Receives list of values over a network via the Open Sound Control protocol.
    Uses the OSC protocol to receive list of floating-point values from other
    softwares or other computers. The list are converted into audio streams.
    Get values at the beginning of each buffersize and fill buffers with them.
    :Parent: :py:class:`PyoObject`
    :Args:
        port: int
            Port on which values are received. Sender should output on
            the same port. Unlike OscSend object, there can be only one
            port per OscListReceive object. Available at initialization time
            only.
        address: string
            Address used on the port to identify values. Address is in
            the form of a Unix path (ex.: '/pitch').
        num: int, optional
            Length of the lists in input. The object will generate `num` audio
            streams per given address. Available at initialization time only.
            This value can't be a list. That means all addresses managed by an
            OscListReceive object are of the same length. Defaults to 8.
    .. note::
        Audio streams are accessed with the `address` string parameter.
        The user should call :
        OscReceive['/pitch'] to retreive list of streams named '/pitch'.
        The out() method is bypassed. OscReceive's signal can not be sent
        to audio outs.
    >>> s = Server().boot()
    >>> s.start()
    >>> # 8 oscillators
    >>> a = OscListReceive(port=10001, address=['/pitch', '/amp'], num=8)
    >>> b = Sine(freq=a['/pitch'], mul=a['/amp']).mix(2).out()
    """
    def __init__(self, port, address, num=8, mul=1, add=0):
        assertOSCSupport(self)
        pyoArgsAssert(self, "IsIOO", port, address, num, mul, add)
        PyoObject.__init__(self, mul, add)
        self._num = num
        self._op_duplicate = self._num
        address, mul, add, lmax = convertArgsToLists(address, mul, add)
        self._address = address
        self._mainReceiver = OscListReceiver_base(port, address, num)
        self._base_objs = [
            OscListReceive_base(self._mainReceiver, wrap(address, i), j, wrap(mul, i), wrap(add, i))
            for i in range(lmax)
            for j in range(self._num)
        ]
        self._init_play()
    def __getitem__(self, i):
        if type(i) in [bytes, str]:
            first = self._address.index(i) * self._num
            return self._base_objs[first : first + self._num]
        elif i < len(self._base_objs):
            first = i * self._num
            return self._base_objs[first : first + self._num]
        else:
            print("'i' too large!")
[docs]    def getAddresses(self):
        """
        Returns the addresses managed by the object.
        """
        return self._address 
[docs]    def addAddress(self, path, mul=1, add=0):
        """
        Adds new address(es) to the object's handler.
        :Args:
            path: string or list of strings
                New path(s) to receive from.
            mul: float or PyoObject
                Multiplication factor. Defaults to 1.
            add: float or PyoObject
                Addition factor. Defaults to 0.
        """
        pyoArgsAssert(self, "sOO", path, mul, add)
        path, lmax = convertArgsToLists(path)
        mul, add, lmax2 = convertArgsToLists(mul, add)
        for i, p in enumerate(path):
            if p not in self._address:
                self._mainReceiver.addAddress(p)
                self._address.append(p)
                self._base_objs.extend(
                    [
                        OscListReceive_base(self._mainReceiver, p, j, wrap(mul, i), wrap(add, i))
                        for j in range(self._num)
                    ]
                )
        self.play() 
[docs]    def delAddress(self, path):
        """
        Removes address(es) from the object's handler.
        :Args:
            path: string or list of strings
                Path(s) to remove.
        """
        pyoArgsAssert(self, "s", path)
        path, lmax = convertArgsToLists(path)
        self._mainReceiver.delAddress(path)
        indexes = [self._address.index(p) for p in path if p in self._address]
        for ind in reversed(indexes):
            self._address.pop(ind)
            first = ind * self._num
            for i in reversed(list(range(first, first + self._num))):
                obj = self._base_objs.pop(i) 
[docs]    def setInterpolation(self, x):
        """
        Activate/Deactivate interpolation. Activated by default.
        :Args:
            x: boolean
                True activates the interpolation, False deactivates it.
        """
        pyoArgsAssert(self, "B", x)
        [obj.setInterpolation(x) for obj in self._base_objs] 
[docs]    def setValue(self, path, value):
        """
        Sets value for a given address.
        :Args:
            path: string
                Address to which the value should be attributed.
            value: list of floats
                List of values to attribute to the given address.
        """
        pyoArgsAssert(self, "sl", path, value)
        path, lmax = convertArgsToLists(path)
        for i in range(lmax):
            p = wrap(path, i)
            if p in self._address:
                if type(value[0]) == list:
                    val = wrap(value, i)
                else:
                    val = value
                if len(val) == self._num:
                    self._mainReceiver.setValue(p, val)
                else:
                    print("Error: OscListReceive.setValue, value must be of the same length as the `num` attribute.")
            else:
                print('Error: OscListReceive.setValue, Illegal address "%s"' % p) 
[docs]    def get(self, identifier=None, all=False):
        """
        Return the first list of samples of the current buffer as floats.
        Can be used to convert audio stream to usable Python data.
        Address as string must be given to `identifier` to specify
        which stream to get value from.
        :Args:
            identifier: string
                Address string parameter identifying audio stream.
                Defaults to None, useful when `all` is True to
                retreive all streams values.
            all: boolean, optional
                If True, the first list of values of each object's stream
                will be returned as a list of lists. Otherwise, only the
                the list of the object's identifier will be returned as a
                list of floats. Defaults to False.
        """
        if not all:
            first = self._address.index(identifier) * self._num
            return [obj._getStream().getValue() for obj in self._base_objs[first : first + self._num]]
        else:
            outlist = []
            for add in self._address:
                first = self._address.index(add) * self._num
                l = [obj._getStream().getValue() for obj in self._base_objs[first : first + self._num]]
                outlist.append(l)
            return outlist 
[docs]    def play(self, dur=0, delay=0):
        self._mainReceiver.play(dur, delay)
        return PyoObject.play(self, dur, delay) 
[docs]    def out(self, chnl=0, inc=1, dur=0, delay=0):
        return self.play(dur, delay) 
[docs]    def stop(self, wait=0):
        self._mainReceiver.stop(wait)
        return PyoObject.stop(self, wait)