File: span.ha

package info (click to toggle)
hare 0.25.2-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 6,948 kB
  • sloc: asm: 1,264; makefile: 123; sh: 114; lisp: 101
file content (93 lines) | stat: -rw-r--r-- 2,760 bytes parent folder | download
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
87
88
89
90
91
92
93
// SPDX-License-Identifier: MPL-2.0
// (c) Hare authors <https://harelang.org>

use fmt;

// Represents a span of time in the Gregorian chronology; a set of nominal
// periods of time. Used for chronological arithmetic.
export type span = struct {
	years: i64,
	months: i64,
	weeks: i64,
	days: i64,
	hours: i64,
	minutes: i64,
	seconds: i64,
	nanoseconds: i64,
};

// Returns true if two [[span]]s are numerically equal, false otherwise.
export fn equalspan(x: span, y: span) bool = {
	return (
		x.years          == y.years
		&& x.months      == y.months
		&& x.weeks       == y.weeks
		&& x.days        == y.days
		&& x.hours       == y.hours
		&& x.minutes     == y.minutes
		&& x.seconds     == y.seconds
		&& x.nanoseconds == y.nanoseconds
	);
};

// Returns the sum [[span]] of a set of spans.
export fn sum(spans: span...) span = {
	let x = span { ... };
	for (let y &.. spans) {
		x.years       += y.years;
		x.months      += y.months;
		x.weeks       += y.weeks;
		x.days        += y.days;
		x.hours       += y.hours;
		x.minutes     += y.minutes;
		x.seconds     += y.seconds;
		x.nanoseconds += y.nanoseconds;
	};
	return x;
};

// Returns a [[span]] with its fields negated.
export fn neg(x: span) span = span {
	years       = -x.years,
	months      = -x.months,
	weeks       = -x.weeks,
	days        = -x.days,
	hours       = -x.hours,
	minutes     = -x.minutes,
	seconds     = -x.seconds,
	nanoseconds = -x.nanoseconds,
};

// Returns a [[span]] with its fields made absolute and positive.
export fn abs(x: span) span = span {
	years       = if (x.years < 0)       -x.years       else x.years,
	months      = if (x.months < 0)      -x.months      else x.months,
	weeks       = if (x.weeks < 0)       -x.weeks       else x.weeks,
	days        = if (x.days < 0)        -x.days        else x.days,
	hours       = if (x.hours < 0)       -x.hours       else x.hours,
	minutes     = if (x.minutes < 0)     -x.minutes     else x.minutes,
	seconds     = if (x.seconds < 0)     -x.seconds     else x.seconds,
	nanoseconds = if (x.nanoseconds < 0) -x.nanoseconds else x.nanoseconds,
};

// Formats a [[span]] and writes it into an internal static buffer. The returned
// string is borrowed from this buffer. The following layout is used:
//
//     "?Y ?M ?W ?D ?h ?m ?s ?n"
//
// where each '?' is replaced with the appopriate span field values in order of
// decreasing significance, and the inbetween spaces are replaced with the given
// sep argument.
export fn formatspan(s: span, sep: rune = ' ') str = {
	static let buf: [128]u8 = [0...];
	return fmt::bsprintf(buf, "{}Y{}{}M{}{}W{}{}D{}{}h{}{}m{}{}s{}{}n",
		s.years, sep,
		s.months, sep,
		s.weeks, sep,
		s.days, sep,
		s.hours, sep,
		s.minutes, sep,
		s.seconds, sep,
		s.nanoseconds,
	)!;
};