aboutsummaryrefslogtreecommitdiffstats
path: root/mass/mass.go
blob: 29ecbd539e4283abea8ded658b3815899ab9212b (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
package mass

import (
	"errors"
	"fmt"
	"time"
)

type unit float32

const (
	Gram     unit = 1
	Kilogram unit = 1_000
	Pound    unit = 453.5924
)

type Mass struct {
	Val  float32
	Unit unit
}

func (m Mass) AsUnit(u unit) float32 {
	g := m.Val * float32(m.Unit) // Convert to grams.
	return g / float32(u)        // Convert to desired unit.
}

type flowRateUnit float32

const (
	KilogramsPerSecond flowRateUnit = 1
	PoundsPerMinute    flowRateUnit = 0.007_559_872_833
)

// FlowRateUnitStrings returns a slice of strings, each representing a
// flowRateUnit.
// This is necessary because giu.Combo only works with strings.
func FlowRateUnitStrings() []string {
	return []string{"kg/s", "lb/min"}
}

const (
	DefaultFlowRateUnit flowRateUnit = KilogramsPerSecond
	// DefaultFlowRateUnitIndex is used to index FlowRateUnitStrings()
	DefaultFlowRateUnitIndex int32 = 0 // kg/s
)

func FlowRateUnitFromString(s string) (flowRateUnit, error) {
	// Each case corresponds to a value in FlowRateUnitStrings().
	switch s {
	case "kg/s":
		return KilogramsPerSecond, nil
	case "lb/min":
		return PoundsPerMinute, nil
	default:
		return *new(flowRateUnit), errors.New(
			fmt.Sprintf("invalid massFlowRateUnit: '%s'", s))
	}
}

type FlowRate struct {
	Val  float32
	Unit flowRateUnit
}

func NewFlowRate(m Mass, t time.Duration, u flowRateUnit) (FlowRate, error) {
	switch u {
	case KilogramsPerSecond:
		return FlowRate{
			m.AsUnit(Kilogram) / float32(t.Seconds()),
			u,
		}, nil
	case PoundsPerMinute:
		return FlowRate{
			m.AsUnit(Pound) / float32(t.Minutes()),
			u,
		}, nil
	default:
		return *new(FlowRate), errors.New(
			fmt.Sprintf("invalid massFlowRateUnit: '%v'", u))
	}
}

func (fr FlowRate) AsUnit(u flowRateUnit) float32 {
	kgps := fr.Val * float32(fr.Unit) // Convert to kilogramsPerSecond.
	return kgps / float32(u)          // Convert to desired unit.
}