You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

board.rb 3.0KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. # Redmine - project management software
  2. # Copyright (C) 2006-2017 Jean-Philippe Lang
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; either version 2
  7. # of the License, or (at your option) any later version.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  17. class Board < ActiveRecord::Base
  18. include Redmine::SafeAttributes
  19. belongs_to :project
  20. has_many :messages, lambda {order("#{Message.table_name}.created_on DESC")}, :dependent => :destroy
  21. belongs_to :last_message, :class_name => 'Message'
  22. acts_as_tree :dependent => :nullify
  23. acts_as_positioned :scope => [:project_id, :parent_id]
  24. acts_as_watchable
  25. validates_presence_of :name, :description
  26. validates_length_of :name, :maximum => 30
  27. validates_length_of :description, :maximum => 255
  28. validate :validate_board
  29. scope :visible, lambda {|*args|
  30. joins(:project).
  31. where(Project.allowed_to_condition(args.shift || User.current, :view_messages, *args))
  32. }
  33. safe_attributes 'name', 'description', 'parent_id', 'position'
  34. def visible?(user=User.current)
  35. !user.nil? && user.allowed_to?(:view_messages, project)
  36. end
  37. def reload(*args)
  38. @valid_parents = nil
  39. super
  40. end
  41. def to_s
  42. name
  43. end
  44. # Returns a scope for the board topics (messages without parent)
  45. def topics
  46. messages.where(:parent_id => nil)
  47. end
  48. def valid_parents
  49. @valid_parents ||= project.boards - self_and_descendants
  50. end
  51. def reset_counters!
  52. self.class.reset_counters!(id)
  53. end
  54. # Updates topics_count, messages_count and last_message_id attributes for +board_id+
  55. def self.reset_counters!(board_id)
  56. board_id = board_id.to_i
  57. Board.where(:id => board_id).
  58. update_all(["topics_count = (SELECT COUNT(*) FROM #{Message.table_name} WHERE board_id=:id AND parent_id IS NULL)," +
  59. " messages_count = (SELECT COUNT(*) FROM #{Message.table_name} WHERE board_id=:id)," +
  60. " last_message_id = (SELECT MAX(id) FROM #{Message.table_name} WHERE board_id=:id)", :id => board_id])
  61. end
  62. def self.board_tree(boards, parent_id=nil, level=0)
  63. tree = []
  64. boards.select {|board| board.parent_id == parent_id}.sort_by(&:position).each do |board|
  65. tree << [board, level]
  66. tree += board_tree(boards, board.id, level+1)
  67. end
  68. if block_given?
  69. tree.each do |board, level|
  70. yield board, level
  71. end
  72. end
  73. tree
  74. end
  75. protected
  76. def validate_board
  77. if parent_id && parent_id_changed?
  78. errors.add(:parent_id, :invalid) unless valid_parents.include?(parent)
  79. end
  80. end
  81. end