divert(-1)

 libSLD.m4                    Draft macros for single-line diagram elements

* Circuit_macros Version 10.8, copyright (c) 2025 J. D. Aplevich under     *
* the LaTeX Project Public Licence in file Licence.txt. The files of       *
* this distribution may be redistributed or modified provided that this    *
* copyright notice is included and provided that modifications are clearly *
* marked to distinguish them from this distribution.  There is no warranty *
* whatsoever for these files.                                              *

==========================================================================

# These *DRAFT* definitions are for single-line diagram (SLD) elements
# with or without attached circuit breakers or slashes.  Many other elements
# applicable to SLD drawings are already in libcct.m4.  The contributions
# and suggestions of Benjamin Vilmann and Peter Jan Randewijk are
# acknowledged with thanks.

# The line
#   include(libSLD.m4)
# loads these definitions for use in a diagram.

# Notes for 1-terminal SLD elements:
# Argument 1 is normally the linespec of the stem to set the direction
#   and length.; e.g., D1: sl_disk draws a default disk with stem named D1.
# For a 0-length stem (which has undefined direction):
#   arg1 can also be U, D, L, R (for up, down, left, right),
#   or a number to set the direction in degrees, optionally followed by
#   `at position' to set the position (Here by default).
#   Zero-length stem examples: sl_box(U), sl_box(45 at Here+(1,0))
# Argument 2 contains semicolon (;)-separated key-value attributes
#   of the element head as applicable: e.g., name=Carol; text="Stop"; lgth=expr
# If argument 3 is blank then a plain stem is drawn as described below.
# The element body (head) can be named with name= . The default name is Head.
# The head is overlaid with or contained in a [] block.

# Notes for 2-terminal SLD elements:
# These obey the normal Circuit_macro two-terminal conventions.
# They can be labelled using rlabel() or llabel() as well as directly.
# Argument 2 contains key-value pairs to customize the element body,
#   e.g., name=Name; text="text"; wdth=expr; ...
# Except for sl_drawout and sl_breaker which do not have series breakers
# or slashes, nonblank arguments 3 and 4 put a breaker slash symbol
# in the input and output respectively as described below.

# Notes for attached breakers and slashes:
# Nonblank arguments 3 and 4 of the two-terminal elements and argument 3 of
#   the 1-terminal elements specify a breaker in the input, output,
#   and stem respectivlely.  A non-blank argument is C for a default
#   closed breaker in the stem, O for an open breaker, or X, /, or \ to
#   put these symbols in the box; if the argument begins with S: or Sn:
#   (where n is an integer) then an n-line slash symbol is drawn rather
#   than a breaker; otherwise or in addition, key-value pairs specify
#   details of the object.  The separation of the breaker or slash
#   from the element body or head is given by parameter sl_breakersep_
#   or key sep=expr.  If this key is in the body or head keys then it
#   applies to both 2-terminal stem lines; it can also be given for each
#   stem individually.

# Notes for composite elements within a [ ] block:
#   The current transformer macro sl_ct has internal labels defined as
#   appropriate to the element.  The macro sl_transormer3 has 3 principal
#   termianls and other defined points.  Macro sl_busbar has internal
#   labels P1 to Pnp, Start, and End.

define(`sldlib_')
ifdef(`libcct_',,`include(libcct.m4)divert(-1)')

# Default size parameters.  These can be redefined in a diagram source.

define(`sl_breakersize_',`dimen_*3/16')  # breaker box size
define(`sl_breakersep_',`dimen_/2')      # breaker separation from body
define(`sl_ttboxlen_',`dimen_*3/4')      # inline box length
define(`sl_ttboxwid_',`dimen_*3/4')      # inline box width
define(`sl_sboxlen_',`dimen_*2/3')       # stem box length
define(`sl_sboxwid_',`dimen_*2/3')       # stem box wid
define(`sl_diskdia_',`dimen_*2/3')       # sl_disk diam
define(`sl_chevronsiz_',`dimen_/4')      # sl_drawout (chevron) size
define(`sl_loadwid_',`dimen_*0.32')      # load width
define(`sl_loadlen_',`dimen_*0.45')      # load length
define(`sl_transcale_',1)                # transformer body scale factor
define(`sl_busthick_',linethick*2)       # sl_bus line thickness
define(`sl_busindent_',`min(dimen_/5,rp_len/5)') # busbar end indent

# One-terminal elements ###################################################

                           `sl_disk( stem linespec, keys, breaker or Sn:slash)
                                keys: name=Name;
                                      text="text";
                                      diam=expr;
                                      circle=other circle attributes; eg shade'
                               `default breaker name Br'
define(`sl_disk',
`sl_eleminit_(`$1')
pushkeys_(`$2',`name:Head:N; circle::N; text::N; diam:sl_diskdia_')dnl
ifelse(`$3',,
 `m4name: circle diam m4diam \
    at last line.end + vec_(m4diam/2,0) m4circle m4text
  [ box invis wid_ m4diam ht_ m4diam ] at last circle
  line from last line.end  to last line.start',
 `m4_one(`sl_disk',$@)') dnl
popdef(`m4name', `m4circle', `m4text', `m4diam') ')

                           `sl_box( stem linespec, keys, breaker or Sn:slash)
                                keys: name=Name; lgth=expr; wdth=expr;
                                      text="text";
                                      box= box attributes; (e.g. shade "red")'
                               `default breaker name Br'
define(`sl_box',
`sl_eleminit_(`$1')
pushkeys_(`$2',`name:Head:N; wdth:sl_sboxwid_; lgth:sl_sboxlen_; box::N;
  text::N')dnl
ifelse(`$3',,
 `line from last line.end to last line.start
  { m4name: [S:Here; lbox(m4lgth,m4wdth,m4box)] \
      with .S at last line.start }
  ifelse(m4text,,,`{m4text at last []}')',
 `m4_one(`sl_box',$@)') dnl
popdef(`m4name', `m4wdth', `m4lgth', `m4box', `m4text') ')

                           `sl_grid( stem linespec, keys, breaker or Sn:slash)
                                keys: name=Name; lgth=expr; wdth=expr;'
                               `default breaker name Br'
define(`sl_grid',
`sl_eleminit_(`$1')
pushkeys_(`$2',`name:Head:N; wdth:sl_sboxwid_; lgth:sl_sboxlen_; box::N')dnl
ifelse(`$3',,
 `line from last line.end to last line.start
  { m4name: [ S:Here
    { lbox(m4lgth,m4wdth) }
    { line to rvec_(m4lgth/2, m4wdth/2)
      line to rvec_(m4lgth/2,-m4wdth/2) }
    { line to rvec_(m4lgth/2,-m4wdth/2)
      line to rvec_(m4lgth/2, m4wdth/2) }
    { line from rvec_(0,m4wdth/2) to rvec_(m4lgth,-m4wdth/2) }
    line from rvec_(0,-m4wdth/2) to rvec_(m4lgth,m4wdth/2)
    ] with .S at last line.start } ',
 `m4_one(`sl_grid',$@)') dnl
popdef(`m4name', `m4wdth', `m4lgth', `m4box') ')

                           `sl_load( stem linespec, keys, breaker or Sn:slash)
                                keys: name=Name; lgth=expr; wdth=expr;
                                      head= arrowhead attributes;
                                default breaker name is Br'
define(`sl_load',
`sl_eleminit_(`$1')
pushkeys_(`$2',`name:Head:N; wdth:sl_loadwid_; lgth:sl_loadlen_; head::N')dnl
ifelse(`$3',,
 `line from last line.end to last line.start
  { m4name: [S:Here; line to rvec_(0,m4wdth/2) \
      then to rvec_(m4lgth,0) then to rvec_(0,-m4wdth/2) \
      then to Here m4head ] with .S at last line.start } ',
 `m4_one(`sl_load',$@)') dnl
popdef(`m4name', `m4wdth', `m4lgth', `m4head') ')

                           `sl_meterbox( stem linespec, keys, breaker or Sn:slash)
                                 keys: sl_box attributes
                                default breaker name is Br'
define(`sl_meterbox',
`sl_eleminit_(`$1')
pushkeys_(`$2',`name:Head:N; wdth:sl_ttboxwid_; lgth:sl_ttboxlen_; box::N;
  text::N')dnl
ifelse(`$3',,
 `line from last line.end to last line.start
  {m4name: [S:Here;
    { B: rotbox( m4lgth, m4wdth, m4box ) with .W at S }
    a = rp_ang*rtod_
    if (abs(a-90) < 45) || (abs(a-180) < 45) || (abs(a+180) < 45) then {
      C: rvec_(m4lgth*2/5,0)
      line from rvec_(m4lgth*4/5,m4wdth/2) to rvec_(m4lgth*4/5,-m4wdth/2) } \
    else { C: rvec_(m4lgth*3/5,0)
       line from rvec_(m4lgth/5,m4wdth/2) to rvec_(m4lgth/5,-m4wdth/2) }
    ifelse(m4text,,,`m4text at C')
    ] with .S at last line.start}',
 `m4_one(`sl_meterbox',$@)') dnl
popdef(`m4name', `m4wdth', `m4lgth', `m4box', `m4text') ')

                           `sl_generator( stem linespec, keys,
                                breaker or Sn:slash)
                                keys:
                                  name= head name;
                                  type=AC|WT|BS|StatG|PV|Y|Delta;
                                  (type PV): sl_box keys name, lgth, wdth,
                                    text,box;
                                  (otherwise): sl_disk body keys name,
                                    text, diam, circle;
                                default breaker name is Br
                                Arg5 can be Y;name=Name with default
                                  name headname_Y,
                                  or Delta;name=Name with
                                  default name headname_D'
define(`sl_generator',
`pushkeys_(`$2',`type:AC:N; name:Head:N; lgth:sl_sboxlen_; wdth:m4lgth/2;
  box::N; text::N')dnl
ifelse(m4type,PV,
 `sl_box(`$1',wdth=m4lgth*0.6;`$2',`$3')
  {line from m4name.S+vec_(0,m4wdth*0.45) \
   to m4name.S+vec_(m4wdth/2,0) \
   then to m4name.S+vec_(0,-m4wdth*0.45)}',
`sl_disk($@)
m4lcd = last circle.diam
{ ifinstr(m4type,AC,
   `{ ACsymbol(at last circle,,,R) }',
  m4type,Y,
   `pushkey_(`$3',name,m4name`'_Y,N)dnl
    {m4name:Ysymbol(with .N at last circle,size=m4lcd/4) } popdef(`m4name')',
  m4type,Delta,
   `pushkey_(`$3',name,m4name`'_Y,N)dnl
    {m4name:Deltasymbol(with .N at last circle,size=m4lcd/4) }dnl
      popdef(`m4name')',
  m4type,WT,
   `{ Windturbine(with .N at last circle.c,size=m4lcd*0.9) }',
  m4type,BS,
   `{ m4angt = rp_ang; battery(up_ m4lcd*3/4 with .c at m4name.c)
      cross(at m4name.c+(m4lcd/4,m4lcd/5),m4lcd/6)
      line right_ m4lcd/6 with .c at m4name.c+(m4lcd/4,-m4lcd/6)
      point_(m4angt) }',
  m4type,StatG,
   `{ m4angt= rp_ang; pushdef(`dimen_',m4lcd*2)
      diode(right_ dimen_/6 with .c at m4name.c,DK)
      popdef(`dimen_') point_(m4angt)}')
  [ box invis wid_ m4lcd ht_ m4lcd ] at last circle }') dnl
popdef(`m4type',`m4name', `m4lgth', `m4wdth', `m4box', `m4text')')

                           `sl_syncmeter( stem linespec, keys, breaker or Sn:slash)'
                               `default breaker name Br'
define(`sl_syncmeter',`sl_disk($@)
{ Syncsymb(at last circle)
  m4lcd = last circle.diam
  [ box invis wid_ m4lcd ht_ m4lcd ] at last circle } popdef(`m4thype')')

                           `sl_lamp( stem linespec, keys, breaker or Sn:slash)'
                               `default breaker name Br'
define(`sl_lamp',`sl_disk($@)
{ line from last circle.ne to last circle.sw
  line from last circle.nw to last circle.se
  m4lcd = last circle.diam
  [ box invis wid_ m4lcd ht_ m4lcd ] at last circle } ')

# One-terminal utilities ##################################################

                           `Syncsymb(at position, rad)
                               Symbol for sync meter'
define(`Syncsymb',`[ define(`m4ssrad',`ifelse(`$2',,(dimen_/4),`($2)')')dnl
Origin: Here
{arc <-> ht arrowht/2 wid arrowwid*2/3 \
  from Rect_(m4ssrad,30) to Rect_(m4ssrad,150) with .c at Here}
line from (0,m4ssrad) to (0,-m4ssrad/2)
`$3' ] with .Origin ifelse(`$1',,`at Here',`$1')')


                       `Windturbine(at position, keys, U|D|L|R|degrees)
                         keys: size=expr; color="color string"
                         Arg3: drawing direction (default: Right)'
define(`Windturbine',`[
pushkeys_(`$2',`size:sl_diskdia_:;color:"black":N')dnl
setdir_(`$3',R)
C: Here; N: C
circle diam m4size/18 at C colored m4color
for t = 30 to 360 by 120 do {
tr = t*dtor_
spline 0.55 thick 0.2 shaded m4color outlined m4color \
 from M4wt(4,-0.8) to M4wt(1.810794,-0.921724)
Loopover_(`M4LO',`continue to M4LO',
 M4wt(1.135237,-0.069654), M4wt(1.777783,1.339647),
 M4wt(4.198063,2.031772), M4wt(5.033839,2.04),
 M4wt(5.598904,2.029888), M4wt(17.372737,0.390951),
 M4wt(18.164674,0.024408), M4wt(17.639221,-0.537401), M4wt(4,-0.8)) }
`$4'; resetdir_ popdef(`m4color',`m4size') ] ifelse(`$1',,`at Here',`$1')')
define(`M4wt',`(vec_(rot_(`$1',`$2',tr))*m4size/36)')

                           `sl_eleminit_(linespec or (for zero length)
                               U|D|L|R|number [at location])'
define(`sl_eleminit_',
`ifelse(regexp(`$1',^ *[UDLR0123456789]),-1,
 `eleminit_(`$1',dimen_)',
 `pushdef(`M4pos',`ifinstr(`$1',` at ',`patsubst(`$1',^.* at *)')')dnl
  ifelse(M4pos,,,`move to M4pos;') setdir_(patsubst(`$1',` at.*'))
 line invis from Here to Here popdef(`M4pos')')')

# Two-terminal elements ###################################################

                           `sl_transformer(linespec,
                              keys,
                              input breaker keys,
                              output breaker keys
                              type S input circle inner object,
                              type S output circle inner object)
                            keys:
                              name=body name (default Body);
                              scale=expr; (body size factor: default 1)
                              type=I|S|A[R]
                              (type=I) cycles=n; (default 4)
                                       core= A[ir]|M[n]|P[n]|K[n]
                                             n=integer (default 2 lines)
                              (type=S) body=circle attributes;
                                       Note arg5 and arg6
                              (type=A) body=circle attributes ;
                                       type AR means right orientation
                              (breaker default names are BrI, BrO)
                            Args 3 and 4 specify input and output stem objects
                              (breakers or slash symbols);
                            Args 5 and 6 for the input and output circles are
                              name=symbol name; (optional)
                              Y for a Y symbol
                              YN for a Y symbol with ground
                                Default name for a Y in arg5 is Body_Y1; in
                                  arg6 it is Body_Y2
                              Delta for a Delta symbol
                                Default name for arg5 is Body_D1; in
                                  arg6 it is Body_D2
                              otherwise other customization commands in
                                a {} pair.  '
define(`sl_transformer',
`pushkeys_(`$2',name:Body:N; type:I:N; cycles:4:N; core::N; body::N;
  scale:sl_transcale_; )define(`m4scf',(m4scale*1.3))dnl
ifinstr(m4type,S,
 `define(`m4blgth',sourcerad_*3.5*m4scf)',
 m4type,A,
 `define(`m4blgth',sourcerad_*3*m4scf)',
 `ifelse(ifinstr(m4core,M,T,m4core,P,T,m4core,K,T),T,
   `define(`m4nL',ifelse(len(m4core),1,2,substr(m4core,1)))dnl
    define(`m4blgth',(dimen_/8 + dimen_/12 + (m4nL-1)*dimen_/16)*m4scf)',dnl
   `define(`m4blgth',(dimen_/8 + dimen_/12)*m4scf)')')dnl
ifelse(`$3'`$4',,
 `ifinstr(m4type,S,
   `pushdef(`sourcerad_',sourcerad_*m4scf)dnl
    source(`$1',G,,,m4body,m4name); C: last [].c popdef(`sourcerad_')
    ifelse(`$5',,,`m4DY(`$5',, 1,m4name.C1,sl_diskdia_*0.18,rp_ang*rtod_)')
    ifelse(`$6',,,`m4DY(`$6',-,2,m4name.C2,sl_diskdia_*0.18,rp_ang*rtod_)')',
  m4type,A,
   `eleminit_(`$1'); C: last line.c
    { line to rvec_(rp_len/2-m4blgth/2,0); round
      pushdef(`m4R',ifinstr(m4type,R,-))dnl
     {m4name: [ C2: circle rad m4blgth/3 m4body at (0,0)
        Arc: arc ifinstr(m4type,R,c)`'cw \
          from C2-vec_(m4blgth/3*2,0) to C2+vec_(0,m4R`'m4blgth/3) \
           with .c at C2-vec_(0,m4R`'m4blgth/2) ] \
        with .C2 at rvec_(m4blgth/3*2,0)} popdef(`m4R')
    line from rvec_(m4blgth,0) to rvec_(rp_len/2+m4blgth/2,0) }
    line invis to rvec_(rp_len,0)',
   `eleminit_(`$1'); C: last line.c; m4atmp = rp_ang; m4slen = rp_len
    { line to rvec_((m4slen-m4blgth)/2,0)
      {m4name: [ linewid = linewid*m4scf
        {L1: inductor(to vec_(0,-m4cycles*dimen_/8),,m4cycles,m4core)}
        point_(m4atmp)
        L2: inductor(from vec_(m4blgth/m4scf,-m4cycles*dimen_/8) \
              to vec_(m4blgth/m4scf,0),,m4cycles)
        point_(m4atmp) ] with .L1.c at Here}
    line from rvec_(m4blgth,0) to rvec_((m4slen+m4blgth)/2,0) }
    line invis to rvec_(rp_len,0)')',
 `m4_two(`sl_transformer',m4blgth,$@)')
popdef(`m4name', `m4type', `m4cycles', `m4core', `m4body', `m4scale') ')
define(`m4trground',
`if pmod(drawdir_(`$1'),180)==0 then {
  line from `$2' to `$2'+(0,-sourcerad_*5/4*m4scf); ground(,T) } \
else {
  line from `$2' to `$2'+(sourcerad_*3/2*m4scf,0); corner; ground } ')

                          `sl_transformer3(linespec,
                             keys,
                             breaker1:breaker2:breaker3,
                             symbol1:symbol2:symbol3)
                           keys:
                             name=Name;
                             type=S|C; (default S)
                             scale=expr;
                             direct=L|R;
                             body=circle attributes; '
define(`sl_transformer3',
`[ pushkeys_(`$2',name:Body:N; scale:sl_transcale_; direct:L:N; body::N;
    type:S:N )dnl
  ifelse(m4type,S,
   `m4trAr(`m4Br',patsubst(`$3',:,`,'))m4trAr(`m4Sy',patsubst(`$4',:,`,'))dnl
    pushdef(`sl_breakersep_',sl_breakersep_*2/3)dnl
    sl_transformer($1,type=S;name=m4name;scale=m4scale;$2,
      m4Br1,m4Br2,m4Sy1,m4Sy2)
    Tstart: last line.start; C1: m4name.C1; m4atmp = rp_ang
    Tend: last line.end;   C2: m4name.C2
    define(`m4R',`ifinstr(m4direct,R,-)')dnl
    C3: m4name.C1+vec_(Rect_(m4name.C1.rad*3/2,m4R`'60))
    move to C3+vec_(0,m4R`'m4name.C1.rad)
    Tc: rvec_(0,m4R`'m4name.C1.diam`'ifelse(m4Br3,,/3))
    sl_disk(from Tc to Here,
      diam=m4name.C1.diam;name=C3;circle=m4body;`$2',m4Br3)
    m4DY(m4Sy3,,3,C3,C3.diam*0.18,rp_ang*rtod_); popdef(`sl_breakersep_')
    point_(m4atmp)',
   `pushdef(`m4rad',`sl_diskdia_/6*m4scale')dnl
    eleminit_(`$1',m4rad*8)
    Tstart: Here
    line to rvec_(rp_len/2-m4rad*2,0); round
    Arc: arc cw to rvec_(m4rad*4,0) with .c at rvec_(m4rad*2,0)
    round; line to rvec_(rp_len/2-m4rad*2,0)
    Tend: Here
    C: circle rad m4rad at Arc.c
    line from C+vec_(0,-m4rad) to C+vec_(0,-m4rad*3)
    Tc: Here popdef(`m4rad') ')
  popdef(`m4name',`m4scale',`m4direct',`m4body',`m4type') ]')
define(`m4trAr',`define(`$1'`1',$2)define(`$1'`2',$3)define(`$1'`3',$4)')

                          `m4DY(symbol,neg,number,loc,size,angle)
                           Insert Delta or Y symbol'
define(`m4DY',
`pushkey_(`$1',name,m4name`'_`'ifinstr(`$1',Delta,D,Y)`$3',N)dnl
define(`m4ang',``$6'`'ifelse(`$2',-,+180)')dnl
define(`m4sn',m4name)popdef(`m4name')dnl
ifinstr(`$1',Delta,
 `{m4sn: Deltasymbol(with .N at `$4',size=`$5',m4ang-180)} ',
`$1',Y,
 `{m4sn:     Ysymbol(with .N at `$4',size=`$5',m4ang)}
  ifinstr(`$1',N,`{m4trground(m4ang,m4sn.N)}') ',
`{`$1'}')')

                           `Two-terminal box'
                           `sl_ttbox(linespec,keys,breaker keys,breaker keys)
                            keys= lgth=expr; wdth=expr; box=attributes;
                              supp=additional rotbox commands;
                              name=body name (default Body);
                              text="text";
                              (breaker default names BrI, BrO)'
define(`sl_ttbox',
`pushkeys_(`$2',`lgth:sl_ttboxlen_; wdth:sl_ttboxwid_;
  name:Body:N; box::N; text::N; supp::N')dnl
ifelse(`$3'`$4',,
 `eleminit_(`$1')
  {line to rvec_((rp_len-m4lgth)/2,0)
    {m4name: rotbox(m4lgth,m4wdth,m4box,,m4supp) \
      with .W at Here }
    ifelse(m4text,,,`{m4text at rvec_(m4lgth/2,0)};') dnl
   line from rvec_(m4lgth,0) to rvec_((rp_len+m4lgth)/2,0)}
  line invis to rvec_(rp_len,0) ',
 `m4_two(`sl_ttbox',m4lgth,$@)') dnl
popdef(`m4lgth', `m4wdth', `m4name', `m4box', `m4text', `m4supp') ')

                           `sl_rectifier(ttbox args)'
define(`sl_rectifier',
`pushkeys_(`$2',`lgth:sl_ttboxlen_; wdth:sl_ttboxwid_; name:Body:N')dnl
sl_ttbox($@)
{ line from m4name.ne to m4name.sw
  AC: ACsymbol(at m4name.c+(-m4lgth/6, m4wdth/4),,,R)
  DC: DCsymbol(at m4name.c+( m4lgth/6,-m4wdth/4),,,R) } dnl
popdef(`m4lgth',`m4wdth',`m4name') ')

                           `sl_inverter(ttbox args)'
define(`sl_inverter',
`pushkeys_(`$2',`lgth:sl_ttboxlen_; wdth:sl_ttboxwid_; name:Body:N')dnl
sl_ttbox($@)
{ line from m4name.ne to m4name.sw
  DC: DCsymbol(at m4name.c+(-m4lgth/6, m4wdth/4),,,R)
  AC: ACsymbol(at m4name.c+( m4lgth/6,-m4wdth/4),,,R) } dnl
popdef(`m4lgth',`m4wdth',`m4name') ')

                           `m4_one( `elementname',
                              linespec (placeholder not used),
                              body keys,
                              breaker keys or S[n]: keys )'
                           `Draw the breaker slash in the element stem'
define(`m4_one',
`pushkeys_(`$4',lgth:sl_breakersize_; sep:-1)define(`m4s_p',`ifelse(m4sep,(-1),
 `pushkey_(`$3',sep,sl_breakersep_)`'m4sep`'popdef(`m4sep')',m4sep)')dnl
M4end: last line.end
{ifelse(regexp(`$4',^ *S),-1,
 `ifelse(`$4',,,
   `line to last line.end-vec_(m4lgth+m4s_p,0)
    pushdef(`m4bri',`ifelse(`$4',C,,`$4',O,`box=fill_(0)')')dnl
    sl_breaker(to rvec_(m4lgth,0),`$4';m4bri;name=Br) popdef(`m4bri')')
  $1(to M4end,`$3')',
 `{ $1(to M4end,`$3') }
  ifelse(`$4',,,
   `define(`m4in',`ifelse(`$4',S,1,`$4',S:,1,
     `patsubst(`$4',.*S\([0-9][0-9]*\):?.*,\1)')')dnl
    define(`m4inkey',`patsubst(`$4',^ *S[0-9]*:?)') pushkey_(m4inkey,name,SL,N)
    sl_slash(at M4end-vec_(m4lgth+m4s_p*2/3,0),
      m4inkey;name=m4name,m4in:) popdef(`m4name')')')}
  popdef(`m4lgth',`m4sep')')

                           `sl_breaker(linespec, type=[A|C][D]; ttbox keys)
                              C is for curved breaker
                              D is for sl_drawout with two chevrons
                              Default body name is Br'
define(`sl_breaker',
`pushkeys_(`$2',`lgth:sl_breakersize_; wdth:sl_breakersize_;
  name:Br:N; type:A:N')dnl
ifinstr(ifinstr(m4type,CD,T,m4type,DC,T),T,
 `m4ch_two(`cbreaker',`$1',,,m4name)',
m4type,C,
 `cbreaker(`$1',,,m4name)',
m4type,D,
 `m4ch_two(`sl_ttbox',`$1',
    lgth=m4lgth;wdth=m4wdth;`$2'ifinstr(`$2',name=,;name=m4name))',
`$2',X;,
 `sl_ttbox(`$1',supp=line from NW to SE
  line from NE to SW;lgth=m4lgth;wdth=m4wdth;`$2';name=m4name)',
`$2',/;,
 `sl_ttbox(`$1',
  supp=line from SE to NW;lgth=m4lgth;wdth=m4wdth;`$2';name=m4name)',
`$2',\;,
 `sl_ttbox(`$1',
  supp=line from NE to SW;lgth=m4lgth;wdth=m4wdth;`$2';name=m4name)',
`sl_ttbox(`$1',lgth=m4lgth;wdth=m4wdth;`$2')' ) dnl
popdef(`m4lgth', `m4wdth', `m4name', `m4type') ')

                           `sl_reactor(linespec,keys,breaker keys,breaker keys)
                            keys=
                              name=Name (body name, default Body);
                              diam=expr;
                            Default breakers BrI, BrO'
define(`sl_reactor',
`pushkeys_(`$2',`diam:sl_diskdia_*2/3;name:Body:N')dnl
ifelse(`$3'`$4',,
 `eleminit_(`$1')
  { line to rvec_(rp_len/2,0) then to rvec_(rp_len/2,-m4diam/2); round
    arc rad m4diam/2 cw from Here to rvec_(m4diam/2,m4diam/2) \
      with .c at rvec_(0,m4diam/2); round
    line to rvec_(rp_len/2-m4diam/2,0) }
  {m4name: [ box invis ht m4diam wid m4diam ] at rvec_(rp_len/2,0)}
  line invis to rvec_(rp_len,0) ',
 `m4_two(`sl_reactor',m4diam,$@)') dnl
popdef(`m4diam',`m4name') ')

                           `sl_drawout(linespec, keys, R)
                            Drawout (i.e. plugin) chevron element;
                              keys:  type=T; (truncated leads)
                                     lgth=expr; (body size)
                                     wdth=expr;
                                     name=Name; (body name)
                                     line= line attributes (e.g. thick 2)
                              arg3=R reverse direction'
define(`sl_drawout',
`pushkeys_(`$2',`lgth:sl_chevronsiz_; wdth:sl_chevronsiz_; type::N;
  name:Body:N; line::N')dnl
eleminit_(`$1',ifelse(m4type,T,m4lgth))
ifelse(`$3',R,`{M4ds: Here; move to last line.end; rp_ang = rp_ang+pi_')
{line to rvec_(rp_len/2,0)
 m4name: [
  S: Here; {line from rvec_(-m4lgth/2,m4wdth/2) to Here then
    to rvec_(-m4lgth/2,-m4wdth/2) m4line }
  E: rvec_(m4lgth/2,0); line from rvec_(0,m4wdth/2) to E then
    to rvec_(0,-m4wdth/2) m4line ] with .S at Here
  ifelse(m4type,T,,
   line from last [].E to last [].E+vec_((rp_len-m4lgth)/2,0))}
 ifelse(`$3',R,`rp_ang = rp_ang-pi_; move to M4ds}')
line invis to rvec_(rp_len,0) dnl
popdef(`m4lgth', `m4wdth', `m4type', `m4name', `m4line') ')

# Two-terminal utilities ##################################################

                           `Breakers or slashes in the input and output lines:
                           `m4_two(`2-term element macroname in quotes',
                              body length,
                              linespec,
                              body keys,
                              [Sn:] input breaker or slash keys,
                              [Sn:] output breaker or slash keys)'
                            In args 5 and 6 a prefix S: or Sn: draws slashes
                            (Default breaker names are BrI and BrO)
                            (Default slash names are SLI and SLO)'
define(`m4_two',
`pushkeys_(`$4',lgth:ifelse(`$2',,sl_ttboxlen_*2,`$2'); sep:-1)dnl
define(`m4s_pI',`ifelse(m4sep,(-1),
 `pushkey_(`$5',sep,sl_breakersep_) m4sep popdef(`m4sep')',m4sep)')dnl
define(`m4s_pO',`ifelse(m4sep,(-1),
 `pushkey_(`$6',sep,sl_breakersep_) m4sep popdef(`m4sep')',m4sep)')dnl
ifelse(regexp(`$5',^ *S),-1,
 `define(`m4il',`ifelse(`$5',,0,
   `pushkeys_(`$5',lgth:sl_breakersize_) m4lgth popdef(`m4lgth')')')dnl
  define(`m4ol',`ifelse(`$6',,0,
   `pushkey_(`$6',lgth,sl_breakersize_) m4lgth popdef(`m4lgth')')')dnl
  eleminit_(`$3',max(elen_,m4lgth+m4il*3.5+m4ol*3.5+m4s_pI+m4s_pO))
  M4end: last line.end
  { line to last line.c-vec_((m4lgth + m4il + m4ol + m4s_pI+m4s_pO)/2,0)
    pushdef(`m4bri',`ifelse(`$5',,,`$5',C,,`$5',O,`box=fill_(0)')')dnl
    ifelse(`$5',,,`sl_breaker(to rvec_(m4il,0),`$5';m4bri;name=BrI)
      line to rvec_(m4s_pI,0)')
    $1(to rvec_(m4lgth,0),`$4',,,`$7',`$8')
    pushdef(`m4bro',`ifelse(`$6',,,`$6',C,,`$6',O,`box=fill_(0)')')dnl
    ifelse(`$6',,,`line to rvec_(m4s_pO,0)
      sl_breaker(to rvec_(m4ol,0),`$6';m4bro;name=BrO)')
    line to M4end popdef(`m4bri', `m4bro') }',
 `eleminit_(`$3',max(elen_,m4lgth+(m4s_pI+m4s_pO)*2)) ; dnl slashes
  M4end: last line.end
  { $1(to M4end,`$4',,,`$7',`$8') }
  ifelse(`$5',,,`define(`m4in',`ifelse(`$5',S,1,`$5',S:,1,
     `patsubst(`$5',.*S\([0-9][0-9]*\):?.*,\1)')')dnl
    define(`m4inkey',`patsubst(`$5',^ *S[0-9]*:?)')dnl
    pushkey_(m4inkey,name,SLI,N)dnl
    {sl_slash(at rvec_(rp_len/2-m4lgth/2-m4s_pI,0),
      m4inkey;name=m4name,m4in:)} popdef(`m4name')')
  ifelse(`$6',,,`define(`m4on',`ifelse(`$6',S,1,`$6',S:,1,
     `patsubst(`$6',.*S\([0-9][0-9]*\):*.*,\1)')')dnl
    define(`m4onkey',`patsubst(`$6',^ *S[0-9]*:?)')dnl
    pushkey_(m4onkey,name,SLO,N)dnl
    {sl_slash(at rvec_(rp_len/2+m4lgth/2+m4s_pO,0),
      m4onkey;name=m4name,m4on:)} popdef(`m4name')') ')
  line invis to M4end popdef(`m4lgth', `m4sep') ')

                           `Chevrons in the input and output lines:'
                           `m4ch_two(`2-term element macroname in quotes',
                               linespec,
                               body keys)'
define(`m4ch_two',
`define(`m4bl',`setkey_(`$3',lgth,dimen_*3/8) m4lgth')popdef(`m4lgth')dnl
eleminit_(`$2',dimen_*2)
M4start: Here; M4end: last line.end
M4elem: $1(to rvec_(m4bl+2*sl_chevronsiz_,0) with .c at last line.c,`$3')
sl_drawout(from last line.start-vec_(sl_chevronsiz_,0) to last line.start,
  type=T,R)
line from last line.start to M4start
sl_drawout(from M4elem.end to M4elem.end+vec_(sl_chevronsiz_,0),type=T)
line to M4end
line invis from M4start to M4end')

# Composite elements ###################################################

                           `sl_ct( at position, keys, R|L|U|D|degrees )
                              keys:
                               type=L|N|S[n] (default L;
                                 Sn draws n slashes, default 2)
                               scale=expr (default 1);
                               grnd=degrees (type S or N);
                               stemlgth=expr (type L or S);
                               sep=expr (type S slashes to head separation);
                              type=N omits the stem and slashes
                              type=Sn draws n slashes, e.g., type=S2
                              Arg3 sets the stem direction'
define(`sl_ct',`[ Origin: Here
pushkeys_(`$2', type:L:N; scale:1; grnd::N ) dnl
define(`m4scf',`(m4scale*1.2)')dnl
setdir_(`$3',R)
ifinstr(m4type,L,`define(`m4lw',`dimen_/8*m4scf')dnl
  L: inductor(to vec_(2*m4lw,0) with .c at Origin,,2,,m4lw); C: L.c
    pushkey_(`$2',stemlgth,dimen_/5)dnl
    line from L.start to L.start+vec_(0,-m4stemlgth)
  Tstart: Here
    line from L.end to L.end+vec_(0,-m4stemlgth)
  Tend: Here
    line from L.c to L.c+vec_(0,-m4stemlgth)
  Tc: Here popdef(`m4stemlgth')',
m4type,S,
 `C: circle rad sourcerad_*m4scale*0.5 at Origin
  pushkeys_(`$2',stemlgth:C.rad*6; sep:C.rad*2)dnl
  Stem: line from C to C+vec_(m4stemlgth,0) chop C.rad chop 0; Tc: Here
  ifelse(m4grnd,,,`ground(at C+(Rect_(C.rad,m4grnd)),,,m4grnd)'); \
  pushdef(`m4n',`ifelse(m4type,S,2,`eval(patsubst(m4type,.*S))')')dnl
  sl_slash(at C+vec_(C.rad+m4sep,0),,m4n:rp_ang*rtod_)
  popdef(`m4stemlgth',`m4sep',`m4n')',
m4type,N,
 `C: circle rad sourcerad_*m4scale*0.5 at Origin
  ifelse(m4grnd,,,`ground(at C+(Rect_(C.rad,m4grnd)),,,m4grnd)') ')
resetdir_ popdef(`m4type', `m4scale', `m4grnd' )
] with .Origin ifelse(`$1',,at Here,`$1') ')

                           `sl_busbar( linespec, np, keys )
                            Labels P1, P2 ... Pnp are defined on the line.
                                 keys: line=line attributes;
                                       port=D; (dotted ports)
                                       indent=expr;
                                        (distance of points from Start and End)
                            The bus extends beyond the first and last points
                            by the indent value (default) sl_busindent_'
define(`sl_busbar',
`define(`m4npoints',`ifelse(`$2',,2,`$2')')dnl
 pushkeys_(`$3',`line:thick sl_busthick_:N; port::N; indent:sl_busindent_')dnl
 [ tmp_ang = rp_ang
   eleminit_(`$1',(m4npoints-1)*dimen_)
   Start: last line.start; End: last line.end
   Line: line from Start to End m4line; C: Line.c
   M: move from Start to End chop m4indent
   for_(1,m4npoints,1,
    `P`'m4x: (m4x-1)/(m4npoints-1) between M.start and M.end dnl
    ifinstr(m4port,D,` ;dot(at P`'m4x)')')
   point_(tmp_ang) ] popdef(`m4line', `m4port', `m4indent') ')

                         `sl_slash( at position, keys, [n:]R|L|U|D|degrees)
                          keys:
                            lines=line attributes;
                            size=expr (default ht dimen_/3); '
define(`sl_slash',`[ C: Here
pushkeys_(`$2',lines::N; size:dimen_/3 )dnl
define(`m4nsl',`ifelse(`$3',,1,`ifinstr(`$3',:,patsubst(`$3',:.*),1)')')dnl
pushdef(`m4ACd',`ifinstr(`$3',:,`patsubst(`$3',.*:)',`$3')')dnl
setdir_(ifelse(m4ACd,,`ifdef(`m4a_',rp_ang*rtod_,0)',m4ACd))
dx = m4size/2.5
for_(1,m4nsl,1, `{ line to vec_(m4size/sqrt(3),m4size) m4lines \
  with .c at C+vec_((m4x-(1+m4nsl)/2)*dx,0) } ')
resetdir_ popdef(`m4lines', `m4size',`m4ACd')
`$8' ] ifelse(`$1',,at Here,`$1')')

# #######################################################################

divert(0)dnl