2013-01-03 02:06:07 -05:00
|
|
|
# == Issuable concern
|
2012-12-30 09:19:31 -05:00
|
|
|
#
|
2012-10-09 15:09:46 -04:00
|
|
|
# Contains common functionality shared between Issues and MergeRequests
|
2012-12-30 09:19:31 -05:00
|
|
|
#
|
|
|
|
# Used by Issue, MergeRequest
|
|
|
|
#
|
2013-01-03 02:06:07 -05:00
|
|
|
module Issuable
|
2012-08-08 21:40:57 -04:00
|
|
|
extend ActiveSupport::Concern
|
2015-04-21 09:23:20 -04:00
|
|
|
include Participable
|
2015-10-14 10:20:11 -04:00
|
|
|
include Mentionable
|
2012-08-08 21:40:57 -04:00
|
|
|
|
|
|
|
included do
|
2012-08-10 18:07:50 -04:00
|
|
|
belongs_to :author, class_name: "User"
|
|
|
|
belongs_to :assignee, class_name: "User"
|
2015-07-30 08:45:54 -04:00
|
|
|
belongs_to :updated_by, class_name: "User"
|
2012-10-30 03:22:24 -04:00
|
|
|
belongs_to :milestone
|
2012-08-10 18:07:50 -04:00
|
|
|
has_many :notes, as: :noteable, dependent: :destroy
|
2014-07-29 12:19:26 -04:00
|
|
|
has_many :label_links, as: :target, dependent: :destroy
|
|
|
|
has_many :labels, through: :label_links
|
2015-03-16 09:22:50 -04:00
|
|
|
has_many :subscriptions, dependent: :destroy, as: :subscribable
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2012-10-08 20:10:04 -04:00
|
|
|
validates :author, presence: true
|
|
|
|
validates :title, presence: true, length: { within: 0..255 }
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2013-08-10 13:25:53 -04:00
|
|
|
scope :authored, ->(user) { where(author_id: user) }
|
2013-06-17 04:57:37 -04:00
|
|
|
scope :assigned_to, ->(u) { where(assignee_id: u.id)}
|
2013-02-12 02:16:45 -05:00
|
|
|
scope :recent, -> { order("created_at DESC") }
|
2013-06-17 04:57:37 -04:00
|
|
|
scope :assigned, -> { where("assignee_id IS NOT NULL") }
|
|
|
|
scope :unassigned, -> { where("assignee_id IS NULL") }
|
2013-08-08 10:29:31 -04:00
|
|
|
scope :of_projects, ->(ids) { where(project_id: ids) }
|
2014-02-28 15:43:16 -05:00
|
|
|
scope :opened, -> { with_state(:opened, :reopened) }
|
2014-06-04 11:10:53 -04:00
|
|
|
scope :only_opened, -> { with_state(:opened) }
|
|
|
|
scope :only_reopened, -> { with_state(:reopened) }
|
2014-02-28 15:43:16 -05:00
|
|
|
scope :closed, -> { with_state(:closed) }
|
2015-02-05 23:21:21 -05:00
|
|
|
scope :order_milestone_due_desc, -> { joins(:milestone).reorder('milestones.due_date DESC, milestones.id DESC') }
|
|
|
|
scope :order_milestone_due_asc, -> { joins(:milestone).reorder('milestones.due_date ASC, milestones.id ASC') }
|
2013-08-19 15:10:56 -04:00
|
|
|
|
2012-08-08 21:40:57 -04:00
|
|
|
delegate :name,
|
|
|
|
:email,
|
2012-08-10 18:07:50 -04:00
|
|
|
to: :author,
|
|
|
|
prefix: true
|
2012-08-08 21:40:57 -04:00
|
|
|
|
|
|
|
delegate :name,
|
|
|
|
:email,
|
2012-08-10 18:07:50 -04:00
|
|
|
to: :assignee,
|
|
|
|
allow_nil: true,
|
|
|
|
prefix: true
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2013-09-09 09:32:43 -04:00
|
|
|
attr_mentionable :title, :description
|
2015-10-16 05:26:48 -04:00
|
|
|
participant :author, :assignee, :notes_with_associations
|
2012-08-08 21:40:57 -04:00
|
|
|
end
|
|
|
|
|
2012-08-09 13:45:12 -04:00
|
|
|
module ClassMethods
|
|
|
|
def search(query)
|
2014-06-03 11:02:03 -04:00
|
|
|
where("LOWER(title) like :query", query: "%#{query.downcase}%")
|
2012-08-09 13:45:12 -04:00
|
|
|
end
|
2014-01-14 13:49:32 -05:00
|
|
|
|
2014-08-27 05:47:30 -04:00
|
|
|
def full_search(query)
|
|
|
|
where("LOWER(title) like :query OR LOWER(description) like :query", query: "%#{query.downcase}%")
|
|
|
|
end
|
|
|
|
|
2014-01-14 13:49:32 -05:00
|
|
|
def sort(method)
|
2015-02-05 23:21:21 -05:00
|
|
|
case method.to_s
|
|
|
|
when 'milestone_due_asc' then order_milestone_due_asc
|
|
|
|
when 'milestone_due_desc' then order_milestone_due_desc
|
|
|
|
else
|
|
|
|
order_by(method)
|
|
|
|
end
|
2014-01-14 13:49:32 -05:00
|
|
|
end
|
2012-08-08 21:40:57 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def today?
|
|
|
|
Date.today == created_at.to_date
|
|
|
|
end
|
|
|
|
|
|
|
|
def new?
|
|
|
|
today? && created_at == updated_at
|
|
|
|
end
|
2012-10-09 18:25:29 -04:00
|
|
|
|
|
|
|
def is_assigned?
|
|
|
|
!!assignee_id
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_being_reassigned?
|
|
|
|
assignee_id_changed?
|
|
|
|
end
|
|
|
|
|
2015-10-13 03:41:46 -04:00
|
|
|
def open?
|
|
|
|
opened? || reopened?
|
|
|
|
end
|
|
|
|
|
2013-01-14 18:52:25 -05:00
|
|
|
#
|
|
|
|
# Votes
|
|
|
|
#
|
|
|
|
|
|
|
|
# Return the number of -1 comments (downvotes)
|
|
|
|
def downvotes
|
2014-11-05 15:45:18 -05:00
|
|
|
filter_superceded_votes(notes.select(&:downvote?), notes).size
|
2013-01-03 02:06:07 -05:00
|
|
|
end
|
|
|
|
|
2013-01-14 18:52:25 -05:00
|
|
|
def downvotes_in_percent
|
2013-01-03 02:06:07 -05:00
|
|
|
if votes_count.zero?
|
|
|
|
0
|
|
|
|
else
|
2013-01-14 18:52:25 -05:00
|
|
|
100.0 - upvotes_in_percent
|
2013-01-03 02:06:07 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-01-14 18:52:25 -05:00
|
|
|
# Return the number of +1 comments (upvotes)
|
|
|
|
def upvotes
|
2014-11-05 15:45:18 -05:00
|
|
|
filter_superceded_votes(notes.select(&:upvote?), notes).size
|
2013-01-03 02:06:07 -05:00
|
|
|
end
|
|
|
|
|
2013-01-14 18:52:25 -05:00
|
|
|
def upvotes_in_percent
|
2013-01-03 02:06:07 -05:00
|
|
|
if votes_count.zero?
|
|
|
|
0
|
|
|
|
else
|
2013-01-14 18:52:25 -05:00
|
|
|
100.0 / votes_count * upvotes
|
2013-01-03 02:06:07 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Return the total number of votes
|
|
|
|
def votes_count
|
|
|
|
upvotes + downvotes
|
|
|
|
end
|
2013-05-09 19:37:47 -04:00
|
|
|
|
2015-03-16 11:20:17 -04:00
|
|
|
def subscribed?(user)
|
2015-03-16 09:22:50 -04:00
|
|
|
subscription = subscriptions.find_by_user_id(user.id)
|
|
|
|
|
|
|
|
if subscription
|
|
|
|
return subscription.subscribed
|
2015-03-15 12:17:12 -04:00
|
|
|
end
|
|
|
|
|
2015-03-27 07:19:34 -04:00
|
|
|
participants(user).include?(user)
|
2015-03-15 12:17:12 -04:00
|
|
|
end
|
|
|
|
|
2015-03-16 11:20:17 -04:00
|
|
|
def toggle_subscription(user)
|
|
|
|
subscriptions.
|
|
|
|
find_or_initialize_by(user_id: user.id).
|
|
|
|
update(subscribed: !subscribed?(user))
|
|
|
|
end
|
|
|
|
|
2014-10-05 09:03:15 -04:00
|
|
|
def to_hook_data(user)
|
2013-12-03 04:31:56 -05:00
|
|
|
{
|
|
|
|
object_kind: self.class.name.underscore,
|
2014-10-05 09:03:15 -04:00
|
|
|
user: user.hook_attrs,
|
2015-09-10 23:42:14 -04:00
|
|
|
repository: {
|
|
|
|
name: project.name,
|
|
|
|
url: project.url_to_repo,
|
|
|
|
description: project.description,
|
|
|
|
homepage: project.web_url
|
|
|
|
},
|
2014-09-15 03:10:35 -04:00
|
|
|
object_attributes: hook_attrs
|
2013-12-03 04:31:56 -05:00
|
|
|
}
|
|
|
|
end
|
2014-07-30 10:17:29 -04:00
|
|
|
|
|
|
|
def label_names
|
|
|
|
labels.order('title ASC').pluck(:title)
|
|
|
|
end
|
|
|
|
|
2014-08-17 16:22:01 -04:00
|
|
|
def remove_labels
|
|
|
|
labels.delete_all
|
|
|
|
end
|
|
|
|
|
2014-07-30 10:17:29 -04:00
|
|
|
def add_labels_by_names(label_names)
|
|
|
|
label_names.each do |label_name|
|
2015-01-22 12:37:47 -05:00
|
|
|
label = project.labels.create_with(color: Label::DEFAULT_COLOR).
|
|
|
|
find_or_create_by(title: label_name.strip)
|
2014-07-30 10:17:29 -04:00
|
|
|
self.labels << label
|
|
|
|
end
|
|
|
|
end
|
2014-11-05 15:45:18 -05:00
|
|
|
|
2015-07-24 15:31:15 -04:00
|
|
|
# Convert this Issuable class name to a format usable by Ability definitions
|
|
|
|
#
|
|
|
|
# Examples:
|
|
|
|
#
|
|
|
|
# issuable.class # => MergeRequest
|
|
|
|
# issuable.to_ability_name # => "merge_request"
|
|
|
|
def to_ability_name
|
|
|
|
self.class.to_s.underscore
|
|
|
|
end
|
|
|
|
|
2015-10-13 05:49:01 -04:00
|
|
|
def notes_with_associations
|
|
|
|
notes.includes(:author, :project)
|
|
|
|
end
|
|
|
|
|
2014-11-05 15:45:18 -05:00
|
|
|
private
|
|
|
|
|
|
|
|
def filter_superceded_votes(votes, notes)
|
|
|
|
filteredvotes = [] + votes
|
2015-01-22 12:37:47 -05:00
|
|
|
|
2014-11-05 15:45:18 -05:00
|
|
|
votes.each do |vote|
|
|
|
|
if vote.superceded?(notes)
|
|
|
|
filteredvotes.delete(vote)
|
|
|
|
end
|
|
|
|
end
|
2015-01-22 12:37:47 -05:00
|
|
|
|
2014-11-05 15:45:18 -05:00
|
|
|
filteredvotes
|
|
|
|
end
|
2012-08-08 21:26:56 -04:00
|
|
|
end
|