2 # Salutations plugin: respond to salutations
\r
3 # TODO allow online editing of salutations
\r
4 # TODO *REMEMBER* to set @changed to true after edit
\r
5 # TODO or changes won't be saved
\r
7 unless Array.respond_to?(:pick_one)
\r
8 debug "Defining the pick_one method for Array"
\r
11 return nil if self.empty?
\r
12 self[rand(self.length)]
\r
18 class SalutPlugin < Plugin
\r
19 BotConfig.register BotConfigBooleanValue.new('salut.all_languages',
\r
21 :desc => "Check for a salutation in all languages and not just in the one defined by core.language",
\r
22 :on_change => Proc.new {|bot, v| bot.plugins['salut'].reload}
\r
24 BotConfig.register BotConfigBooleanValue.new('salut.address_only',
\r
26 :desc => "When set to true, the bot will only reply to salutations directed at him",
\r
27 :on_change => Proc.new {|bot, v| bot.plugins['salut'].reload}
\r
32 @salutations = Hash.new
\r
34 @main_lang_str = nil
\r
42 def set_language(what)
\r
51 @salutations.each { |lang, hash|
\r
52 hash.each { |situation, array|
\r
63 @match[:in] = Regexp.new("\\b(?:" + ar_in.uniq.map { |txt|
\r
65 }.join('|') + ")\\b", Regexp::IGNORECASE) unless ar_in.empty?
\r
66 @match[:out] = Regexp.new("\\b(?:" + ar_out.uniq.map { |txt|
\r
68 }.join('|') + ")\\b", Regexp::IGNORECASE) unless ar_out.empty?
\r
69 @match[:both] = Regexp.new("\\b(?:" + ar_both.uniq.map { |txt|
\r
71 }.join('|') + ")\\b", Regexp::IGNORECASE) unless ar_both.empty?
\r
72 debug "Matches: #{@match.inspect}"
\r
76 return unless m.kind_of?(PrivMessage)
\r
77 if @bot.config['salut.address_only']
\r
78 return unless m.address? or m.message =~ /#{Regexp.escape(@bot.nick)}/
\r
81 [:both, :in, :out].each { |k|
\r
82 next unless @match[k]
\r
83 debug "Checking salutations #{k} (#{@match[k].inspect})"
\r
84 if m.message =~ /^#{@match[k]}/
\r
93 salut_reply(m, salut, :morning)
\r
95 salut_reply(m, salut, :afternoon)
\r
97 salut_reply(m, salut, :evening)
\r
101 def salut_reply(m, k, time)
\r
102 debug "Replying to #{k} in the #{time}"
\r
109 debug "Building array ..."
\r
111 rep_ar += @salutations[@main_lang].fetch("#{time}#{sfx}".to_sym, [])
\r
112 rep_ar += @salutations[@main_lang].fetch("#{time}".to_sym, []) unless sfx.empty?
\r
113 rep_ar += @salutations[@main_lang].fetch("generic#{sfx}".to_sym, [])
\r
114 rep_ar += @salutations[@main_lang].fetch("generic".to_sym, []) unless sfx.empty?
\r
115 debug "Choosing reply in #{rep_ar.inspect} ..."
\r
117 if m.public? and (m.address? or m =~ /#{Regexp.escape(@bot.nick)}/)
\r
118 choice = @bot.lang.get("hello_X") % m.sourcenick
\r
120 choice = @bot.lang.get("hello") % m.sourcenick
\r
123 choice = rep_ar.pick_one
\r
124 if m.public? and (m.address? or m.message =~ /#{Regexp.escape(@bot.nick)}/)
\r
125 choice += "#{[',',''].pick_one} #{m.sourcenick}"
\r
126 choice += [" :)", " :D", "!", "", "", ""].pick_one
\r
129 debug "Replying #{choice}"
\r
130 m.plainreply choice
\r
135 @main_lang_str = @bot.config['core.language']
\r
136 @main_lang = @main_lang_str.to_sym
\r
137 @all_langs = @bot.config['salut.all_languages']
\r
139 # Get all available languages
\r
140 langs = Dir.new("#{@bot.botclass}/salut").collect {|f|
\r
141 f =~ /salut-([^.]+)/ ? $1 : nil
\r
143 langs.each { |lang|
\r
144 @salutations[lang.to_sym] = load_lang(lang)
\r
148 @salutations[@main_lang] = load_lang(@main_lang_str)
\r
154 def load_lang(lang)
\r
155 dir = "#{@bot.botclass}/salut"
\r
156 if not File.exist?(dir)
\r
159 file = "#{@bot.botclass}/salut/salut-#{lang}"
\r
160 if File.exist?(file)
\r
162 salutations = Hash.new
\r
163 content = YAML::load_file(file)
\r
164 content.each { |key, val|
\r
165 salutations[key.to_sym] = val
\r
169 error "failed to read salutations in #{lang}: #{$!}"
\r
176 return if @salutations.empty?
\r
177 return unless @changed
\r
178 @salutations.each { |lang, val|
\r
180 save_lang(lang, val)
\r
185 def save_lang(lang, val)
\r
186 fn = "#{@bot.botclass}/salut/salut-#{lang}"
\r
187 Utils.safe_save(fn) { |file|
\r
188 file.puts val.to_yaml
\r
194 plugin = SalutPlugin.new
\r