mirror of
https://github.com/redmine/redmine.git
synced 2025-11-01 10:56:17 +01:00
111 lines
3.6 KiB
Ruby
111 lines
3.6 KiB
Ruby
|
|
# frozen_string_literal: true
|
||
|
|
|
||
|
|
# Redmine - project management software
|
||
|
|
# Copyright (C) 2006-2022 Jean-Philippe Lang
|
||
|
|
#
|
||
|
|
# This program is free software; you can redistribute it and/or
|
||
|
|
# modify it under the terms of the GNU General Public License
|
||
|
|
# as published by the Free Software Foundation; either version 2
|
||
|
|
# of the License, or (at your option) any later version.
|
||
|
|
#
|
||
|
|
# This program is distributed in the hope that it will be useful,
|
||
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
|
# GNU General Public License for more details.
|
||
|
|
#
|
||
|
|
# You should have received a copy of the GNU General Public License
|
||
|
|
# along with this program; if not, write to the Free Software
|
||
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
|
|
||
|
|
module Redmine
|
||
|
|
module Acts
|
||
|
|
module Mentionable
|
||
|
|
def self.included(base)
|
||
|
|
base.extend ClassMethods
|
||
|
|
end
|
||
|
|
|
||
|
|
module ClassMethods
|
||
|
|
def acts_as_mentionable(options = {})
|
||
|
|
class_attribute :mentionable_attributes
|
||
|
|
self.mentionable_attributes = options[:attributes]
|
||
|
|
|
||
|
|
attr_accessor :mentioned_users
|
||
|
|
|
||
|
|
send :include, Redmine::Acts::Mentionable::InstanceMethods
|
||
|
|
|
||
|
|
after_save :parse_mentions
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
module InstanceMethods
|
||
|
|
def self.included(base)
|
||
|
|
base.extend ClassMethods
|
||
|
|
end
|
||
|
|
|
||
|
|
def notified_mentions
|
||
|
|
notified = mentioned_users.to_a
|
||
|
|
notified.reject! {|user| user.mail.blank? || user.mail_notification == 'none'}
|
||
|
|
if respond_to?(:visible?)
|
||
|
|
notified.select! {|user| visible?(user)}
|
||
|
|
end
|
||
|
|
notified
|
||
|
|
end
|
||
|
|
|
||
|
|
private
|
||
|
|
|
||
|
|
def parse_mentions
|
||
|
|
mentionable_attrs = self.mentionable_attributes
|
||
|
|
saved_mentionable_attrs = self.saved_changes.select{|a| mentionable_attrs.include?(a)}
|
||
|
|
|
||
|
|
saved_mentionable_attrs.each do |key, attr|
|
||
|
|
old_value, new_value = attr
|
||
|
|
get_mentioned_users(old_value, new_value)
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
def get_mentioned_users(old_content, new_content)
|
||
|
|
self.mentioned_users = []
|
||
|
|
|
||
|
|
previous_matches = scan_for_mentioned_users(old_content)
|
||
|
|
current_matches = scan_for_mentioned_users(new_content)
|
||
|
|
new_matches = (current_matches - previous_matches).flatten
|
||
|
|
|
||
|
|
if new_matches.any?
|
||
|
|
self.mentioned_users = User.visible.active.where(login: new_matches)
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
def scan_for_mentioned_users(content)
|
||
|
|
return [] if content.nil?
|
||
|
|
|
||
|
|
# remove quoted text
|
||
|
|
content = content.gsub(%r{\r\n(?:\>\s)+(.*?)\r\n}m, '')
|
||
|
|
|
||
|
|
text_formatting = Setting.text_formatting
|
||
|
|
# Remove text wrapped in pre tags based on text formatting
|
||
|
|
case text_formatting
|
||
|
|
when 'textile'
|
||
|
|
content = content.gsub(%r{<pre>(.*?)</pre>}m, '')
|
||
|
|
when 'markdown', 'common_mark'
|
||
|
|
content = content.gsub(%r{(~~~|```)(.*?)(~~~|```)}m, '')
|
||
|
|
end
|
||
|
|
|
||
|
|
users = content.scan(MENTION_PATTERN).flatten
|
||
|
|
end
|
||
|
|
|
||
|
|
MENTION_PATTERN = /
|
||
|
|
(?:^|\W) # beginning of string or non-word char
|
||
|
|
@((?>[a-z0-9][a-z0-9-]*)) # @username
|
||
|
|
(?!\/) # without a trailing slash
|
||
|
|
(?=
|
||
|
|
\.+[ \t\W]| # dots followed by space or non-word character
|
||
|
|
\.+$| # dots at end of line
|
||
|
|
[^0-9a-zA-Z_.]| # non-word character except dot
|
||
|
|
$ # end of line
|
||
|
|
)
|
||
|
|
/ix
|
||
|
|
end
|
||
|
|
end
|
||
|
|
end
|
||
|
|
end
|