
    rh	+                        S SK Jr  / SQrS SKrS SKrS SKrS SKr " S S\R                  5      r " S S\	5      r
SS 0r " S	 S
5      r " S S5      r " S S\5      r " S S\5      r " S S5      r\S:X  a  S SKr\R&                  " 5         gg)    )annotations)defaultlistSingletonCounterRelativeCounterSlottedObjectMixinEqualSlottedObjectMixinFrozenObjectTimerNc                  >   ^  \ rS rSrSrU 4S jrS rS rS rSr	U =r
$ )r      a  
A counter that iterates from most common to least common
and can return new RelativeCounters that adjust for proportion or percentage.

>>> l = ['b', 'b', 'a', 'a', 'a', 'a', 'c', 'd', 'd', 'd'] + ['e'] * 10
>>> rc = common.RelativeCounter(l)
>>> for k in rc:
...     print(k, rc[k])
e 10
a 4
d 3
b 2
c 1

Ties are iterated according to which appeared first in the generated list.

>>> rcProportion = rc.asProportion()
>>> rcProportion['b']
0.1
>>> rcProportion['e']
0.5
>>> rcPercentage = rc.asPercentage()
>>> rcPercentage['b']
10.0
>>> rcPercentage['e']
50.0

>>> for k, perc in rcPercentage.items():
...     print(k, perc)
e 50.0
a 20.0
d 15.0
b 10.0
c 5.0
c              #  ^   >^ #    [        [        TT ]	  5       U 4S jSS9nU S h  vN   g  N7f)Nc                   > TU    $ N )xselfs    P/home/james-whalen/.local/lib/python3.13/site-packages/music21/common/objects.py<lambda>*RelativeCounter.__iter__.<locals>.<lambda>E   s	    d1g    T)keyreverse)sortedsuper__iter__)r   
sortedKeys	__class__s   ` r   r   RelativeCounter.__iter__D   s&     EG,.4EtT
s   !-+-c              #  .   #    U  H  nXU   4v   M     g 7fr   r   )r   ks     r   itemsRelativeCounter.itemsH   s     A!W* s   c                ~   ^  [        U 4S jT  5       5      n0 nT  H  nT U   U-  X#'   M     T R                  U5      nU$ )Nc              3  .   >#    U  H
  nTU   v   M     g 7fr   r   .0r   r   s     r   	<genexpr>/RelativeCounter.asProportion.<locals>.<genexpr>M        ,t!d1gt   sumr   r   selfLenoutDictynews   `    r   asProportionRelativeCounter.asProportionL   sG    ,t,,Aa7*GJ  nnW%
r   c                   ^  [        U 4S jT  5       5      n0 nT  H  nT U   S-  U-  X#'   M     T R                  U5      nU$ )Nc              3  .   >#    U  H
  nTU   v   M     g 7fr   r   r%   s     r   r'   /RelativeCounter.asPercentage.<locals>.<genexpr>V   r)   r*   d   r+   r-   s   `    r   asPercentageRelativeCounter.asPercentageU   sK    ,t,,Aa30GJ  nnW%
r   r   )__name__
__module____qualname____firstlineno____doc__r   r!   r2   r8   __static_attributes____classcell__r   s   @r   r   r      s"    "J r   r   c                  >   ^  \ rS rSrSrU 4S jrS rS rS rSr	U =r
$ )r   _   zm
Call a function for every time something is missing:

>>> a = common.defaultlist(lambda:True)
>>> a[5]
True
c                .   > [         TU ]  5         Xl        g r   )r   __init___fx)r   fxr   s     r   rE   defaultlist.__init__g   s    r   c                    [        U 5      U::  a1  U R                  U R                  5       5        [        U 5      U::  a  M0  g g r   )lenappendrF   r   indexs     r   _filldefaultlist._fillk   s.    $i5 KK
# $i5 r   c                R    U R                  U5        [        R                  XU5        g r   )rN   list__setitem__)r   rM   values      r   rR   defaultlist.__setitem__o   s    

5e,r   c                N    U R                  U5        [        R                  X5      $ r   )rN   rQ   __getitem__rL   s     r   rV   defaultlist.__getitem__s   s    

5,,r   )rF   )r:   r;   r<   r=   r>   rE   rN   rR   rV   r?   r@   rA   s   @r   r   r   _   s!    $-- -r   r   rS   c                  $    \ rS rSrSrS rS rSrg)r   {   a#  
A simple counter that can produce unique numbers (in ascending order)
regardless of how many instances exist.

Instantiate and then call it.

>>> sc = common.SingletonCounter()
>>> v0 = sc()
>>> v1 = sc()
>>> v1 > v0
True
>>> sc2 = common.SingletonCounter()
>>> v2 = sc2()
>>> v2 > v1
True
c                    g r   r   r   s    r   rE   SingletonCounter.__init__   s    r   c                :    [         S   n[         S==   S-  ss'   U$ )NrS      )_singletonCounter)r   posts     r   __call__SingletonCounter.__call__   s!     )'"a'"r   r   N)r:   r;   r<   r=   r>   rE   ra   r?   r   r   r   r   r   {   s     r   r   c                  :    \ rS rSr% SrSrS\S'   S rS rS r	Sr
g	)
r      aH  
Provides template for classes implementing slots allowing it to be pickled
properly, even if there are weakrefs in the slots, or it is subclassed
by something that does not define slots.

Only use SlottedObjectMixins for objects that we expect to make so many of
that memory storage and speed become an issue. Thus, unless you are Xenakis,
Glissdata is probably not the best example:

>>> import pickle
>>> class Glissdata(common.SlottedObjectMixin):
...     __slots__ = ('time', 'frequency')
>>> s = Glissdata()
>>> s.time = 0.125
>>> s.frequency = 440.0
>>> #_DOCS_SHOW out = pickle.dumps(s)
>>> #_DOCS_SHOW pickleLoad = pickle.loads(out)
>>> pickleLoad = s #_DOCS_HIDE -- cannot define classes for pickling in doctests
>>> pickleLoad.time, pickleLoad.frequency
(0.125, 440.0)

OMIT_FROM_DOCS

>>> class BadSubclass(Glissdata):
...     pass

>>> bsc = BadSubclass()
>>> bsc.amplitude = 2
>>> #_DOCS_SHOW out = pickle.dumps(bsc)
>>> #_DOCS_SHOW outLoad = pickle.loads(out)
>>> outLoad = bsc #_DOCS_HIDE -- cannot define classes for pickling in doctests
>>> outLoad.amplitude
2

This is in OMIT
r   tuple[str, ...]	__slots__c                &   [        U SS 5      b  [        U S5      R                  5       nO0 nU R                  5       nU HO  n[        XS 5      n[        U[        R
                  5      (       a  U" 5       n[        SU < SU S3S-   5        XAU'   MQ     U$ )N__dict__z#Warning: uncaught weakref found in z - z, zwill not be wrapped again)getattrcopy_getSlotsRecursive
isinstanceweakrefReferenceTypeprint)r   stateslotsslotsValues        r   __getstate__SlottedObjectMixin.__getstate__   s    4T*6D*-224EE'')DT.F&'"7"788;D83tfBO34 5 $K  r   c                N    UR                  5        H  u  p#[        XU5        M     g r   )r!   setattr)r   rp   rr   rS   s       r   __setstate__SlottedObjectMixin.__setstate__   s     ;;=KDD& )r   c                    [        5       nU R                  R                  5        H  nUR                  [	        USS5      5        M!     U$ )a  
Find all slots recursively.

A private attribute so as not to change the contents of inheriting
objects private interfaces:

>>> b = beam.Beam()
>>> sSet = b._getSlotsRecursive()

sSet is a set -- independent order.  Thus, for the doctest
we need to preserve the order:

>>> sorted(list(sSet))
['_editorial', '_style', 'direction', 'id', 'independentAngle', 'number', 'type']

When a normal Beam won't cut it:

>>> class FunkyBeam(beam.Beam):
...     __slots__ = ('funkiness', 'groovability')

>>> fb = FunkyBeam()
>>> sSet = fb._getSlotsRecursive()
>>> sorted(list(sSet))
['_editorial', '_style', 'direction', 'funkiness', 'groovability',
    'id', 'independentAngle', 'number', 'type']
rf   r   )setr   mroupdateri   )r   rq   clss      r   rk   %SlottedObjectMixin._getSlotsRecursive   s<    6 >>%%'CLLk267 (r   N)r:   r;   r<   r=   r>   rf   __annotations__rt   rx   rk   r?   r   r   r   r   r      s#    #N "$I#'r   r   c                  4    \ rS rSr% SrSrS\S'   S rS rSr	g)	r      z
Same as above, but __eq__ and __ne__ functions are defined based on the slots.

Slots are the only things compared, so do not mix with a __dict__ based object.

The equal comparison ignores differences in .id
r   re   rf   c                    [        U 5      [        U5      La  gU R                  5        H%  nUS:X  a  M  [        X5      [        X5      :w  d  M%    g   g)NFidT)typerk   ri   )r   otherthisSlots      r   __eq__EqualSlottedObjectMixin.__eq__  sM    :T%[(//1H4t&'%*BB	 2
 r   c                    X:X  + $ )zC
Defining __ne__ explicitly so that it inherits the same as __eq__
r   )r   r   s     r   __ne__EqualSlottedObjectMixin.__ne__  s     ""r   N)
r:   r;   r<   r=   r>   rf   r   r   r   r?   r   r   r   r   r      s     "$I##r   r   c                  |   ^  \ rS rSr% SrS\S'   SSS jjrSU 4S jjrSU 4S jjrU 4S jr	U 4S	 jr
SS
 jrSrU =r$ )r	   i  r   re   rf   c                   US:X  a  g[        U SS5      (       d  g[        R                  " 5        Hx  nUR                  R                  R
                  S;   d  M)  SUR                  R                  ;   d  ME  UR                  R                  S   R                  U R                  :X  d  Mx    g   [        SU R                  R                   S35      e)Nr   Tfrozen)rE   __new__rx   r   zThis z instance is immutable.)
ri   inspectstackframef_codeco_namef_localsr   	TypeErrorr:   )r   r   sts      r   _check_initFrozenObject._check_init  s    +tXt,,--/B''+RR"(("3"33))&1;;t~~M	 "
 % 7 788OPQQr   c                F   > U R                  U5        [        TU ]	  X5        g r   )r   r   __setattr__r   r   rS   r   s      r   r   FrozenObject.__setattr__$  s    C'r   c                F   > U R                  U5        [        TU ]	  U5        g r   )r   r   __delattr__r   r   r   s     r   r   FrozenObject.__delattr__(  s    C r   c                   > [        [        5       S5      (       a  U R                  5         [        TU ]  X5        [	        U R
                   S35      e)NrR    object is not subscriptable)hasattrr   r   rR   r   r   r   s      r   rR   FrozenObject.__setitem__,  sC    57M**G+4>>**FGHHr   c                   > [        [        5       S5      (       a  U R                  5         [        TU ]  U5        [	        U R
                   S35      e)N__delitem__r   )r   r   r   r   r   r   r   s     r   r   FrozenObject.__delitem__2  sC    57M**G$4>>**FGHHr   c                    / nU R                  5        H  nUR                  U[        X5      45        M!     [        [	        U5      5      $ r   )rk   rK   ri   hashtuple)r   outss      r   __hash__FrozenObject.__hash__8  s?    ((*AJJ74+,- +E#Jr   r   )returnbool)r   str)r   int)r:   r;   r<   r=   rf   r   r   r   r   rR   r   r   r?   r@   rA   s   @r   r	   r	     s3    !#I#R(!II   r   r	   c                  <    \ rS rSrSrS rS rS rS rS r	S r
S	rg
)r
   i@  a  
An object for timing. Call it to get the current time since starting.

>>> timer = common.Timer()
>>> now = timer()
>>> import time  #_DOCS_HIDE
>>> time.sleep(0.01)  #_DOCS_HIDE  -- some systems are extremely fast or have wide deltas
>>> nowNow = timer()
>>> nowNow > now
True

Call `stop` to stop it. Calling `start` again will reset the number

>>> timer.stop()
>>> stopTime = timer()
>>> stopNow = timer()
>>> stopTime == stopNow
True

All this had better take less than one second!

>>> stopTime < 1
True
c                T    [         R                   " 5       U l        SU l        S U l        g Nr   )time_tStart_tDif_tStopr[   s    r   rE   Timer.__init__Y  s    yy{
r   c                T    [         R                   " 5       U l        SU l        SU l        g)z]
Explicit start method; will clear previous values.

Start always happens on initialization.
Nr   )r   r   r   r   r[   s    r   startTimer.start_  s     yy{
r   c                t    [         R                   " 5       U l        U R                  U R                  -
  U l        g r   )r   r   r   r   r[   s    r   stop
Timer.stopi  s#    iik[[4<</
r   c                .    S U l         SU l        S U l        g r   )r   r   r   r[   s    r   clearTimer.clearm  s    
r   c                    U R                   c$  [        R                  " 5       U R                  -
  nU$ U R                  nU$ )z4
Reports current time or, if stopped, stopped time.
)r   r   r   r   r   timeDifferences     r   ra   Timer.__call__r  s;    
 ;;!YY[4<<7N  "ZZNr   c                    U R                   c#  [        R                  " 5       U R                  -
  nOU R                  n[	        [        US5      5      $ )N   )r   r   r   r   r   roundr   s     r   __str__Timer.__str__}  s;    ;;!YY[4<<7N!ZZN5+,,r   )r   r   r   N)r:   r;   r<   r=   r>   rE   r   r   r   ra   r   r?   r   r   r   r
   r
   @  s%    00
	-r   r
   __main__)
__future__r   __all__collectionsr   r   rm   Counterr   rQ   r   r_   r   r   r   r	   r
   r:   music21mainTestr   r   r   <module>r      s    #    >k)) >B-$ -2 aL  8] ]@#0 #:( * ( XB- B-J z r   