Majority calculation for assignments.
Majorities when config YN(A) and simple voting method.
This commit is contained in:
parent
9440842e6a
commit
32aaaf5b9e
@ -15,6 +15,7 @@ Agenda:
|
|||||||
Assignments:
|
Assignments:
|
||||||
- Remove unused assignment config to publish winner election results only.
|
- Remove unused assignment config to publish winner election results only.
|
||||||
- Added options to calculate percentages on different bases.
|
- Added options to calculate percentages on different bases.
|
||||||
|
- Added majority calculation.
|
||||||
|
|
||||||
Core:
|
Core:
|
||||||
- Added support for big assemblies with lots of users.
|
- Added support for big assemblies with lots of users.
|
||||||
|
@ -44,6 +44,17 @@ def get_config_variables():
|
|||||||
group='Elections',
|
group='Elections',
|
||||||
subgroup='Ballot and ballot papers')
|
subgroup='Ballot and ballot papers')
|
||||||
|
|
||||||
|
# TODO: Add server side validation of the choices.
|
||||||
|
yield ConfigVariable(
|
||||||
|
name='assignments_poll_default_majority_method',
|
||||||
|
default_value='simple_majority',
|
||||||
|
input_type='majorityMethod',
|
||||||
|
label='Required majority',
|
||||||
|
help_text='Default method to check whether a candidate has reached the required majority.',
|
||||||
|
weight=425,
|
||||||
|
group='Elections',
|
||||||
|
subgroup='Ballot and ballot papers')
|
||||||
|
|
||||||
yield ConfigVariable(
|
yield ConfigVariable(
|
||||||
name='assignments_pdf_ballot_papers_selection',
|
name='assignments_pdf_ballot_papers_selection',
|
||||||
default_value='CUSTOM_NUMBER',
|
default_value='CUSTOM_NUMBER',
|
||||||
|
@ -5,7 +5,8 @@
|
|||||||
angular.module('OpenSlidesApp.assignments.site', [
|
angular.module('OpenSlidesApp.assignments.site', [
|
||||||
'OpenSlidesApp.assignments',
|
'OpenSlidesApp.assignments',
|
||||||
'OpenSlidesApp.core.pdf',
|
'OpenSlidesApp.core.pdf',
|
||||||
'OpenSlidesApp.assignments.pdf'
|
'OpenSlidesApp.assignments.pdf',
|
||||||
|
'OpenSlidesApp.poll.majority'
|
||||||
])
|
])
|
||||||
|
|
||||||
.config([
|
.config([
|
||||||
@ -244,6 +245,100 @@ angular.module('OpenSlidesApp.assignments.site', [
|
|||||||
}
|
}
|
||||||
])
|
])
|
||||||
|
|
||||||
|
// Cache for AssignmentPollDetailCtrl so that users choices are keeped during user actions (e. g. save poll form).
|
||||||
|
.value('AssignmentPollDetailCtrlCache', {})
|
||||||
|
|
||||||
|
// Child controller of AssignmentDetailCtrl for each single poll.
|
||||||
|
.controller('AssignmentPollDetailCtrl', [
|
||||||
|
'$scope',
|
||||||
|
'MajorityMethodChoices',
|
||||||
|
'MajorityCalculation',
|
||||||
|
'Config',
|
||||||
|
'AssignmentPollDetailCtrlCache',
|
||||||
|
function ($scope, MajorityMethodChoices, MajorityCalculation, Config, AssignmentPollDetailCtrlCache) {
|
||||||
|
$scope.poll_options_with_majorities = $scope.poll.options;
|
||||||
|
// Define choices.
|
||||||
|
$scope.methodChoices = MajorityMethodChoices;
|
||||||
|
// TODO: Get $scope.baseChoices from config_variables.py without copying them.
|
||||||
|
|
||||||
|
// Setup empty cache with default values.
|
||||||
|
if (AssignmentPollDetailCtrlCache[$scope.poll.id] === undefined) {
|
||||||
|
AssignmentPollDetailCtrlCache[$scope.poll.id] = {
|
||||||
|
isMajorityCalculation: true,
|
||||||
|
isMajorityDetails: false,
|
||||||
|
method: $scope.config('assignments_poll_default_majority_method'),
|
||||||
|
base: $scope.config('assignments_poll_100_percent_base')
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch users choices from cache.
|
||||||
|
$scope.isMajorityCalculation = AssignmentPollDetailCtrlCache[$scope.poll.id].isMajorityCalculation;
|
||||||
|
$scope.isMajorityDetails = AssignmentPollDetailCtrlCache[$scope.poll.id].isMajorityDetails;
|
||||||
|
$scope.method = AssignmentPollDetailCtrlCache[$scope.poll.id].method;
|
||||||
|
$scope.base = AssignmentPollDetailCtrlCache[$scope.poll.id].base;
|
||||||
|
|
||||||
|
// (re)calculate the base of poll calculations.
|
||||||
|
$scope.calculateBase = function() {
|
||||||
|
var base;
|
||||||
|
switch($scope.base) {
|
||||||
|
case 'YES_NO_ABSTAIN':
|
||||||
|
case 'YES_NO':
|
||||||
|
if ($scope.poll.pollmethod == 'votes') {
|
||||||
|
base = MajorityCalculation.options_yes_sum($scope.poll);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'VALID':
|
||||||
|
base = $scope.poll.votesvalid;
|
||||||
|
break;
|
||||||
|
case 'CAST':
|
||||||
|
base = $scope.poll.votescast;
|
||||||
|
break;
|
||||||
|
// case 'DISABLED have no bases
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
};
|
||||||
|
|
||||||
|
// calculates if majority thresholds have been reached. Returns 0+ when reached, or negative int when not reached
|
||||||
|
$scope.recalculateMajorities = function (method) {
|
||||||
|
$scope.method = method;
|
||||||
|
$scope.calculationError = false;
|
||||||
|
var base_nmbr = $scope.calculateBase();
|
||||||
|
_.forEach($scope.poll_options_with_majorities, function(option) {
|
||||||
|
var optionvotes = {};
|
||||||
|
_.forEach(option.votes, function(vote) {
|
||||||
|
switch (vote.value) {
|
||||||
|
case 'Yes':
|
||||||
|
case 'Votes':
|
||||||
|
optionvotes.yes = vote.weight;
|
||||||
|
break;
|
||||||
|
case 'No':
|
||||||
|
optionvotes.no = vote.weight;
|
||||||
|
break;
|
||||||
|
case 'Abstain':
|
||||||
|
optionvotes.abstain = vote.weight;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
option.isReached = MajorityCalculation.isReached($scope.base, method, optionvotes, base_nmbr);
|
||||||
|
if (option.reached === 'undefined'){
|
||||||
|
$scope.calculationError = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
$scope.recalculateMajorities($scope.method);
|
||||||
|
|
||||||
|
// Save current values to cache on destroy of this controller.
|
||||||
|
$scope.$on('$destroy', function() {
|
||||||
|
AssignmentPollDetailCtrlCache[$scope.poll.id] = {
|
||||||
|
isMajorityCalculation: $scope.isMajorityCalculation,
|
||||||
|
isMajorityDetails: $scope.isMajorityDetails,
|
||||||
|
method: $scope.method,
|
||||||
|
base: $scope.base
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
])
|
||||||
|
|
||||||
.controller('AssignmentListCtrl', [
|
.controller('AssignmentListCtrl', [
|
||||||
'$scope',
|
'$scope',
|
||||||
'ngDialog',
|
'ngDialog',
|
||||||
@ -883,6 +978,12 @@ angular.module('OpenSlidesApp.assignments.site', [
|
|||||||
gettext('Number of all participants');
|
gettext('Number of all participants');
|
||||||
gettext('Use the following custom number');
|
gettext('Use the following custom number');
|
||||||
gettext('Custom number of ballot papers');
|
gettext('Custom number of ballot papers');
|
||||||
|
gettext('Required majority');
|
||||||
|
gettext('Default method to check whether a candidate has reached the required majority.');
|
||||||
|
gettext('Simple majority');
|
||||||
|
gettext('Two-thirds majority');
|
||||||
|
gettext('Three-quarters majority');
|
||||||
|
gettext('Disabled');
|
||||||
gettext('Title for PDF document (all elections)');
|
gettext('Title for PDF document (all elections)');
|
||||||
gettext('Preamble text for PDF document (all elections)');
|
gettext('Preamble text for PDF document (all elections)');
|
||||||
//other translations
|
//other translations
|
||||||
|
@ -126,9 +126,8 @@
|
|||||||
|
|
||||||
<uib-tabset ng-if="assignment.polls.length > 0" class="spacer ballot-tabs" active="activeTab">
|
<uib-tabset ng-if="assignment.polls.length > 0" class="spacer ballot-tabs" active="activeTab">
|
||||||
<uib-tab ng-repeat="poll in assignment.polls | orderBy:'-id'"
|
<uib-tab ng-repeat="poll in assignment.polls | orderBy:'-id'"
|
||||||
index="$index"
|
index="$index" heading="{{ 'Ballot' | translate }} {{ assignment.polls.length - $index }}">
|
||||||
heading="{{ 'Ballot' | translate }} {{ assignment.polls.length - $index }}">
|
<div ng-controller="AssignmentPollDetailCtrl">
|
||||||
|
|
||||||
<!-- action buttons -->
|
<!-- action buttons -->
|
||||||
<div class="pull-right">
|
<div class="pull-right">
|
||||||
<!-- delete -->
|
<!-- delete -->
|
||||||
@ -175,7 +174,7 @@
|
|||||||
<div ng-if="!poll.has_votes">
|
<div ng-if="!poll.has_votes">
|
||||||
<strong translate>Candidates</strong>
|
<strong translate>Candidates</strong>
|
||||||
<ul class="list-unstyled">
|
<ul class="list-unstyled">
|
||||||
<li ng-repeat="option in poll.options">
|
<li ng-repeat="option in options">
|
||||||
<a ui-sref="users.user.detail({id: option.candidate.id})">
|
<a ui-sref="users.user.detail({id: option.candidate.id})">
|
||||||
{{ option.candidate.get_full_name() }}
|
{{ option.candidate.get_full_name() }}
|
||||||
</a>
|
</a>
|
||||||
@ -186,14 +185,23 @@
|
|||||||
<span ng-if="poll.pollmethod == 'yn'" translate>Yes/No per candidate</span>
|
<span ng-if="poll.pollmethod == 'yn'" translate>Yes/No per candidate</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Settings for majority calculations -->
|
||||||
|
<div os-perms="assignments.can_manage" ng-hide="config('assignments_poll_default_majority_method') == 'disabled' || !poll.has_votes" ng-cloak>
|
||||||
|
<div class="input-group">
|
||||||
|
<span><translate>Required majority</translate>: </span>
|
||||||
|
<select ng-model="method"
|
||||||
|
ng-change="recalculateMajorities(method)" ng-options="option.value as option.display_name | translate for option in methodChoices" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<!-- election result of poll -->
|
<!-- election result of poll -->
|
||||||
<table ng-if="poll.has_votes" class="table table-bordered table-striped minimumTable">
|
<table ng-if="poll.has_votes" class="table table-bordered table-striped minimumTable spacer">
|
||||||
<tr>
|
<tr>
|
||||||
<th translate>Candidates
|
<th translate>Candidates
|
||||||
<th translate>Votes</th>
|
<th translate>Votes
|
||||||
|
<th ng-if="method != 'disabled'" translate>Quorum reached
|
||||||
|
</th>
|
||||||
<!-- candidates (poll options) -->
|
<!-- candidates (poll options) -->
|
||||||
<tr ng-repeat="option in poll.options">
|
<tr ng-repeat="option in poll_options_with_majorities">
|
||||||
<!-- candidate name -->
|
<!-- candidate name -->
|
||||||
<td>
|
<td>
|
||||||
<span os-perms="assignments.can_manage">
|
<span os-perms="assignments.can_manage">
|
||||||
@ -218,6 +226,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<td ng-if="method != 'disabled'">
|
||||||
|
<span ng-if="option.isReached < 0" class="text-danger" translate> Quorum not reached, {{ -option.isReached }} votes missing.</span>
|
||||||
|
<span ng-if="option.isReached >= 0" class="text-success" translate> Quorum reached, {{ option.isReached }} votes more than needed.</span>
|
||||||
|
|
||||||
<!-- total votes (valid/invalid/casts) -->
|
<!-- total votes (valid/invalid/casts) -->
|
||||||
<tr>
|
<tr>
|
||||||
@ -239,6 +250,9 @@
|
|||||||
{{ poll.getVote('votescast').value }}
|
{{ poll.getVote('votescast').value }}
|
||||||
{{ poll.getVote('votescast').percentStr }}
|
{{ poll.getVote('votescast').percentStr }}
|
||||||
</table>
|
</table>
|
||||||
|
<!-- majority calculation message-->
|
||||||
|
<span class="text-warning" ng-if="calculationError" translate>Calculation impossible</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</uib-tab>
|
</uib-tab>
|
||||||
|
|
||||||
|
@ -740,10 +740,10 @@ angular.module('OpenSlidesApp.motions.site', [
|
|||||||
.controller('MotionPollDetailCtrl', [
|
.controller('MotionPollDetailCtrl', [
|
||||||
'$scope',
|
'$scope',
|
||||||
'MajorityMethodChoices',
|
'MajorityMethodChoices',
|
||||||
'MotionMajority',
|
'MajorityCalculation',
|
||||||
'Config',
|
'Config',
|
||||||
'MotionPollDetailCtrlCache',
|
'MotionPollDetailCtrlCache',
|
||||||
function ($scope, MajorityMethodChoices, MotionMajority, Config, MotionPollDetailCtrlCache) {
|
function ($scope, MajorityMethodChoices, MajorityCalculation, Config, MotionPollDetailCtrlCache) {
|
||||||
// Define choices.
|
// Define choices.
|
||||||
$scope.methodChoices = MajorityMethodChoices;
|
$scope.methodChoices = MajorityMethodChoices;
|
||||||
// TODO: Get $scope.baseChoices from config_variables.py without copying them.
|
// TODO: Get $scope.baseChoices from config_variables.py without copying them.
|
||||||
@ -766,7 +766,7 @@ angular.module('OpenSlidesApp.motions.site', [
|
|||||||
|
|
||||||
// Define result function.
|
// Define result function.
|
||||||
$scope.isReached = function () {
|
$scope.isReached = function () {
|
||||||
return MotionMajority.isReached($scope.base, $scope.method, $scope.poll);
|
return MajorityCalculation.isReached($scope.base, $scope.method, $scope.poll);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Save current values to cache on detroy of this controller.
|
// Save current values to cache on detroy of this controller.
|
||||||
|
@ -38,14 +38,41 @@ angular.module('OpenSlidesApp.poll.majority', [])
|
|||||||
}
|
}
|
||||||
])
|
])
|
||||||
|
|
||||||
.factory('MotionMajority', [
|
.factory('MajorityCalculation', [
|
||||||
'MajorityMethods',
|
'MajorityMethods',
|
||||||
function (MajorityMethods) {
|
function (MajorityMethods) {
|
||||||
return {
|
return {
|
||||||
isReached: function (base, method, votes) {
|
//calculate the base for yes-based multi-option polls
|
||||||
|
options_yes_sum: function(poll){
|
||||||
|
var yes = 0;
|
||||||
|
var error = false;
|
||||||
|
if (poll.options) {
|
||||||
|
_.forEach(poll.options, function(option) {
|
||||||
|
_.forEach(option.votes, function(vote) {
|
||||||
|
if (vote.value == 'Yes' || vote.value == 'Votes'){
|
||||||
|
if (vote.weight >= 0){
|
||||||
|
yes = yes + vote.weight;
|
||||||
|
} else {
|
||||||
|
error = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
if (!error) {
|
||||||
|
return yes;
|
||||||
|
}
|
||||||
|
} else if (poll.yes && poll.yes >= 0) {
|
||||||
|
return poll.yes;
|
||||||
|
} // else: undefined
|
||||||
|
},
|
||||||
|
|
||||||
|
// returns 0 or positive integer if quorum is reached or surpassed
|
||||||
|
// sum (optional): a different base to calculate with
|
||||||
|
isReached: function (base, method, votes, sum) {
|
||||||
var methodFunction = MajorityMethods[method];
|
var methodFunction = MajorityMethods[method];
|
||||||
var yes = parseInt(votes.yes);
|
var yes = parseInt(votes.yes);
|
||||||
var no = parseInt(votes.no);
|
var no = parseInt(votes.no);
|
||||||
|
var alt_base = parseInt(sum);
|
||||||
var abstain = parseInt(votes.abstain);
|
var abstain = parseInt(votes.abstain);
|
||||||
var valid = parseInt(votes.votesvalid);
|
var valid = parseInt(votes.votesvalid);
|
||||||
var cast = parseInt(votes.votescast);
|
var cast = parseInt(votes.votescast);
|
||||||
@ -53,6 +80,9 @@ angular.module('OpenSlidesApp.poll.majority', [])
|
|||||||
var isValid = function (vote) {
|
var isValid = function (vote) {
|
||||||
return !isNaN(vote) && vote >= 0;
|
return !isNaN(vote) && vote >= 0;
|
||||||
};
|
};
|
||||||
|
if (isValid(alt_base) && isValid(yes)) {
|
||||||
|
result = methodFunction(yes, alt_base);
|
||||||
|
} else {
|
||||||
switch (base) {
|
switch (base) {
|
||||||
case 'YES_NO_ABSTAIN':
|
case 'YES_NO_ABSTAIN':
|
||||||
if (isValid(yes) && isValid(no) && isValid(abstain)) {
|
if (isValid(yes) && isValid(no) && isValid(abstain)) {
|
||||||
@ -76,8 +106,9 @@ angular.module('OpenSlidesApp.poll.majority', [])
|
|||||||
break;
|
break;
|
||||||
// case 'DISABLED': result remains undefined
|
// case 'DISABLED': result remains undefined
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return result;
|
return result;
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
]);
|
]);
|
||||||
|
@ -144,7 +144,7 @@ class TestConfigDBQueries(TestCase):
|
|||||||
|
|
||||||
TODO: The last 57 requests are a bug.
|
TODO: The last 57 requests are a bug.
|
||||||
"""
|
"""
|
||||||
with self.assertNumQueries(61):
|
with self.assertNumQueries(62):
|
||||||
self.client.get(reverse('config-list'))
|
self.client.get(reverse('config-list'))
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user