summaryrefslogtreecommitdiff
path: root/src/position_power.cc
blob: 0dc369bc86d181539cdc8a76af21cbaa8f44c0cb (plain)
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
94
95
96
97
98
99
/* -*- Mode: c++ -*- */
/***************************************************************************
 *            position_power.cc
 *
 *  Wed Jul 24 15:05:27 CEST 2024
 *  Copyright 2024 Bent Bisballe Nyeng
 *  deva@aasimon.org
 ****************************************************************************/

/*
 *  This file is part of DrumGizmo.
 *
 *  DrumGizmo is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU Lesser General Public License as published by
 *  the Free Software Foundation; either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  DrumGizmo is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU Lesser General Public License for more details.
 *
 *  You should have received a copy of the GNU Lesser General Public License
 *  along with DrumGizmo; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA.
 */
#include "position_power.h"

#include "sample.h"
#include "instrument.h"

#include <set>
#include <algorithm>

Instrument::PowerRange positionPower(const std::vector<Sample*>& samplelist, double position)
{
	if(samplelist.empty())
	{
		return {0.0, 1.0};
	}

	struct PosPower
	{
		double position;
		double power;
		Sample* sample;
	};
	auto dist_cmp =
		[position](const PosPower& a, const PosPower& b)
		{
			auto position_delta =
				std::abs(a.position - position) - std::abs(b.position - position);
			if(position_delta != 0)
			{
				return position_delta < 0.0;
			}
			return a.sample < b.sample;
		};
	std::set<PosPower, decltype(dist_cmp)> sorted_samples(dist_cmp);

	std::for_each(samplelist.begin(), samplelist.end(),
	              [&](Sample* s)
	              {
		              sorted_samples.insert({s->getPosition(), s->getPower(), s});
	              });

	// Find the smallest, closest set in terms of delta-position against the note position
	// and find the contained power range.
	double power_min{std::numeric_limits<double>::max()};
	double power_max{std::numeric_limits<double>::min()};
	auto sample_iter = sorted_samples.begin();
	auto final_position_boundary = sample_iter->position;
	for(std::size_t i = 0; i < std::max(sorted_samples.size() / 4, std::size_t(1)); ++i)
	{
		auto power = sample_iter->power;
		final_position_boundary = sample_iter->position;
		power_min = std::min(power_min, power);
		power_max = std::max(power_max, power);
		++sample_iter;
	}

	// Include upcoming samples from the list as long as their distances are contained in
	// the final position range.
	while(sample_iter != sorted_samples.end())
	{
		if(sample_iter->position != final_position_boundary)
		{
			// Position has left the range - and since the list is sorted; stop.
			break;
		}

		auto power = sample_iter->power;
		power_min = std::min(power_min, power);
		power_max = std::max(power_max, power);
		++sample_iter;
	}

	return {power_min, power_max};
}