]> projects.mako.cc - selectricity/commitdiff
Merge from head -- breakage
authorJohn Dong <jdong@mit.edu>
Wed, 22 Aug 2007 04:08:24 +0000 (00:08 -0400)
committerJohn Dong <jdong@mit.edu>
Wed, 22 Aug 2007 04:08:24 +0000 (00:08 -0400)
16 files changed:
app/controllers/graph_controller.rb
app/controllers/quickvote_controller.rb
app/models/election.rb
app/views/quickvote/_pref_tables.rhtml [moved from app/views/quickvote/_pref_table.rhtml with 50% similarity]
app/views/quickvote/_result_approval.rhtml [new file with mode: 0644]
app/views/quickvote/_result_borda.rhtml [new file with mode: 0644]
app/views/quickvote/_result_condorcet.rhtml [new file with mode: 0644]
app/views/quickvote/_result_plurality.rhtml [new file with mode: 0644]
app/views/quickvote/_result_runoff.rhtml [new file with mode: 0644]
app/views/quickvote/_result_ssd.rhtml [new file with mode: 0644]
app/views/quickvote/_victories_ties.rhtml [deleted file]
app/views/quickvote/results.rhtml
config/environment.rb
db/create.sql
db/schema.rb
lib/rubyvote/condorcet.rb

index 442bb65db975b9e4b650bd18815b560561595277..69577c6df84e1cf80566748993209eb145965027 100644 (file)
@@ -99,6 +99,7 @@ class GraphController < ApplicationController
     legend = Hash.new   
     alldata, labels = get_positions_info(@election)    
     @election.results unless @election.condorcet_result || @election.ssd_result
+    ranked_candidates = @election.condorcet_result.ranked_candidates.flatten
     
     names = Hash.new
     candidates = @election.candidates.sort.collect {|candidate| candidate.id}
@@ -106,7 +107,6 @@ class GraphController < ApplicationController
       names[candidate]= (Candidate.find(candidate)).name
     end
     
-    ranked_candidates = @election.condorcet_result.ranked_candidates.flatten
     ranked_candidates.each_with_index \
     {|candidate, index| legend[names[candidate]] = alldata[index]}
     
@@ -119,6 +119,24 @@ class GraphController < ApplicationController
     send_data(*graph.output) 
   end
   
+  def plurality_pie
+    @election = Election.find(params[:id])
+    @election.results unless @election.plurality_result || @election.approval_result
+    votes = @election.votes.size
+    data = Hash.new
+    names = @election.names_by_id
+    
+    @election.plurality_result.points.each do |candidate, votes|
+      data[names[candidate]] = votes
+    end
+     
+    pie = GruffGraff.new ( :graph_type => Gruff::Pie,
+                           :title => "Percentage of First Place Votes",
+                           :data => data)
+    send_data(*pie.output)
+                           
+  end
+  
  private 
   def get_positions_info(election)
     buckets = Hash.new
index bb60b4552b029efff74be54bba9282c7599a671f..54784ef46e735749238ea4fc8617bd261cb72d80 100644 (file)
@@ -161,7 +161,7 @@ class QuickvoteController < ApplicationController
       redirect_to :controller => 'site'
       return
     end
-    @election.results
+    @results = @election.results
     @candidates = {}
     @election.candidates.each {|c| @candidates[c.id] = c}
   end
index d76c13f633cd7ddb7b48ca560deb60d54f266d4a..59a10bfbe601387eac1b72de2c9fcec03c99560c 100644 (file)
@@ -5,13 +5,24 @@ class Election < ActiveRecord::Base
   belongs_to :user
   validates_presence_of :name, :description
   
+  #validate that method is one of the listed election types
+  
   attr_reader :plurality_result
   attr_reader :approval_result
   attr_reader :condorcet_result
   attr_reader :ssd_result
   attr_reader :borda_result
-
+  
   require 'date'
+  
+  def other_methods
+    if election_method
+      @other_methods = ELECTION_TYPES.reject {|i| i == election_method}
+    else
+      @other_methods = nil
+    end
+    @other_methods
+  end
 
   def startdate
     read_attribute( :startdate ) || Time.now
@@ -100,15 +111,19 @@ class Election < ActiveRecord::Base
       preference_tally << voter.vote.rankings.sort.collect \
         { |ranking| ranking.candidate.id }
     end
+    
     @plurality_result = PluralityVote.new(plurality_tally).result
     @approval_result = ApprovalVote.new(approval_tally).result
     @condorcet_result = PureCondorcetVote.new(preference_tally).result
     @ssd_result = CloneproofSSDVote.new(preference_tally).result
     @borda_result = BordaVote.new(preference_tally).result
-    #@runoff_result = InstantRunoffVote.new(preference_tally).result
     
-    nil # to stay consistent
-  end
+    { 'plurality' => @plurality_result,
+      'approval' => @approval_result,
+      'condorcet' => @condorcet_result,
+      'ssd' => @ssd_result,
+      'borda' => @borda_result }
+    end
   
   def names_by_id
     names = Hash.new
similarity index 50%
rename from app/views/quickvote/_pref_table.rhtml
rename to app/views/quickvote/_pref_tables.rhtml
index 011fcbe8ac2d29aa7b2147797e249f99c3b57389..66e0cbf2df1fbf7870581ca6ed849b66db2c7f21 100644 (file)
@@ -1,17 +1,19 @@
-<% candidates = @election.candidates.sort.collect {|candidate| candidate.id}-%>
+<% candidates = @election.condorcet_result.ranked_candidates.flatten -%>
 <% voters = @election.voters.size %>
 <% matrix = @election.condorcet_result.matrix %>
-<% names = Hash.new -%>
-<% candidates.each do |candidate| -%>
-       <%names[candidate] = Candidate.find(candidate).name -%>
-<% end -%>
+<% victories = @election.condorcet_result.victories_and_ties %>
+<% names = @election.names_by_id %>
+
+
+<!-- This table shows how many times each choice was ranked above the other, 
+       with percentages-->
 <table class="voterbox">
   <tr>
        <td> </td>
        <% candidates.each do |candidate| -%>
          <th><%=h names[candidate] -%></th>
   <% end -%>
-</tr>
+ </tr>
 
 <% candidates.each do |winner| -%>
   <tr>
  </tr>
 <% end -%>
 </table>
+
+<!-- This table generates a margin of victory -->
+<table class="voterbox">
+  <% candidates.each do |victor| %>
+  <tr>
+    <th><%=h names[victor] %></th>
+       <% victories[victor].keys.each do |loser| %>
+       <% margin = victories[victor][loser]%>
+       <td><%=h names[loser] %> 
+           <% if margin == 0%>
+                 Tied!
+               <% else -%>
+                 (<%= margin%>)
+               <% end -%>
+                 </td>
+       <% end -%>
+  </tr>
+  <% end -%>
+</table>
+
diff --git a/app/views/quickvote/_result_approval.rhtml b/app/views/quickvote/_result_approval.rhtml
new file mode 100644 (file)
index 0000000..12c7958
--- /dev/null
@@ -0,0 +1,17 @@
+<div class="resultbox">
+<h3>Approval Result</h3>
+<p><font size="-1">(This algorithm assumes that top two choices are "approved.")</font></p>
+
+<div class="rbmoreinfo">
+<h4>About Approval Voting</h4>
+
+<p><%= link_to "Approval voting",
+"http://en.wikipedia.org/wiki/Approval_voting" %> is a voting system in
+which each voter can vote for as many or as few candidates as the voter
+chooses.  Approval voting is a limited form of range voting, where the
+range that voters are allowed to express is extremely constrained:
+accept or not.</p>
+
+</div>
+
+</div>
\ No newline at end of file
diff --git a/app/views/quickvote/_result_borda.rhtml b/app/views/quickvote/_result_borda.rhtml
new file mode 100644 (file)
index 0000000..97d82e5
--- /dev/null
@@ -0,0 +1,16 @@
+<div class="resultbox">
+<h3>Borda Count Results</h3>
+
+<div class="rbmoreinfo">
+<h4>About Borda Count</h4>
+
+<p><%= link_to "Borda count",
+"http://en.wikipedia.org/wiki/Borda_count" %> 
+is an election method in which voters rank
+candidates in order of preference. The Borda count determines the winner
+of an election by giving each candidate a certain number of points
+corresponding to the position in which he or she is ranked by each
+voter. Once all votes have been counted the candidate with the most
+points is the winner.</p>
+</div>
+</div>
\ No newline at end of file
diff --git a/app/views/quickvote/_result_condorcet.rhtml b/app/views/quickvote/_result_condorcet.rhtml
new file mode 100644 (file)
index 0000000..4279bbd
--- /dev/null
@@ -0,0 +1,18 @@
+<div class="resultbox">
+<h3>Simple Condorcet Results</h3>
+
+<div class="rbmoreinfo">
+<h4>About Simple Cordorcet Voting</h4>
+
+<p><%= link_to "Condorcet",
+"http://en.wikipedia.org/wiki/Condorcet_method" %> allows voters to rank
+candidates in order of preference.  If there is a choice whom voters
+prefer to each other choice when compared to one at a time, that choice
+will be the winner.</p>
+
+<p>There is a family of Condorcet methods. This method is referred to as
+"Simple Condorcet" to distinguish it from the Schulze method which is
+another Condorcet system.</p>
+
+</div>
+</div>
\ No newline at end of file
diff --git a/app/views/quickvote/_result_plurality.rhtml b/app/views/quickvote/_result_plurality.rhtml
new file mode 100644 (file)
index 0000000..121c1dc
--- /dev/null
@@ -0,0 +1,17 @@
+<div class="resultbox">
+<h3>Plurality Results</h3>
+
+<div class="rbmoreinfo">
+<h4>About Plurality Voting</h4>
+
+<p><%= link_to "Plurality voting",
+"http://en.wikipedia.org/wiki/Plurality_electoral_system" %> selects the
+winner who has received the most "number one" votes, regardless of
+whether or not he or she has a majority of votes.</p>
+
+<p>Plurality voting is also variously referred to as, "first past the
+post," "winner-take-all," "majoritarian" or "simple majority"
+voting.</p>
+
+</div>
+</div>
\ No newline at end of file
diff --git a/app/views/quickvote/_result_runoff.rhtml b/app/views/quickvote/_result_runoff.rhtml
new file mode 100644 (file)
index 0000000..f242290
--- /dev/null
@@ -0,0 +1,20 @@
+<div class="resultbox">
+<h3>Instant Runoff (IRV) Results</h3>
+
+<div class="rbmoreinfo">
+<h4>About Instant Runoff Voting</h4>
+
+<p><%= link_to "Instant runoff voting",
+"http://en.wikipedia.org/wiki/Instant_Runoff_Voting" %> is an electoral
+system in which voters rank candidates in order of preference. In an IRV
+election, if no candidate receives an overall majority of first
+preferences the candidates with fewest votes are eliminated one by one,
+and their votes transferred according to their second and third
+preferences (and so on), until one candidate achieves a majority.</p>
+
+<p>Instant-runoff voting (IRV) is also known as the Alternative Vote (AV) and
+by several other names.</p>
+
+</div>
+
+</div>
\ No newline at end of file
diff --git a/app/views/quickvote/_result_ssd.rhtml b/app/views/quickvote/_result_ssd.rhtml
new file mode 100644 (file)
index 0000000..436dd67
--- /dev/null
@@ -0,0 +1,17 @@
+<div class="mainresultbox">
+<h3>Schulze Method Results</h3>
+
+<div class="rbmoreinfo">
+<h4>About the Schulze Method</h4>
+
+<p>The <%= link_to "Schulze method",
+"http://en.wikipedia.org/wiki/Schulze_method" %> is a preferential
+voting system. It is based on the Condorcet method but includes a set of
+methods for resolving "circular" defeats.</p>
+
+<p>The Schulze method is also known as Schwartz Sequential Dropping
+(SSD), Cloneproof Schwartz Sequential Dropping (CSSD), Beatpath Method,
+Beatpath Winner, Path Voting, and Path Winner.</p>
+</div>
+
+</div>
diff --git a/app/views/quickvote/_victories_ties.rhtml b/app/views/quickvote/_victories_ties.rhtml
deleted file mode 100644 (file)
index 993caa8..0000000
+++ /dev/null
@@ -1,17 +0,0 @@
-<% victories, tied = @election.condorcet_result.victories_and_ties %>
-<% names = @election.names_by_id %>
-<% %>
-<table class="voterbox">
-  <% victories.keys.each do |victor| %>
-  <tr>
-    <th><%=h names[victor] %></th>
-       <% victories[victor].keys.each do |loser| %>
-       <td><%=h names[loser] %> (<%= victories[victor][loser] %>)</td>
-       <% end -%>
-  </tr>
-  <% end -%>
-</table>
-               
-       
-       
-
index 6c1130f5e56b560ecd03989baf9b6b69406bce72..9727d81289e6690597f298f4d5a0d8bb4a545e5e 100644 (file)
 
 <h2>Winners</h2>
 
-<div class="mainresultbox">
-<h3>Schulze Method Results</h3>
-<%= render :partial => 'result', :object => @election.ssd_result %>
+<%= render :partial => 'result_' + @election.election_method,
+           :object => @results[@election.election_method] %>
 
-<div class="rbmoreinfo">
-<h4>About the Schulze Method</h4>
 
-<p>The <%= link_to "Schulze method",
-"http://en.wikipedia.org/wiki/Schulze_method" %> is a preferential
-voting system. It is based on the Condorcet method but includes a set of
-methods for resolving "circular" defeats.</p>
-
-<p>The Schulze method is also known as Schwartz Sequential Dropping
-(SSD), Cloneproof Schwartz Sequential Dropping (CSSD), Beatpath Method,
-Beatpath Winner, Path Voting, and Path Winner.</p>
-</div>
-
-</div>
-
-<div class="resultbox">
-<h3>Plurality Results</h3>
-<%= render :partial => 'result', :object => @election.plurality_result %>
-
-<div class="rbmoreinfo">
-<h4>About Plurality Voting</h4>
-
-<p><%= link_to "Plurality voting",
-"http://en.wikipedia.org/wiki/Plurality_electoral_system" %> selects the
-winner who has received the most "number one" votes, regardless of
-whether or not he or she has a majority of votes.</p>
-
-<p>Plurality voting is also variously referred to as, "first past the
-post," "winner-take-all," "majoritarian" or "simple majority"
-voting.</p>
-
-</div>
-</div>
-
-<div class="resultbox">
-<h3>Approval Result</h3>
-<p><font size="-1">(This algorithm assumes that top two choices are "approved.")</font></p>
-<%= render :partial => 'result', :object => @election.approval_result %>
-
-<div class="rbmoreinfo">
-<h4>About Approval Voting</h4>
-
-<p><%= link_to "Approval voting",
-"http://en.wikipedia.org/wiki/Approval_voting" %> is a voting system in
-which each voter can vote for as many or as few candidates as the voter
-chooses.  Approval voting is a limited form of range voting, where the
-range that voters are allowed to express is extremely constrained:
-accept or not.</p>
-
-</div>
-
-</div>
-
-<div class="resultbox">
-<h3>Simple Condorcet Results</h3>
-<%= render :partial => 'result', :object => @election.condorcet_result %>
-
-<div class="rbmoreinfo">
-<h4>About Simple Cordorcet Voting</h4>
-
-<p><%= link_to "Condorcet",
-"http://en.wikipedia.org/wiki/Condorcet_method" %> allows voters to rank
-candidates in order of preference.  If there is a choice whom voters
-prefer to each other choice when compared to one at a time, that choice
-will be the winner.</p>
-
-<p>There is a family of Condorcet methods. This method is referred to as
-"Simple Condorcet" to distinguish it from the Schulze method which is
-another Condorcet system.</p>
-
-</div>
-</div>
-
-<div class="resultbox">
-<h3>Borda Count Results</h3>
-<%= render :partial => 'result', :object => @election.borda_result %>
-
-<div class="rbmoreinfo">
-<h4>About Borda Count</h4>
-
-<p><%= link_to "Borda count",
-"http://en.wikipedia.org/wiki/Borda_count" %> 
-is an election method in which voters rank
-candidates in order of preference. The Borda count determines the winner
-of an election by giving each candidate a certain number of points
-corresponding to the position in which he or she is ranked by each
-voter. Once all votes have been counted the candidate with the most
-points is the winner.</p>
-</div>
-</div>
-
-<div class="resultbox">
-<h3>Instant Runoff (IRV) Results</h3>
-
-<div class="rbmoreinfo">
-<h4>About Instant Runoff Voting</h4>
-
-<p><%= link_to "Instant runoff voting",
-"http://en.wikipedia.org/wiki/Instant_Runoff_Voting" %> is an electoral
-system in which voters rank candidates in order of preference. In an IRV
-election, if no candidate receives an overall majority of first
-preferences the candidates with fewest votes are eliminated one by one,
-and their votes transferred according to their second and third
-preferences (and so on), until one candidate achieves a majority.</p>
-
-<p>Instant-runoff voting (IRV) is also known as the Alternative Vote (AV) and
-by several other names.</p>
-
-</div>
+<h3> Other Voting Methods </h3>
+<% for result_type in @election.other_methods %>
+<%= render :partial => 'result_' + result_type, 
+           :object => @results[result_type] %>
+<% end %>
 
-</div>
 
 <div class="clearbox"></div>
 
@@ -180,10 +75,10 @@ by several other names.</p>
 <% end %>
 </table>
 
-<%= render :partial => 'victories_ties' %>
-<%= render :partial => 'pref_table' %>
+<%= render :partial => 'pref_tables' %>
 
 <%=image_tag( graph_url( :action => 'votes_per_interval', :id => @election ))%>
 <br />
 <%= image_tag( graph_url( :action => 'borda_bar', :id => @election ) ) %><br />
-<%= image_tag( graph_url( :action => 'choices_positions', :id => @election ) ) %>
+<%= image_tag( graph_url( :action => 'choices_positions', :id => @election ) ) %><br />
+<%= image_tag(graph_url( :action => 'plurality_pie', :id => @election ) )%>
index e071a1437bf25b6c24074331c4f9d0a9efe09cc0..571638d14a3fc1a0ccefbce34442f8cbc1fa8fc0 100644 (file)
@@ -63,10 +63,12 @@ MAIL_CONFIG = { :from => 'Selectricity <info@selectricity.media.mit.edu>'}
 
 require 'uniq_token'
 require 'randarray'
-require 'rubyvote'
 require 'gruff'
 require 'sparklines'
 
+require 'rubyvote'
+ELECTION_TYPES = %w(ssd plurality approval condorcet borda)
+
 class String
   # alternate capitalization method that does not lowercase the rest of
   # the string -- which is almost never what I want
index 0a2f4b5b9dc3bca3ae9dd5d60ab57590e310014e..7b3fef5600b061a043c5c3972abe066aa75df79f 100644 (file)
@@ -11,6 +11,7 @@ create table elections (
  enddate datetime NOT NULL, 
  active tinyint NOT NULL DEFAULT 0,
  user_id int NULL,
+ election_method varchar(100) DEFAULT 'ssd',
  `type` varchar(100) NOT NULL,
  primary key (id),
  constraint fk_user_election foreign key (user_id) references users(id)
index 45d7b416a2eb0a25c752c23b3ddfe74920236c8f..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,73 +0,0 @@
-# This file is autogenerated. Instead of editing this file, please use the
-# migrations feature of ActiveRecord to incrementally modify your database, and
-# then regenerate this schema definition.
-
-ActiveRecord::Schema.define() do
-
-  create_table "candidates", :force => true do |t|
-    t.column "election_id",      :integer,                                :null => false
-    t.column "name",             :string,  :limit => 100, :default => "", :null => false
-    t.column "description",      :text
-    t.column "picture_filename", :string,  :limit => 200
-    t.column "picture_data",     :binary
-    t.column "picture_type",     :string,  :limit => 100
-  end
-
-  create_table "elections", :force => true do |t|
-    t.column "name",        :string,   :limit => 100, :default => "", :null => false
-    t.column "description", :text,                    :default => "", :null => false
-    t.column "anonymous",   :integer,  :limit => 4,   :default => 1,  :null => false
-    t.column "startdate",   :datetime
-    t.column "enddate",     :datetime,                                :null => false
-    t.column "active",      :integer,  :limit => 4,   :default => 0,  :null => false
-    t.column "user_id",     :integer
-    t.column "type",        :string,   :limit => 100, :default => "", :null => false
-  end
-
-  add_index "elections", ["user_id"], :name => "fk_user_election"
-
-  create_table "rankings", :force => true do |t|
-    t.column "vote_id",      :integer
-    t.column "candidate_id", :integer
-    t.column "rank",         :integer
-  end
-
-  create_table "tokens", :force => true do |t|
-    t.column "token",   :string,  :limit => 100, :default => "", :null => false
-    t.column "vote_id", :integer,                                :null => false
-  end
-
-  add_index "tokens", ["vote_id"], :name => "fk_vote_token"
-
-  create_table "users", :force => true do |t|
-    t.column "login",                     :text
-    t.column "ip",                        :text,                   :default => "", :null => false
-    t.column "email",                     :text
-    t.column "crypted_password",          :string,   :limit => 40
-    t.column "salt",                      :string,   :limit => 40
-    t.column "created_at",                :datetime
-    t.column "updated_at",                :datetime
-    t.column "remember_token",            :text
-    t.column "remember_token_expires_at", :datetime
-  end
-
-  create_table "voters", :force => true do |t|
-    t.column "email",       :string,  :limit => 100
-    t.column "password",    :string,  :limit => 100
-    t.column "contacted",   :integer, :limit => 4,   :default => 0, :null => false
-    t.column "election_id", :integer,                               :null => false
-    t.column "session_id",  :string,  :limit => 32
-    t.column "ipaddress",   :string,  :limit => 32
-  end
-
-  add_index "voters", ["election_id"], :name => "fk_election_voter"
-
-  create_table "votes", :force => true do |t|
-    t.column "voter_id",  :integer
-    t.column "confirmed", :integer,  :limit => 4, :default => 0, :null => false
-    t.column "time",      :datetime
-  end
-
-  add_index "votes", ["voter_id"], :name => "fk_vote_voter"
-
-end
index d0210efe19cfbf4fd09f9feb9dc82e4678583b74..0c6fd5647bd921e5543838b42ddc297b57ca72cd 100644 (file)
@@ -123,33 +123,21 @@ class CondorcetResult < ElectionResult
   end
   
   def victories_and_ties
-    victors = Array.new
-    ties = Array.new
-    victories = Hash.new
+    victories_ties = {}
     candidates = @matrix.keys.sort
     
     candidates.each do |candidate|
       candidates.each do |challenger|
         next if candidate == challenger
         diff = @matrix[candidate][challenger] - @matrix[challenger][candidate]
-        if diff > 0 
-          victors << [candidate, challenger, diff]
-        elsif diff == 0 && ties.include?([challenger, candidate]) == false
-          ties << [candidate, challenger] 
+        victories_ties[candidate] = {} unless victories_ties.key?(candidate)
+        if diff >= 0
+          victories_ties[candidate][challenger] = diff
         end
       end
-    end  
-    
-    victors.each do |list|
-      if victories.has_key?(list[0])
-        victories[list[0]][list[1]] = list[2]       
-      else
-        victories[list[0]] = Hash.new
-        victories[list[0]][list[1]] = list[2]
-      end
     end
     
-    return victoriesties    
+    return victories_ties    
   end
 
   def ranked_candidates

Benjamin Mako Hill || Want to submit a patch?