diff --git a/src/components/components.jl b/src/components/components.jl
index ae4561b7d0e2e3be2e40a4eb783131bcc4ff2987..4bab60504af5d547d80bb2efaba32c0feee50383 100644
--- a/src/components/components.jl
+++ b/src/components/components.jl
@@ -1,5 +1,12 @@
 module Components
 include("abstract_types.jl")
 include("gradient_waveforms/gradient_waveforms.jl")
+include("instant_gradients.jl")
+include("pulses/pulses.jl")
+
+import .AbstractTypes: BaseComponent, GradientWaveform, EventComponent, RFPulseComponent, ReadoutComponent
+import .GradientWaveforms: ConstantGradientBlock, ChangingGradientBlock, NoGradientBlock
+import .InstantGradients: InstantGradient1D, InstantGradient3D
+import .Pulses: GenericPulse, InstantRFPulse, SincPulse, ConstantPulse
 
 end
\ No newline at end of file
diff --git a/src/components/gradient_waveforms/gradient_waveforms.jl b/src/components/gradient_waveforms/gradient_waveforms.jl
index 52705582379a82d49d6c9b12836231f7a086ebde..c3b432d43bd526755d3eeb7f98e1c463282de541 100644
--- a/src/components/gradient_waveforms/gradient_waveforms.jl
+++ b/src/components/gradient_waveforms/gradient_waveforms.jl
@@ -14,7 +14,7 @@ Each part of this gradient waveform can compute:
 - [`qvec`]: area under curve in each dimension
 - [`bmat_gradient`]: diffusion weighting (scalar in 1D or matrix in 3D).
 """
-module Gradients
+module GradientWaveforms
 
 include("changing_gradient_blocks.jl")
 include("constant_gradient_blocks.jl")
diff --git a/src/components/instant_gradients.jl b/src/components/instant_gradients.jl
index a0a726d1ac3feddccd02fe760e6ce306c1f6c967..08e9cc99451c2dc4e601f9b1bf8fd5db7f91e3ca 100644
--- a/src/components/instant_gradients.jl
+++ b/src/components/instant_gradients.jl
@@ -1,6 +1,6 @@
 module InstantGradients
 import StaticArrays: SVector, SMatrix
-import ...Variables: VariableType, duration, qvec, bmat_gradient, get_free_variable, set_simple_constraints, qval, effective_time
+import ...Variables: VariableType, duration, qvec, bmat_gradient, get_free_variable, set_simple_constraints!, qval, effective_time
 import ..AbstractTypes: EventComponent, GradientWaveform
 
 """
@@ -34,7 +34,7 @@ end
 
 function InstantGradient1D(; orientation=[1, 0, 0], group=nothing, qval=nothing, variables...)
     res = InstantGradient1D(qval, orientation, group)
-    set_simple_constraints(res, variables)
+    set_simple_constraints!(res, variables)
     return res
 end
 
@@ -67,7 +67,7 @@ function InstantGradient3D(; qvec=[nothing, nothing, nothing], group=nothing, va
         qvec = [nothing, nothing, nothing]
     end
     res = InstantGradient3D(get_free_variable.(qvec), group)
-    set_simple_constraints(res, variables)
+    set_simple_constraints!(res, variables)
     return res
 end
 
diff --git a/src/pulses/constant_pulses.jl b/src/components/pulses/constant_pulses.jl
similarity index 70%
rename from src/pulses/constant_pulses.jl
rename to src/components/pulses/constant_pulses.jl
index b8366a725b2ff6c1b056dde4ca2f608e60462852..7cb42e612d7ccabffdf5d0e583ff051cc4d00aa1 100644
--- a/src/pulses/constant_pulses.jl
+++ b/src/components/pulses/constant_pulses.jl
@@ -1,35 +1,37 @@
 module ConstantPulses
-import JuMP: VariableRef, @constraint, @variable, value
-import ...BuildingBlocks: RFPulseBlock, set_simple_constraints!
-import ...Variables: variables, get_free_variable, flip_angle, phase, amplitude, frequency, start_time, end_time, VariableType, duration, effective_time, inverse_bandwidth
-import ...BuildSequences: global_model
-import ..GenericPulses: GenericPulse, make_generic
+import JuMP: @constraint
+import ...AbstractTypes: RFPulseComponent
+import ....BuildSequences: global_model
+import ....Variables: duration, amplitude, effective_time, flip_angle, phase, inverse_bandwidth, VariableType, set_simple_constraints!, frequency
+import ..GenericPulses: GenericPulse
 
 """
     ConstantPulse(; variables...)
 
 Represents an radio-frequency pulse with a constant amplitude and frequency (i.e., a rectangular function).
 
-## Properties
+## Parameters
+- `group`: name of the group to which this pulse belongs. This is used for scaling or adding phases/off-resonance frequencies.
+
+## Variables
 - [`flip_angle`](@ref): rotation expected for on-resonance spins in degrees.
 - [`duration`](@ref): duration of the RF pulse in ms.
 - [`amplitude`](@ref): amplitude of the RF pulse in kHz.
 - [`phase`](@ref): phase at the start of the RF pulse in degrees.
 - [`frequency`](@ref): frequency of the RF pulse relative to the Larmor frequency (in kHz).
 """
-struct ConstantPulse <: RFPulseBlock
+struct ConstantPulse <: RFPulseComponent
     amplitude :: VariableType
     duration :: VariableType
     phase :: VariableType
     frequency :: VariableType
-    scale :: Union{Nothing, Symbol}
+    group :: Union{Nothing, Symbol}
 end
 
-function ConstantPulse(amplitude=nothing, duration=nothing, phase=nothing, frequency=nothing, scale=nothing, kwargs...) 
+function ConstantPulse(; amplitude=nothing, duration=nothing, phase=nothing, frequency=nothing, group=nothing, kwargs...) 
     res = ConstantPulse(
         [get_free_variable(value) for value in (amplitude, duration, phase, frequency)]...,
-        scale
-
+        group
     )
     @constraint global_model() res.amplitude >= 0
     set_simple_constraints!(res, kwargs)
diff --git a/src/pulses/generic_pulses.jl b/src/components/pulses/generic_pulses.jl
similarity index 93%
rename from src/pulses/generic_pulses.jl
rename to src/components/pulses/generic_pulses.jl
index ec68ce91458d492477e395dc52672acbdc8daad8..08ac34d4c56ad39afa9c3435ea6b33cbaa08c1f7 100644
--- a/src/pulses/generic_pulses.jl
+++ b/src/components/pulses/generic_pulses.jl
@@ -1,7 +1,7 @@
 module GenericPulses
 
-import ...BuildingBlocks: RFPulseBlock, fixed, make_generic
-import ...Variables: variables, duration, amplitude, effective_time, flip_angle
+import ...AbstractTypes: RFPulseComponent
+import ....Variables: duration, amplitude, effective_time, flip_angle
 
 
 """
@@ -9,14 +9,17 @@ import ...Variables: variables, duration, amplitude, effective_time, flip_angle
     GenericPulse(time, amplitude; phase=0., frequency=0., effective_time=<halfway>)
 
 Create a Pulse profile that has been fully defined by N control point.
+
 All arguments should be arrays of the same length N defining these control points.
 
+This pulse has no free variables.
+
 - `time`: time since the start of this [`BuildingBlock`](@ref) in ms.
 - `amplitude`: amplitude of the RF pulse at every timepoint in kHz.
 - `phase`: phase of the RF pulse at every timpoint in degrees. If not set explicitly it will be determined by the provided starting `phase` (degrees) and the `frequency` (kHz).
 - `effective_time`: the time that the RF pulse should be considered to have taken place when computing a `Pathway` (defaults: whenever half of the final flip angle has been achieved for on-resonance spins).
 """
-struct GenericPulse <: RFPulseBlock
+struct GenericPulse <: RFPulseComponent
     time :: Vector{Float64}
     amplitude :: Vector{Float64}
     phase :: Vector{Float64}
diff --git a/src/components/pulses/instant_pulses.jl b/src/components/pulses/instant_pulses.jl
new file mode 100644
index 0000000000000000000000000000000000000000..826c6db9ac186695eaaf2dcf3ad11083594616a4
--- /dev/null
+++ b/src/components/pulses/instant_pulses.jl
@@ -0,0 +1,44 @@
+module InstantPulses
+import JuMP: @constraint
+import ...AbstractTypes: RFPulseComponent
+import ....BuildSequences: global_model
+import ....Variables: duration, amplitude, effective_time, flip_angle, phase, inverse_bandwidth, VariableType
+
+"""
+    InstantPulse(; flip_angle=nothing, phase=nothing, group=nothing)
+
+Return an instant RF pulse that rotates all spins by `flip_angle` around an axis that has an angle of `phase` with the X-Y plane.
+
+## Parameters
+- `group`: name of the group to which this pulse belongs. This is used for scaling or adding phases/off-resonance frequencies.
+
+## Variables
+- [`flip_angle`](@ref): angle by which spins are rotated in degrees.
+- [`phase`](@ref): angle of axis around which spins are rotated in degrees.
+"""
+struct InstantPulse <: RFPulseComponent
+    flip_angle :: VariableType
+    phase :: VariableType
+    group :: Union{Nothing, Symbol}
+end
+
+function InstantPulse(; flip_angle=nothing, phase=nothing, group=nothing) 
+    res = InstantPulse(
+        get_free_variable(flip_angle),
+        get_free_variable(phase),
+        group
+    )
+    @constraint global_model() res.flip_angle >= 0
+    return res
+end
+
+flip_angle(instant::InstantPulse) = instant.flip_angle
+phase(instant::InstantPulse) = instant.phase
+duration(::InstantPulse) = 0.
+effective_time(::InstantPulse) = 0.
+inverse_bandwidth(::InstantPulse) = 0.
+
+
+make_generic(block::InstantPulse) = block
+
+end
\ No newline at end of file
diff --git a/src/pulses/pulses.jl b/src/components/pulses/pulses.jl
similarity index 67%
rename from src/pulses/pulses.jl
rename to src/components/pulses/pulses.jl
index 0bee276bdec825b679764e7ca82f4200fafeb55f..216f6ff07c0c1603d1abba1e9945249cb92f46e5 100644
--- a/src/pulses/pulses.jl
+++ b/src/components/pulses/pulses.jl
@@ -4,10 +4,9 @@ include("instant_pulses.jl")
 include("constant_pulses.jl")
 include("sinc_pulses.jl")
 
-import ..Variables: effective_time
-import ..BuildingBlocks: RFPulseBlock
+import ..AbstractTypes: RFPulseComponent
 import .GenericPulses: GenericPulse
-import .InstantPulses: InstantRFPulseBlock
+import .InstantPulses: InstantPulse
 import .ConstantPulses: ConstantPulse
 import .SincPulses: SincPulse
 
diff --git a/src/components/pulses/sinc_pulses.jl b/src/components/pulses/sinc_pulses.jl
new file mode 100644
index 0000000000000000000000000000000000000000..3511399e84ebc0750d412b6f9033ee363c52a21a
--- /dev/null
+++ b/src/components/pulses/sinc_pulses.jl
@@ -0,0 +1,94 @@
+module SincPulses
+import JuMP: @constraint
+import QuadGK: quadgk
+import ....BuildSequences: global_model
+import ....Variables: duration, amplitude, effective_time, flip_angle, phase, inverse_bandwidth, VariableType, set_simple_constraints!, frequency
+import ...AbstractTypes: RFPulseComponent
+import ..GenericPulses: GenericPulse
+
+"""
+    SincPulse(; Nzeros=3, apodise=true, variables...)
+
+Represents a radio-frequency pulse with a sinc-like amplitude and constant frequency.
+
+## Parameters
+- `Nzeros`: Number of zero-crossings on each side of the sinc pulse. Can be set to a tuple with two values to have a different number of zero crossings on the left and the right of the sinc pulse.
+- `apodise`: if true (default) applies a Hanning apodising window to the sinc pulse.
+- `group`: name of the group to which this pulse belongs. This is used for scaling or adding phases/off-resonance frequencies.
+
+## Variables
+- [`flip_angle`](@ref): rotation expected for on-resonance spins in degrees.
+- [`duration`](@ref): duration of the RF pulse in ms.
+- [`amplitude`](@ref): amplitude of the RF pulse in kHz.
+- [`phase`](@ref): phase at the start of the RF pulse in degrees.
+- [`frequency`](@ref): frequency of the RF pulse relative to the Larmor frequency (in kHz).
+- [`bandwidth`](@ref): width of the rectangular function in frequency space (in kHz). If the `duration` is short (compared with 1/`bandwidth`), this bandwidth will only be approximate.
+"""
+struct SincPulse <: RFPulseComponent
+    symmetric :: Bool
+    apodise :: Bool
+    Nzeros :: Tuple{Integer, Integer}
+    norm_flip_angle :: Tuple{Float64, Float64}
+    amplitude :: VariableType
+    phase :: VariableType
+    frequency :: VariableType
+    lobe_duration :: VariableType
+    group :: Union{Nothing, Symbol}
+end
+
+function SincPulse(; 
+    Nzeros=3, apodise=true,
+    amplitude=nothing, phase=nothing, frequency=nothing, lobe_duration=nothing, group=nothing, kwargs...
+) 
+    if Nzeros isa Number
+        Nzeros = (Nzeros, Nzeros)
+    end
+    res = SincPulse(
+        symmetric,
+        apodise,
+        Nzeros,
+        integral_nzero.(Nzeros, apodise),
+        [get_free_variable(value) for value in (amplitude, phase, frequency, lobe_duration)]...,
+        group
+    )
+    @constraint global_model() res.amplitude >= 0
+    set_simple_constraints!(res, kwargs)
+    return res
+end
+
+function normalised_function(x; apodise=false)
+    if iszero(x)
+        return 1.
+    end
+    if apodise
+        return (0.54 + 0.46 * cos(Ï€ * x)) * sin(Ï€ * x) / (Ï€ * x)
+    else
+        return sin(Ï€ * x) / (Ï€ * x)
+    end
+end
+
+function integral_nzero(Nzeros, apodise)
+    f = x -> normalised_function(x; apodise=apodise)
+    return quadgk(f, 0, Nzeros)[1]
+end
+
+amplitude(pulse::SincPulse) = pulse.amplitude
+N_left(pulse::SincPulse) = pulse.Nzeros[1]
+N_right(pulse::SincPulse) = pulse.Nzeros[2]
+duration(pulse::SincPulse) = (N_left(pulse) + N_right(pulse)) * lobe_duration(pulse)
+phase(pulse::SincPulse) = pulse.phase
+frequency(pulse::SincPulse) = pulse.frequency
+flip_angle(pulse::SincPulse) = (pulse.norm_flip_angle[1] + pulse.norm_flip_angle[2]) * amplitude(pulse) * lobe_duration(pulse) * 360
+lobe_duration(pulse::SincPulse) = pulse.lobe_duration
+inverse_bandwidth(pulse::SincPulse) = lobe_duration(pulse)
+effective_time(pulse::SincPulse) = N_left(pulse) * lobe_duration(pulse)
+
+function make_generic(block::SincPulse)
+    normed_times = -N_left(block):0.1:N_right(block) + 1e-5
+    times = (normed_times .+ N_left(block)) .* lobe_duration(block)
+    amplitudes = amplitude(block) .* (normalised_function.(normed_times; apodise=block.apodise))
+    phases = [frequency(block) .* lobe_duration(block)] .* normed_times .* 360
+    return GenericPulse(times, amplitudes, phases, effective_time(block))
+end
+
+end
\ No newline at end of file
diff --git a/src/pulses/instant_pulses.jl b/src/pulses/instant_pulses.jl
deleted file mode 100644
index b575f16ba340293382e6877721c3143a197d1557..0000000000000000000000000000000000000000
--- a/src/pulses/instant_pulses.jl
+++ /dev/null
@@ -1,32 +0,0 @@
-module InstantPulses
-import JuMP: @constraint, @variable, VariableRef, value
-import ...BuildingBlocks: RFPulseBlock, make_generic
-import ...Variables: flip_angle, phase, start_time, variables, duration, get_free_variable, VariableType, effective_time, inverse_bandwidth
-import ...BuildSequences: global_model
-
-struct InstantRFPulseBlock <: RFPulseBlock
-    flip_angle :: VariableType
-    phase :: VariableType
-    scale :: Union{Nothing, Symbol}
-end
-
-function InstantRFPulseBlock(; flip_angle=nothing, phase=nothing, scale=nothing) 
-    res = InstantRFPulseBlock(
-        get_free_variable(flip_angle),
-        get_free_variable(phase),
-        scale
-    )
-    @constraint global_model() res.flip_angle >= 0
-    return res
-end
-
-flip_angle(instant::InstantRFPulseBlock) = instant.flip_angle
-phase(instant::InstantRFPulseBlock) = instant.phase
-duration(::InstantRFPulseBlock) = 0.
-effective_time(::InstantRFPulseBlock) = 0.
-inverse_bandwidth(::InstantRFPulseBlock) = 0.
-
-
-make_generic(block::InstantRFPulseBlock) = block
-
-end
\ No newline at end of file
diff --git a/src/pulses/sinc_pulses.jl b/src/pulses/sinc_pulses.jl
deleted file mode 100644
index a06af212205dd193bec75a6498a36c896048e328..0000000000000000000000000000000000000000
--- a/src/pulses/sinc_pulses.jl
+++ /dev/null
@@ -1,115 +0,0 @@
-module SincPulses
-
-import JuMP: VariableRef, @constraint, @variable
-import QuadGK: quadgk
-import Polynomials: fit, Polynomial
-import ...BuildingBlocks: RFPulseBlock, set_simple_constraints!
-import ...Variables: flip_angle, phase, amplitude, frequency, VariableType, variables, get_free_variable, duration, effective_time, inverse_bandwidth
-import ...BuildSequences: global_model
-import ..GenericPulses: GenericPulse, make_generic
-
-"""
-    SincPulse(; symmetric=true, max_Nlobes=nothing, apodise=true, variables...)
-
-Represents an radio-frequency pulse with a constant amplitude and frequency.
-
-## Parameters
-- `symmetric`: by default the sinc pulse will be symmetric (i.e., `N_left` == `N_right`). Set `symmetric=false` to independently control `N_left` and `N_right`.
-- `max_Nlobes`: by default the computed [`flip_angle`](@ref) is only approximated as `amplitude` * `lobe_duration`. By setting `max_Nlobes` the flip_angle will be given by the actual integral of the sinc function, which will be more accurate for small number of lobes. However, the number of lobes will not be able to exceed this `max_Nlobes`.
-- `apodise`: if true (default) applies a Hanning apodising window to the sinc pulse.
-
-## Variables
-- [`N_left`](@ref): number of zero-crossings of the sinc function before the main peak (minimum of 1).
-- [`N_right`](@ref): number of zero-crossings of the sinc function after the main peak (minimum of 1).
-- [`flip_angle`](@ref): rotation expected for on-resonance spins in degrees (only approximate if `max_Nlobes` is not set).
-- [`duration`](@ref): duration of the RF pulse in ms.
-- [`amplitude`](@ref): amplitude of the RF pulse in kHz.
-- [`phase`](@ref): phase at the start of the RF pulse in degrees.
-- [`frequency`](@ref): frequency of the RF pulse relative to the Larmor frequency (in kHz).
-- [`bandwidth`](@ref): width of the rectangular function in frequency space (in kHz). If the `duration` is short (compared with 1/`bandwidth`), this bandwidth will only be approximate.
-"""
-struct SincPulse <: RFPulseBlock
-    symmetric :: Bool
-    apodise :: Bool
-    nlobe_integral :: Polynomial
-    N_left :: VariableType
-    N_right :: VariableType
-    amplitude :: VariableType
-    phase :: VariableType
-    frequency :: VariableType
-    lobe_duration :: VariableType
-    scale :: Union{Nothing, Symbol}
-end
-
-function SincPulse(; 
-    symmetric=true, max_Nlobes=nothing, apodise=true, N_lobes=nothing, N_left=nothing, N_right=nothing, 
-    amplitude=nothing, phase=nothing, frequency=nothing, lobe_duration=nothing, scale=nothing, kwargs...
-) 
-    if symmetric
-        N_lobes = get_free_variable(N_lobes)
-        @assert isnothing(N_left) && isnothing(N_right) "N_left and N_right cannot be set if symmetric=true (default)"
-        N_left_var = N_right_var = N_lobes
-    else
-        @assert isnothing(N_lobes) "N_lobes cannot be set if symmetric=true (default)"
-        N_left_var = get_free_variable(N_left)
-        N_right_var = get_free_variable(N_right)
-    end
-    res = SincPulse(
-        symmetric,
-        apodise,
-        nlobe_integral_params(max_Nlobes, apodise),
-        N_left_var,
-        N_right_var,
-        [get_free_variable(value) for value in (amplitude, phase, frequency, lobe_duration)]...,
-        scale
-    )
-    model = global_model()
-    @constraint model res.amplitude >= 0
-    @constraint model res.N_left >= 1
-    if !symmetric
-        @constraint model res.N_right >= 1
-    end
-    set_simple_constraints!(res, kwargs)
-    return res
-end
-
-function normalised_function(x; apodise=false)
-    if iszero(x)
-        return 1.
-    end
-    if apodise
-        return (0.54 + 0.46 * cos(Ï€ * x)) * sin(Ï€ * x) / (Ï€ * x)
-    else
-        return sin(Ï€ * x) / (Ï€ * x)
-    end
-end
-
-function nlobe_integral_params(Nlobe_max, apodise=false)
-    if isnothing(Nlobe_max)
-        return fit([1], [0.5], 0)
-    end
-    f = x -> normalised_function(x; apodise=apodise)
-    integral_values = [quadgk(f, 0, i)[1] for i in 1:Nlobe_max]
-    return fit(1:Nlobe_max, integral_values, Nlobe_max - 1)
-end
-
-amplitude(pulse::SincPulse) = pulse.amplitude
-N_left(pulse::SincPulse) = pulse.N_left
-N_right(pulse::SincPulse) = pulse.N_right
-duration(pulse::SincPulse) = (N_left(pulse) + N_right(pulse)) * lobe_duration(pulse)
-phase(pulse::SincPulse) = pulse.phase
-frequency(pulse::SincPulse) = pulse.frequency
-flip_angle(pulse::SincPulse) = (pulse.nlobe_integral(N_left(pulse)) + pulse.nlobe_integral(N_right(pulse))) * amplitude(pulse) * lobe_duration(pulse) * 360
-lobe_duration(pulse::SincPulse) = pulse.lobe_duration
-inverse_bandwidth(pulse::SincPulse) = lobe_duration(pulse)
-effective_time(pulse::SincPulse) = N_left(pulse) * lobe_duration(pulse)
-
-function make_generic(block::SincPulse)
-    normed_times = -N_left(block):0.1:N_right(block) + 1e-5
-    times = (normed_times .+ N_left(block)) .* lobe_duration(block)
-    amplitudes = amplitude(block) .* (normalised_function.(normed_times; apodise=block.apodise))
-    phases = [frequency(block) .* lobe_duration(block)] .* normed_times .* 360
-    return GenericPulse(times, amplitudes, phases, effective_time(block))
-end
-
-end
\ No newline at end of file