annotate vendor/plugins/redmine_bibliography/app/controllers/publications_controller.rb @ 677:ec6c2f6a33c4 feature_36

Transplant rev fccacd8505e3 from live branch
author Chris Cannam
date Tue, 13 Sep 2011 17:18:00 +0100
parents 4fcd6ca1825c
children 31739bb8e563 c3e6c51170fa
rev   line source
luisf@613 1 # -*- coding: utf-8 -*-
luis@385 2 # vendor/plugins/redmine_bibliography/app/controllers/publications_controller.rb
luis@385 3
luis@328 4 class PublicationsController < ApplicationController
luis@425 5 unloadable
luis@457 6
luis@560 7 model_object Publication
Chris@677 8 before_filter :find_model_object, :except => [:new, :create, :index, :get_bibtex_required_fields, :autocomplete_for_project, :add_author, :sort_author_order, :autocomplete_for_author, :get_user_info ]
luis@631 9 before_filter :find_project_by_project_id, :authorize, :only => [ :edit, :new, :update, :create ]
luis@519 10
luis@404 11 def new
luis@539 12 find_project_by_project_id
luis@539 13 @publication = Publication.new
luis@445 14
luis@445 15 # we'll always want a new publication to have its bibtex entry
luis@539 16 @publication.build_bibtex_entry
luis@445 17
luis@446 18 # and at least one author
luis@608 19 # @publication.authorships.build.build_author
chris@647 20 @author_options = [["#{User.current.name} (@#{User.current.mail.partition('@')[2]})", "#{User.current.class.to_s}_#{User.current.id.to_s}"]]
luis@614 21
luis@614 22
luis@519 23 end
luis@409 24
luis@539 25 def create
luisf@613 26 @project = Project.find(params[:project_id])
luisf@613 27
chris@661 28 @author_options = []
luis@614 29
luis@390 30 @publication = Publication.new(params[:publication])
luis@553 31 @publication.projects << @project unless @project.nil?
luis@553 32
luis@448 33 if @publication.save
luis@666 34 @publication.notify_authors_publication_added(@project)
luis@666 35
luis@445 36 flash[:notice] = "Successfully created publication."
luisf@613 37 redirect_to :action => :show, :id => @publication, :project_id => @project
luis@445 38 else
luisf@613 39 render :action => 'new', :project_id => @project
luis@445 40 end
luis@445 41 end
luis@445 42
luis@445 43 def index
luis@538 44 if !params[:project_id].nil?
luis@538 45 find_project_by_project_id
luis@538 46 @project = Project.find(params[:project_id])
luis@538 47 @publications = Publication.find :all, :joins => :projects, :conditions => ["project_id = ?", @project.id]
luis@538 48 else
luis@538 49 @publications = Publication.find :all
luis@538 50 end
luis@445 51 end
luis@445 52
luis@445 53 def new_from_bibfile
luis@391 54 @publication.current_step = session[:publication_step]
luis@409 55
luis@404 56 # contents of the paste text area
luis@404 57 bibtex_entry = params[:bibtex_entry]
luis@384 58
luis@404 59 # method for creating "pasted" bibtex entries
luis@424 60 if bibtex_entry
luis@409 61 parse_bibtex_list bibtex_entry
luis@404 62 end
luis@329 63 end
luis@329 64
Chris@677 65 def get_bibtex_required_fields
Chris@677 66 all_fields = ["editor", "publisher", "chapter", "pages", "volume", "series", "address", "edition", "year", "note", "institution", "type", "number", "month", "journal", "howpublished", "key", "school"]
Chris@677 67
Chris@677 68 fields = Hash.new
Chris@677 69 fields[ 'article' ] = [ 'journal', 'year', 'volume', 'number', 'pages', 'month', 'note' ]
Chris@677 70 fields[ 'book' ] = [ 'editor', 'publisher', 'volume', 'series', 'address', 'edition', 'month', 'year', 'note' ]
Chris@677 71 fields[ 'booklet' ] = [ 'howpublished', 'address', 'year', 'month', 'note', 'key' ]
Chris@677 72 fields[ 'conference' ] = [ 'booktitle', 'year', 'editor', 'pages', 'organization', 'publisher', 'address', 'month', 'note' ]
Chris@677 73 fields[ 'inbook' ] = [ 'editor', 'publisher', 'chapter', 'pages', 'volume', 'series', 'address', 'edition', 'year', 'note' ]
Chris@677 74 fields[ 'incollection' ] = [ 'editor', 'publisher', 'chapter', 'pages', 'volume', 'series', 'address', 'edition', 'year', 'note' ]
Chris@677 75 fields[ 'inproceedings' ] = [ 'booktitle', 'year', 'editor', 'pages', 'organization', 'publisher', 'address', 'month', 'note' ]
Chris@677 76 fields[ 'manual' ] = [ 'organization', 'address', 'edition', 'month', 'year', 'note' ]
Chris@677 77 fields[ 'masterthesis' ] = [ 'school', 'year', 'address', 'month', 'note' ]
Chris@677 78 fields[ 'misc' ] = [ 'howpublished', 'month', 'year', 'note' ]
Chris@677 79 fields[ 'phdthesis' ] = [ 'school', 'year', 'address', 'month', 'note' ]
Chris@677 80 fields[ 'proceedings' ] = [ 'booktitle', 'year', 'editor', 'pages', 'organization', 'publisher', 'address', 'month', 'note' ]
Chris@677 81 fields[ 'techreport' ] = [ 'institution', 'year', 'type', 'number', 'address', 'month', 'note' ]
Chris@677 82 fields[ 'unpublished' ] = [ 'note', 'month', 'year' ]
Chris@677 83
Chris@677 84 entrytype = BibtexEntryType.find(params[:value]).name
Chris@677 85
Chris@677 86 respond_to do |format|
Chris@677 87 format.js {
Chris@677 88 render(:update) {|page|
Chris@677 89 all_fields.each do |field|
Chris@677 90
Chris@677 91
Chris@677 92
Chris@677 93 unless fields[entrytype].include? field
Chris@677 94 page["publication_bibtex_entry_attributes_#{field}"].up('p').hide()
Chris@677 95 else
Chris@677 96 page["publication_bibtex_entry_attributes_#{field}"].up('p').show()
Chris@677 97 end
Chris@677 98 end
Chris@677 99 }
Chris@677 100 }
Chris@677 101 end
Chris@677 102 end
Chris@677 103
luis@467 104 def add_author
luis@467 105 if (request.xhr?)
luis@467 106 render :text => User.find(params[:user_id]).name
luis@467 107 else
luis@467 108 # No? Then render an action.
luis@467 109 #render :action => 'view_attribute', :attr => @name
Chris@677 110 logger.error { "Error while adding Author to publication." }
luis@467 111 end
luis@467 112 end
luis@467 113
luis@547 114 def edit
luis@547 115 find_project_by_project_id unless params[:project_id].nil?
luis@609 116
luis@609 117 @edit_view = true;
luis@609 118
luis@428 119 @publication = Publication.find(params[:id])
luis@626 120 @selected_bibtex_entry_type_id = @publication.bibtex_entry.entry_type
luis@626 121
chris@647 122 @author_options = []
luis@430 123 end
luis@430 124
luis@445 125 def update
luis@448 126 @publication = Publication.find(params[:id])
luis@538 127
chris@661 128 @author_options = []
luis@626 129
luis@544 130 logger.error { "INSIDE THE UPDATE ACTION IN THE PUBLICATION CONTROLLER" }
luis@544 131
luis@430 132 if @publication.update_attributes(params[:publication])
luis@430 133 flash[:notice] = "Successfully updated Publication."
luis@538 134
luis@538 135 if !params[:project_id].nil?
luis@538 136 redirect_to :action => :show, :id => @publication, :project_id => params[:project_id]
luis@538 137 else
luis@538 138 redirect_to :action => :show, :id => @publication
luis@538 139 end
luis@430 140 else
luis@448 141 render :action => 'edit'
luis@448 142 end
luis@328 143 end
luis@328 144
luis@425 145 def show
luis@535 146 find_project_by_project_id unless params[:project_id].nil?
luis@547 147
luis@425 148 if @publication.nil?
luis@579 149 @publications = Publication.all
luis@579 150 render "index", :alert => 'The publication was not found!'
luis@425 151 else
luis@425 152 @authors = @publication.authors
luis@425 153 @bibtext_entry = @publication.bibtex_entry
luis@425 154 end
luis@329 155 end
luis@329 156
luis@406 157 # parse string with bibtex authors
luis@406 158 def parse_authors(authors_entry)
luis@406 159 # in bibtex the authors are always seperated by "and"
luis@407 160 return authors_entry.split(" and ")
luis@406 161 end
luis@406 162
luis@406 163 # parses a list of bibtex
luis@406 164 def parse_bibtex_list(bibtex_list)
luis@406 165 bibliography = BibTeX.parse bibtex_list
luis@406 166
luis@406 167 no_entries = bibliography.data.length
luis@406 168
luis@406 169 # parses the bibtex entries
luis@406 170 bibliography.data.map do |d|
luis@407 171
luis@407 172 if d.class == BibTeX::Entry
luis@407 173 create_bibtex_entry d
luis@407 174 end
luis@406 175 end
luis@407 176 end
luis@407 177
luis@409 178 def create_bibtex_entry(d)
luis@407 179 @publication = Publication.new
luis@407 180 @bentry = BibtexEntry.new
luis@407 181 authors = []
luis@407 182 institution = ""
luis@407 183 email = ""
luis@409 184
luis@407 185 d.fields.keys.map do |field|
luis@407 186 case field.to_s
luis@407 187 when "author"
luis@407 188 authors = parse_authors d[field]
luis@407 189 when "title"
luis@407 190 @publication.title = d[field]
luis@407 191 when "institution"
luis@407 192 institution = d[field]
luis@407 193 when "email"
luis@407 194 email = d[field]
luis@407 195 else
luis@407 196 @bentry[field] = d[field]
luis@407 197 end
luis@407 198 end
luis@406 199
luis@406 200 @publication.bibtex_entry = @bentry
luis@407 201 @publication.save
luis@409 202
luis@424 203 # what is this for???
luis@424 204 # @created_publications << @publication.id
luis@409 205
luis@407 206 # need to save all authors
luis@407 207 # and establish the author-publication association
luis@407 208 # via the authorships table
luis@407 209 authors.each_with_index.map do |authorname, idx|
luis@407 210 author = Author.new(:name => authorname)
luis@407 211 if author.save!
luis@407 212 puts "SAVED"
luis@407 213 else
luis@407 214 puts "NOT SAVED"
luis@406 215 end
luis@406 216
luis@407 217 author.authorships.create!(
luis@444 218 :publication => @publication,
luis@444 219 :institution => institution,
luis@444 220 :email => email,
luis@444 221 :order => idx)
luis@407 222 end
luis@407 223 end
luis@409 224
luis@407 225 # parses the bibtex file
luis@407 226 def parse_bibtex_file
luis@407 227
luis@406 228 end
luis@406 229
luis@444 230 def import
luis@444 231 @publication = Publication.new
luis@444 232
luis@444 233
luis@444 234 end
luis@461 235
luis@461 236 def autocomplete_for_project
luis@461 237 @publication = Publication.find(params[:id])
luis@477 238
luis@464 239 @projects = Project.active.like(params[:q]).find(:all, :limit => 100) - @publication.projects
luis@461 240 logger.debug "Query for \"#{params[:q]}\" returned \"#{@projects.size}\" results"
luis@461 241 render :layout => false
luis@409 242 end
luis@471 243
luis@601 244 def autocomplete_for_author
luis@519 245 @results = []
luis@480 246
luis@596 247 object_id = params[:object_id]
luis@598 248 @object_name = "publications[authorships_attributes][#{object_id}][search_results]"
luis@601 249
Chris@676 250 # cc 20110909 -- revert to like instead of like_unique -- see #289
Chris@676 251 authorships_list = Authorship.like(params[:q]).find(:all, :limit => 100)
luis@480 252 users_list = User.active.like(params[:q]).find(:all, :limit => 100)
luis@480 253
luis@591 254 logger.debug "Query for \"#{params[:q]}\" returned \"#{authorships_list.size}\" authorships and \"#{users_list.size}\" users"
luis@480 255
luis@601 256 @results = users_list
luis@601 257
luis@601 258 # TODO: can be optimized…
luis@601 259 authorships_list.each do |authorship|
luis@601 260 flag = true
luis@601 261
luis@601 262 users_list.each do |user|
luis@601 263 if authorship.name == user.name && authorship.email == user.mail && authorship.institution == user.institution
luis@601 264 Rails.logger.debug { "Rejecting Authorship #{authorship.id}" }
luis@601 265 flag = false
luis@601 266 break
luis@601 267 end
luis@601 268 end
luis@601 269
luis@601 270 @results << authorship if flag
luis@592 271 end
luis@592 272
luis@598 273 render :layout => false
luis@598 274 end
luis@598 275
luis@598 276
luis@598 277 def get_user_info
luis@598 278 object_id = params[:object_id]
luis@598 279 value = params[:value]
luis@598 280 classname = Kernel.const_get(value.split('_')[0])
luis@592 281
luis@598 282 item = classname.find(value.split('_')[1])
luis@598 283
luis@598 284 name_field = "publication_authorships_attributes_#{object_id}_name_on_paper".to_sym
luis@598 285 email_field = "publication_authorships_attributes_#{object_id}_email".to_sym
luis@600 286 institution_field = "publication_authorships_attributes_#{object_id}_institution".to_sym
luis@600 287
luis@600 288 yes_radio = "publication_authorships_attributes_#{object_id}_identify_author_yes".to_sym
luis@598 289
luis@598 290 respond_to do |format|
Chris@677 291 format.js {
luis@598 292 render(:update) {|page|
luis@598 293 page[name_field].value = item.name
luis@598 294 page[email_field].value = item.mail
luis@600 295 page[institution_field].value = item.institution
luis@600 296
luis@600 297 page[yes_radio].checked = true
luis@601 298 page[name_field].readOnly = true
luis@601 299 page[email_field].readOnly = true
luis@601 300 page[institution_field].readOnly = true
luis@598 301 }
luis@598 302 }
luis@598 303 end
luis@477 304 end
luis@471 305
luis@557 306 def sort_author_order
luis@557 307 params[:authorships].each_with_index do |id, index|
luis@557 308 Authorship.update_all(['auth_order=?', index+1], ['id=?', id])
luis@471 309 end
luis@471 310 render :nothing => true
luis@471 311 end
luis@574 312
luis@574 313 def add_project
luis@574 314 @projects = Project.find(params[:publication][:project_ids])
luis@574 315 @publication.projects << @projects
luis@579 316 @project = Project.find(params[:project_id])
luis@574 317
luis@574 318 # TODO luisf should also respond to HTML???
luis@574 319 respond_to do |format|
luis@574 320 format.html { redirect_to :back }
luis@574 321 format.js {
luis@574 322 render(:update) {|page|
luis@574 323 page[:add_project_form].reset
luis@574 324 page.replace_html :list_projects, :partial => 'list_projects'
luis@574 325 }
luis@574 326 }
luis@574 327 end
luis@574 328 end
luis@554 329
luis@579 330
luis@574 331 def remove_project
luis@579 332 @project = Project.find(params[:project_id])
luis@579 333 proj = Project.find(params[:remove_project_id])
luis@554 334
luis@574 335 if @publication.projects.length > 1
luis@579 336 if @publication.projects.exists? proj
luis@579 337 @publication.projects.delete proj if request.post?
luis@554 338 end
luis@554 339 else
luis@579 340 logger.error { "Cannot remove project from publication list" }
luis@554 341 end
luis@554 342
luis@591 343 logger.error { "CURRENT project name#{proj.name} and wanna delete #{@project.name}" }
luis@579 344
luis@579 345 render(:update) {|page|
luis@579 346 page.replace_html "list_projects", :partial => 'list_projects', :id => @publication
luis@579 347 }
luis@554 348 end
luis@579 349
luis@560 350 def destroy
luis@560 351 find_project_by_project_id
luis@560 352
luis@560 353 @publication.destroy
luis@560 354
luis@560 355 flash[:notice] = "Successfully deleted Publication."
luis@560 356 redirect_to :controller => :publications, :action => 'index', :project_id => @project
luis@560 357 end
luis@471 358
luis@538 359 private
luis@478 360
luis@328 361 end