Added sliders and made the ADSR controllabe with them
This commit is contained in:
		
							parent
							
								
									c33faa402b
								
							
						
					
					
						commit
						5e448edb21
					
				
							
								
								
									
										44
									
								
								elements/basic/lerpslider.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								elements/basic/lerpslider.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | |||||||
|  | package basicElements | ||||||
|  | 
 | ||||||
|  | // Numeric is a type constraint representing a number. | ||||||
|  | type Numeric interface { | ||||||
|  | 	~float32 | ~float64 | | ||||||
|  | 	~int     | ~int8    | ~int16  | ~int32  | ~int64  | | ||||||
|  | 	~uint    | ~uint8   | ~uint16 | ~uint32 | ~uint64 | ~uintptr | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type LerpSlider[T Numeric] struct { | ||||||
|  | 	*Slider | ||||||
|  | 	min T | ||||||
|  | 	max T | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func NewLerpSlider[T Numeric] (min, max T, value T, vertical bool) (element *LerpSlider[T]) { | ||||||
|  | 	if min > max { | ||||||
|  | 		temp := max | ||||||
|  | 		max = min | ||||||
|  | 		min = temp | ||||||
|  | 	} | ||||||
|  | 	element = &LerpSlider[T] { | ||||||
|  | 		Slider: NewSlider(0, vertical), | ||||||
|  | 		min: min, | ||||||
|  | 		max: max, | ||||||
|  | 	} | ||||||
|  | 	element.SetValue(value) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *LerpSlider[T]) SetValue (value T) { | ||||||
|  | 	value -= element.min | ||||||
|  | 	element.Slider.SetValue(float64(value) / float64(element.Range())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *LerpSlider[T]) Value () (value T) { | ||||||
|  | 	return T ( | ||||||
|  | 		float64(element.Slider.Value()) * float64(element.Range())) + | ||||||
|  | 		element.min | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *LerpSlider[T]) Range () T { | ||||||
|  | 	return element.max - element.min | ||||||
|  | } | ||||||
							
								
								
									
										197
									
								
								elements/basic/slider.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								elements/basic/slider.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,197 @@ | |||||||
|  | package basicElements | ||||||
|  | 
 | ||||||
|  | import "image" | ||||||
|  | import "git.tebibyte.media/sashakoshka/tomo/input" | ||||||
|  | import "git.tebibyte.media/sashakoshka/tomo/theme" | ||||||
|  | import "git.tebibyte.media/sashakoshka/tomo/config" | ||||||
|  | import "git.tebibyte.media/sashakoshka/tomo/artist" | ||||||
|  | import "git.tebibyte.media/sashakoshka/tomo/elements/core" | ||||||
|  | 
 | ||||||
|  | type Slider struct { | ||||||
|  | 	*core.Core | ||||||
|  | 	*core.FocusableCore | ||||||
|  | 	core core.CoreControl | ||||||
|  | 	focusableControl core.FocusableCoreControl | ||||||
|  | 
 | ||||||
|  | 	value   float64 | ||||||
|  | 	vertical bool | ||||||
|  | 	dragging bool | ||||||
|  | 	dragOffset int | ||||||
|  | 	track image.Rectangle | ||||||
|  | 	bar image.Rectangle | ||||||
|  | 	 | ||||||
|  | 	config config.Wrapped | ||||||
|  | 	theme  theme.Wrapped | ||||||
|  | 	 | ||||||
|  | 	onSlide   func () | ||||||
|  | 	onRelease func () | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func NewSlider (value float64, vertical bool) (element *Slider) { | ||||||
|  | 	element = &Slider { | ||||||
|  | 		value: value, | ||||||
|  | 		vertical: vertical, | ||||||
|  | 	} | ||||||
|  | 	if vertical { | ||||||
|  | 		element.theme.Case = theme.C("basic", "sliderVertical") | ||||||
|  | 	} else { | ||||||
|  | 		element.theme.Case = theme.C("basic", "sliderHorizontal") | ||||||
|  | 	} | ||||||
|  | 	element.Core, element.core = core.NewCore(element.draw) | ||||||
|  | 	element.FocusableCore, | ||||||
|  | 	element.focusableControl = core.NewFocusableCore(element.redo) | ||||||
|  | 	element.updateMinimumSize() | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleMouseDown (x, y int, button input.Button) { | ||||||
|  | 	if !element.Enabled() { return } | ||||||
|  | 	element.Focus() | ||||||
|  | 	if button == input.ButtonLeft { | ||||||
|  | 		element.dragging = true | ||||||
|  | 		element.value = element.valueFor(x, y) | ||||||
|  | 		if element.onSlide != nil { | ||||||
|  | 			element.onSlide() | ||||||
|  | 		} | ||||||
|  | 		element.redo() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleMouseUp (x, y int, button input.Button) { | ||||||
|  | 	if button != input.ButtonLeft || !element.dragging { return } | ||||||
|  | 	element.dragging = false | ||||||
|  | 	if element.onRelease != nil { | ||||||
|  | 		element.onRelease() | ||||||
|  | 	} | ||||||
|  | 	element.redo() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleMouseMove (x, y int) { | ||||||
|  | 	if element.dragging { | ||||||
|  | 		element.dragging = true | ||||||
|  | 		element.value = element.valueFor(x, y) | ||||||
|  | 		if element.onSlide != nil { | ||||||
|  | 			element.onSlide() | ||||||
|  | 		} | ||||||
|  | 		element.redo() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleMouseScroll (x, y int, deltaX, deltaY float64) { } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleKeyDown (key input.Key, modifiers input.Modifiers) { | ||||||
|  | 	// TODO: handle left and right arrows | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) HandleKeyUp (key input.Key, modifiers input.Modifiers) { } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) Value () (value float64) { | ||||||
|  | 	return element.value | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) SetEnabled (enabled bool) { | ||||||
|  | 	element.focusableControl.SetEnabled(enabled) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) SetValue (value float64) { | ||||||
|  | 	if value < 0 { value = 0 } | ||||||
|  | 	if value > 1 { value = 1 } | ||||||
|  | 	 | ||||||
|  | 	if element.value == value { return } | ||||||
|  | 
 | ||||||
|  | 	element.value = value | ||||||
|  | 	element.redo() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) OnSlide (callback func ()) { | ||||||
|  | 	element.onSlide = callback | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) OnRelease (callback func ()) { | ||||||
|  | 	element.onRelease = callback | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // SetTheme sets the element's theme. | ||||||
|  | func (element *Slider) SetTheme (new theme.Theme) { | ||||||
|  | 	if new == element.theme.Theme { return } | ||||||
|  | 	element.theme.Theme = new | ||||||
|  | 	element.redo() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // SetConfig sets the element's configuration. | ||||||
|  | func (element *Slider) SetConfig (new config.Config) { | ||||||
|  | 	if new == element.config.Config { return } | ||||||
|  | 	element.config.Config = new | ||||||
|  | 	element.updateMinimumSize() | ||||||
|  | 	element.redo() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) valueFor (x, y int) (value float64) { | ||||||
|  | 	if element.vertical { | ||||||
|  | 		value = | ||||||
|  | 			float64(y - element.track.Min.Y - element.bar.Dy() / 2) / | ||||||
|  | 			float64(element.track.Dy() - element.bar.Dy()) | ||||||
|  | 	} else { | ||||||
|  | 		value = | ||||||
|  | 			float64(x - element.track.Min.X - element.bar.Dx() / 2) / | ||||||
|  | 			float64(element.track.Dx() - element.bar.Dx()) | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	if value < 0 { value = 0 } | ||||||
|  | 	if value > 1 { value = 1 } | ||||||
|  | 	value = 1 - value | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) updateMinimumSize () { | ||||||
|  | 	if element.vertical { | ||||||
|  | 		element.core.SetMinimumSize ( | ||||||
|  | 			element.config.HandleWidth(), | ||||||
|  | 			element.config.HandleWidth() * 2) | ||||||
|  | 	} else { | ||||||
|  | 		element.core.SetMinimumSize ( | ||||||
|  | 			element.config.HandleWidth() * 2, | ||||||
|  | 			element.config.HandleWidth()) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) redo () { | ||||||
|  | 	if element.core.HasImage () { | ||||||
|  | 		element.draw() | ||||||
|  | 		element.core.DamageAll() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (element *Slider) draw () { | ||||||
|  | 	bounds := element.Bounds() | ||||||
|  | 	element.track = element.theme.Inset(theme.PatternGutter).Apply(bounds) | ||||||
|  | 	if element.vertical { | ||||||
|  | 		barSize := element.track.Dx() | ||||||
|  | 		element.bar = image.Rect(0, 0, barSize, barSize).Add(bounds.Min) | ||||||
|  | 		barOffset := | ||||||
|  | 			float64(element.track.Dy() - barSize) * | ||||||
|  | 			(1 - element.value) | ||||||
|  | 		element.bar = element.bar.Add(image.Pt(0, int(barOffset))) | ||||||
|  | 	} else { | ||||||
|  | 		barSize := element.track.Dy() | ||||||
|  | 		element.bar = image.Rect(0, 0, barSize, barSize).Add(bounds.Min) | ||||||
|  | 		barOffset := | ||||||
|  | 			float64(element.track.Dx() - barSize) * | ||||||
|  | 			element.value | ||||||
|  | 		element.bar = element.bar.Add(image.Pt(int(barOffset), 0)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	state := theme.PatternState { | ||||||
|  | 		Focused:  element.Focused(), | ||||||
|  | 		Disabled: !element.Enabled(), | ||||||
|  | 		Pressed:  element.dragging, | ||||||
|  | 	} | ||||||
|  | 	artist.FillRectangle ( | ||||||
|  | 		element, | ||||||
|  | 		element.theme.Pattern(theme.PatternGutter, state), | ||||||
|  | 		bounds) | ||||||
|  | 	artist.FillRectangle ( | ||||||
|  | 		element, | ||||||
|  | 		element.theme.Pattern(theme.PatternHandle, state), | ||||||
|  | 		element.bar) | ||||||
|  | } | ||||||
| @ -17,6 +17,12 @@ const bufferSize = 256 | |||||||
| var   tuning     = music.EqualTemparment { A4: 440 } | var   tuning     = music.EqualTemparment { A4: 440 } | ||||||
| var   waveform   = 0 | var   waveform   = 0 | ||||||
| var   playing    = map[music.Note] *toneStreamer { } | var   playing    = map[music.Note] *toneStreamer { } | ||||||
|  | var   adsr = ADSR { | ||||||
|  | 	Attack:  5 * time.Millisecond, | ||||||
|  | 	Decay:   400 * time.Millisecond, | ||||||
|  | 	Sustain: 0.7, | ||||||
|  | 	Release: 500 * time.Millisecond, | ||||||
|  | }  | ||||||
| 
 | 
 | ||||||
| func main () { | func main () { | ||||||
| 	speaker.Init(sampleRate, bufferSize) | 	speaker.Init(sampleRate, bufferSize) | ||||||
| @ -27,11 +33,9 @@ func run () { | |||||||
| 	window, _ := tomo.NewWindow(2, 2) | 	window, _ := tomo.NewWindow(2, 2) | ||||||
| 	window.SetTitle("Piano") | 	window.SetTitle("Piano") | ||||||
| 	container := basicElements.NewContainer(basicLayouts.Vertical { true, true }) | 	container := basicElements.NewContainer(basicLayouts.Vertical { true, true }) | ||||||
| 	window.Adopt(container) |  | ||||||
| 
 | 
 | ||||||
| 	controlBar := basicElements.NewContainer(basicLayouts.Horizontal { true, false }) | 	controlBar := basicElements.NewContainer(basicLayouts.Horizontal { true, false }) | ||||||
| 	label := basicElements.NewLabel("Play a song!", false) | 	label := basicElements.NewLabel("Play a song!", false) | ||||||
| 	controlBar.Adopt(label, true) |  | ||||||
| 	waveformButton := basicElements.NewButton("Sine") | 	waveformButton := basicElements.NewButton("Sine") | ||||||
| 	waveformButton.OnClick (func () { | 	waveformButton.OnClick (func () { | ||||||
| 		waveform = (waveform + 1) % 5 | 		waveform = (waveform + 1) % 5 | ||||||
| @ -43,15 +47,41 @@ func run () { | |||||||
| 		case 4: waveformButton.SetText("Supersaw") | 		case 4: waveformButton.SetText("Supersaw") | ||||||
| 		} | 		} | ||||||
| 	}) | 	}) | ||||||
| 	controlBar.Adopt(waveformButton, false) | 
 | ||||||
| 	container.Adopt(controlBar, false) | 	attackSlider  := basicElements.NewLerpSlider(0, 3 * time.Second, adsr.Attack, true) | ||||||
|  | 	decaySlider   := basicElements.NewLerpSlider(0, 3 * time.Second, adsr.Decay, true) | ||||||
|  | 	sustainSlider := basicElements.NewSlider(adsr.Sustain, true) | ||||||
|  | 	releaseSlider := basicElements.NewLerpSlider(0, 3 * time.Second, adsr.Release, true) | ||||||
|  | 
 | ||||||
|  | 	attackSlider.OnRelease (func () { | ||||||
|  | 		adsr.Attack = attackSlider.Value() | ||||||
|  | 	}) | ||||||
|  | 	decaySlider.OnRelease (func () { | ||||||
|  | 		adsr.Decay = decaySlider.Value() | ||||||
|  | 	}) | ||||||
|  | 	sustainSlider.OnRelease (func () { | ||||||
|  | 		adsr.Sustain = sustainSlider.Value() | ||||||
|  | 	}) | ||||||
|  | 	releaseSlider.OnRelease (func () { | ||||||
|  | 		adsr.Release = releaseSlider.Value() | ||||||
|  | 	}) | ||||||
| 	 | 	 | ||||||
| 	piano := fun.NewPiano(2, 5) | 	piano := fun.NewPiano(2, 5) | ||||||
| 	container.Adopt(piano, true) |  | ||||||
| 	piano.OnPress(playNote) | 	piano.OnPress(playNote) | ||||||
| 	piano.OnRelease(stopNote) | 	piano.OnRelease(stopNote) | ||||||
| 	piano.Focus() | 	piano.Focus() | ||||||
| 	 | 	 | ||||||
|  | 	window.Adopt(container) | ||||||
|  | 	controlBar.Adopt(label, true) | ||||||
|  | 	controlBar.Adopt(waveformButton, false) | ||||||
|  | 	controlBar.Adopt(basicElements.NewSpacer(true), false) | ||||||
|  | 	controlBar.Adopt(attackSlider, false) | ||||||
|  | 	controlBar.Adopt(decaySlider, false) | ||||||
|  | 	controlBar.Adopt(sustainSlider, false) | ||||||
|  | 	controlBar.Adopt(releaseSlider, false) | ||||||
|  | 	container.Adopt(controlBar, true) | ||||||
|  | 	container.Adopt(piano, false) | ||||||
|  | 	 | ||||||
| 	window.OnClose(tomo.Stop) | 	window.OnClose(tomo.Stop) | ||||||
| 	window.Show() | 	window.Show() | ||||||
| } | } | ||||||
| @ -71,12 +101,7 @@ func playNote (note music.Note) { | |||||||
| 		int(tuning.Tune(note)), | 		int(tuning.Tune(note)), | ||||||
| 		waveform, | 		waveform, | ||||||
| 		0.3, | 		0.3, | ||||||
| 		ADSR { | 		adsr) | ||||||
| 			Attack:  100 * time.Millisecond, |  | ||||||
| 			Decay:   400 * time.Millisecond, |  | ||||||
| 			Sustain: 0.7, |  | ||||||
| 			Release: 500 * time.Millisecond, |  | ||||||
| 		}) |  | ||||||
| 
 | 
 | ||||||
| 	stopNote(note) | 	stopNote(note) | ||||||
| 	speaker.Lock() | 	speaker.Lock() | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user