c32d585dd53cc441d8e36d7a72fe74be9edaec90
[selectricity-live] / app / controllers / quickvote_controller.rb
1 # Selectricity: Voting Machinery for the Masses
2 # Copyright (C) 2007, 2008 Benjamin Mako Hill <mako@atdot.cc>
3 # Copyright (C) 2007 Massachusetts Institute of Technology
4 #
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License as
7 # published by the Free Software Foundation, either version 3 of the
8 # License, or (at your option) any later version.
9 #
10 # This program is distributed in the hope that it will be useful, but
11 # WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13 # Affero General Public License for more details.
14 #
15 # You should have received a copy of the GNU Affero General Public
16 # License along with this program.  If not, see
17 # <http://www.gnu.org/licenses/>.
18
19 class QuickvoteController < ApplicationController
20   helper :sparklines
21   layout 'main'
22   require_dependency "quick_voter"
23   require_dependency "quick_vote"
24   require_dependency "vote"
25   require_dependency "election"
26   
27   #############################################################
28   # the following methods pertain to creating quickvotes
29   #############################################################
30
31   def create
32     if params[:quickvote]
33       @quickvote = QuickVote.new(params[:quickvote])
34
35       # check to see if any of the advanced options have been changed
36       new_qv = QuickVote.new
37       if @quickvote.election_method != new_qv.election_method \
38         or @quickvote.enddate.day != new_qv.enddate.day \
39         or @quickvote.viewable != new_qv.viewable \
40         or @quickvote.notices != new_qv.notices
41         show_advanced = true
42       end
43     end
44
45     show_advanced ||= false
46
47     if params[:quickvote]
48
49       # store the candidate grabbed through ajax and stored in flash
50       @quickvote.candidate_names = flash[:candidate_names]
51       @quickvote.description=@quickvote.description
52
53       #record who created the quickvote so that person can monitor it easily
54       @quickvote.quickuser = session.session_id
55
56       #Give registered users additional QuickVote functionality 
57       @quickvote.user_id = session[:user][:id] if session[:user]
58
59       # try to save, if it fails, show the page again (the flash should
60       # still be intact
61       if @quickvote.save
62         @quickvote = @quickvote.reload
63         # blank sidebar and show the success page
64         @sidebar_content = ''
65         render :action => 'success'
66       else
67         # render the sidebar
68         @sidebar_content = render_to_string(:partial => 'create_sidebar',
69           :locals => {:show_advanced => show_advanced})
70         flash.keep(:candidate_names)
71       end 
72
73     else
74       # if we don't have a quickvote param, it means that the person
75       # here has not been hitting this page and we can clear any
76       # candidate_names list in the flash
77       flash.delete(:candidate_names) if flash.has_key?(:candidate_names)
78       @quickvote = QuickVote.new
79       @sidebar_content = render_to_string(:partial => 'create_sidebar',
80         :locals => {:show_advanced => show_advanced})
81     end
82
83   end
84
85   def add_candidate
86     candidate_name = params[:ajax][:newcandidate]
87     unless candidate_name.strip.empty?
88       if flash.has_key?(:candidate_names) \
89         and flash[:candidate_names].instance_of?(Array) 
90         unless flash[:candidate_names].index(candidate_name)
91           flash[:candidate_names] << candidate_name
92         end
93      else
94        flash[:candidate_names] = [ candidate_name ]
95       end
96     end
97     flash.keep(:candidate_names)
98     render :partial => 'candidate_list'
99   end
100  
101   #############################################################
102   # the following methods pertain to *voting* in the quickvotes
103   #############################################################
104
105   def index
106     @election = QuickVote.ident_to_quickvote(params[:ident])
107     # if the person has specified an election, we show them the voting
108     # page. otherwise, we redirect back to main the page
109     if @election
110       # look to see that the voter has been created and has voted in
111       # this election, and has confirmed their vote
112       @voter = QuickVoter.find(:all,
113         :conditions => ["session_id = ? and election_id = ?",
114                         session.session_id, @election.id])[0]
115
116       # if the voter has not voted we destroy them
117       if @voter and not @voter.voted?
118         @voter.destroy
119         @voter = nil
120       end
121
122       # if the voter does not exist or has has been destroyed, lets
123       # create a new one
124       unless @voter
125         # create a new voter and populate it
126         @voter = QuickVoter.new
127         @voter.election = @election
128         @voter.session_id = session.session_id
129               
130         # create new vote and make it the defaulted sorted list
131         @voter.vote = Vote.new
132               @voter.save
133               @voter.vote.set_defaults!
134               @voter.reload
135       end
136     else
137       redirect_to :controller => 'front'
138     end
139   end
140
141   def confirm
142     
143     # we need the election to verify that we have the right voter
144     election = QuickVote.ident_to_quickvote(params[:ident])
145
146     # find out who the voter is for this election
147     @voter = QuickVoter.find(:all,
148       :conditions => ["session_id = ? and election_id = ?", 
149                       session.session_id, election.id])[0]
150    
151     if not @voter
152       # we have not seen this  voter before. something is wrong, try
153       # again
154       redirect_to quickvote_url( :ident => params[:ident] ) 
155       
156     elsif @voter.voted? 
157       # this person has already voted, we try again
158       flash[:notice] = "You have already voted!"
159       redirect_to quickvote_url( :ident => params[:ident] )
160       
161     else
162       
163       # record the ip address for posterity
164       @voter.ipaddress = request.env["REMOTE_ADDR"]
165       @voter.save
166       
167       # toggle the confirmation bit      
168       @voter.vote.confirm!
169      
170       @voter.reload
171       render :action => 'thanks'
172     end
173   end
174  
175   def change
176     voter = QuickVoter.find(:all, :conditions => ["session_id = ?",
177                                                   session.session_id])[0]
178     voter.destroy
179     redirect_to quickvote_url( :ident => params[:ident] )
180   end
181                 
182   def list_voters
183     @map = GMap.new("map_div_id") 
184     @map.control_init(:large_map => true, :map_type => true) 
185     center = nil
186     @election=QuickVote.ident_to_quickvote(params[:id])
187     @election.voters.each do |voter|
188       next unless voter.ipaddress
189       location=nil
190       if defined? Cache and location=Cache.get("GEO:#{voter.ipaddress}")
191       elsif defined? Cache
192         location = GeoKit::Geocoders::IpGeocoder.geocode(voter.ipaddress)
193         Cache.set "GEO:#{voter.ipaddress}", location
194       else
195         location = GeoKit::Geocoders::IpGeocoder.geocode(voter.ipaddress)
196       end
197       next unless location.lng and location.lat
198
199       unless center
200         center = [location.lat, location.lng]
201         @map.center_zoom_init(center, 4)
202       end
203
204       marker = GMarker.new([location.lat,location.lng],
205                            :title => "Voter",
206                            :info_window => (voter.ipaddress or "unknown"))
207       @map.overlay_init(marker)
208     end
209   end
210
211   ###############################################################
212   # the following method pertains to displaying the results of a
213   # quickvote
214   ###############################################################
215
216   def results
217     unless @election = QuickVote.ident_to_quickvote(params[:ident])
218       flash[:notice] = "Cannot find quickvote #{params[:ident]}."
219       redirect_to :controller => 'front'
220       return
221     end
222     if @election.viewable == 0 && @election.active == 1
223       render :action => 'not_viewable' and return
224     end
225     @results = @election.results
226     @candidates = {}
227     @election.candidates.each {|c| @candidates[c.id] = c}
228     @names = @election.names_by_id
229     @sidebar_content = render_to_string :partial => 'results_sidebar'
230   end
231   
232   def my_quickvotes
233     @myqvs = QuickVote.find(:all, :conditions => ["quickuser = ?",
234                                 session.session_id])
235   end
236   
237 end
238

Benjamin Mako Hill || Want to submit a patch?