X-Git-Url: https://vcs.fsf.org/?p=rainbowstream.git;a=blobdiff_plain;f=rainbowstream%2Fdraw.py;h=767c9d56d68c63f0d127f8970abcb5ffc6d7d8ff;hp=4fbe82723f46861c422dafb28d7ab3cb4abb7bf8;hb=7a8a52fc755547ee739938c0086b40e11a9eda31;hpb=ffc77ba882f254e01b3e8b6f48ac92b777932f9d;ds=sidebyside diff --git a/rainbowstream/draw.py b/rainbowstream/draw.py index 4fbe827..767c9d5 100644 --- a/rainbowstream/draw.py +++ b/rainbowstream/draw.py @@ -1,10 +1,11 @@ import random +import textwrap import itertools import requests -import time import locale import arrow import re +import os from twitter.util import printNicely from functools import wraps @@ -144,18 +145,10 @@ def color_func(func_name): return globals()[func_name] -def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): +def draw(t, keyword=None, humanize=True, noti=False, fil=[], ig=[]): """ Draw the rainbow """ - - # Check the semaphore pause and lock (stream process only) - if check_semaphore: - if c['pause']: - return - while c['lock']: - time.sleep(0.5) - # Check config check_config() @@ -184,6 +177,14 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): try: text = 'RT @' + t['retweeted_status']['user']['screen_name'] + ': ' +\ t['retweeted_status']['text'] + # Display as a notification + target = t['retweeted_status']['user']['screen_name'] + if all([target == c['original_name'], not noti]): + # Add to evens for 'notification' command + t['event'] = 'retweet' + c['events'].append(t) + notify_retweet(t) + return except: pass @@ -212,6 +213,7 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): media_url = None # Filter and ignore + mytweet = screen_name == c['original_name'] screen_name = '@' + screen_name fil = list(set((fil or []) + c['ONLY_LIST'])) ig = list(set((ig or []) + c['IGNORE_LIST'])) @@ -229,7 +231,10 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): # Format info name = cycle_color(name) - nick = color_func(c['TWEET']['nick'])(screen_name) + if mytweet: + nick = color_func(c['TWEET']['mynick'])(screen_name) + else: + nick = color_func(c['TWEET']['nick'])(screen_name) clock = clock id = str(rid) fav = '' @@ -256,7 +261,7 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): # Highlight link tweet = lmap( lambda x: color_func(c['TWEET']['link'])(x) - if x[0:4] == 'http' + if x.startswith('http') else x, tweet) # Highlight hashtag @@ -265,6 +270,15 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): if x.startswith('#') else x, tweet) + # Highlight my tweet + if mytweet: + tweet = [color_func(c['TWEET']['mytweet'])(x) + for x in tweet + if not any([ + x == 'RT', + x.startswith('http'), + x.startswith('#')]) + ] # Highlight keyword tweet = ' '.join(tweet) if keyword: @@ -284,27 +298,34 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): formater = fav.join(formater.split('#fav')) formater = tweet.join(formater.split('#tweet')) # Change clock word - word = [w for w in formater.split() if '#clock' in w][0] + word = [wo for wo in formater.split() if '#clock' in wo][0] delimiter = color_func(c['TWEET']['clock'])( clock.join(word.split('#clock'))) formater = delimiter.join(formater.split(word)) # Change id word - word = [w for w in formater.split() if '#id' in w][0] + word = [wo for wo in formater.split() if '#id' in wo][0] delimiter = color_func(c['TWEET']['id'])(id.join(word.split('#id'))) formater = delimiter.join(formater.split(word)) # Change retweet count word - word = [w for w in formater.split() if '#rt_count' in w][0] + word = [wo for wo in formater.split() if '#rt_count' in wo][0] delimiter = color_func(c['TWEET']['retweet_count'])( str(retweet_count).join(word.split('#rt_count'))) formater = delimiter.join(formater.split(word)) # Change favorites count word - word = [w for w in formater.split() if '#fa_count' in w][0] + word = [wo for wo in formater.split() if '#fa_count' in wo][0] delimiter = color_func(c['TWEET']['favorite_count'])( str(favorite_count).join(word.split('#fa_count'))) formater = delimiter.join(formater.split(word)) except: pass + # Add spaces in begining of line if this is inside a notification + if noti: + formater = '\n '.join(formater.split('\n')) + # Reformat + if formater.startswith('\n'): + formater = formater[1:] + # Draw printNicely(formater) @@ -318,18 +339,203 @@ def draw(t, keyword=None, humanize=True, check_semaphore=False, fil=[], ig=[]): printNicely(red('Sorry, image link is broken')) -def print_message(m, check_semaphore=False): +def print_threads(d): """ - Print direct message + Print threads of messages """ + id = 1 + rel = {} + for partner in d: + messages = d[partner] + count = len(messages) + screen_name = '@' + partner[0] + name = partner[1] + screen_name = color_func(c['MESSAGE']['partner'])(screen_name) + name = cycle_color(name) + thread_id = color_func(c['MESSAGE']['id'])('thread_id:' + str(id)) + line = ' ' * 2 + name + ' ' + screen_name + \ + ' (' + str(count) + ' message) ' + thread_id + printNicely(line) + rel[id] = partner + id += 1 + dg['thread'] = d + return rel - # Check the semaphore pause and lock (stream process only) - if check_semaphore: - if c['pause']: - return - while c['lock']: - time.sleep(0.5) +def print_thread(partner, me_nick, me_name): + """ + Print a thread of messages + """ + # Sort messages by time + messages = dg['thread'][partner] + messages.sort(key=lambda x: parser.parse(x['created_at'])) + # Use legacy display on non-ascii text message + ms = [m['text'] for m in messages] + ums = [m for m in ms if not all(ord(c) < 128 for c in m)] + if ums: + for m in messages: + print_message(m) + printNicely('') + return + # Print the first line + dg['frame_margin'] = margin = 2 + partner_nick = partner[0] + partner_name = partner[1] + left_size = len(partner_nick) + len(partner_name) + 2 + right_size = len(me_nick) + len(me_name) + 2 + partner_nick = color_func(c['MESSAGE']['partner'])('@' + partner_nick) + partner_name = cycle_color(partner_name) + me_screen_name = color_func(c['MESSAGE']['me'])('@' + me_nick) + me_name = cycle_color(me_name) + left = ' ' * margin + partner_name + ' ' + partner_nick + right = me_name + ' ' + me_screen_name + ' ' * margin + h, w = os.popen('stty size', 'r').read().split() + w = int(w) + line = '{}{}{}'.format( + left, ' ' * (w - left_size - right_size - 2 * margin), right) + printNicely('') + printNicely(line) + printNicely('') + # Print messages + for m in messages: + if m['sender_screen_name'] == me_nick: + print_right_message(m) + elif m['recipient_screen_name'] == me_nick: + print_left_message(m) + + +def print_right_message(m): + """ + Print a message on the right of screen + """ + h, w = os.popen('stty size', 'r').read().split() + w = int(w) + frame_width = w // 3 - dg['frame_margin'] + frame_width = max(c['THREAD_MIN_WIDTH'], frame_width) + step = frame_width - 2 * dg['frame_margin'] + slicing = textwrap.wrap(m['text'], step) + spaces = w - frame_width - dg['frame_margin'] + dotline = ' ' * spaces + '-' * frame_width + dotline = color_func(c['MESSAGE']['me_frame'])(dotline) + # Draw the frame + printNicely(dotline) + for line in slicing: + fill = step - len(line) + screen_line = ' ' * spaces + '| ' + line + ' ' * fill + ' ' + if slicing[-1] == line: + screen_line = screen_line + ' >' + else: + screen_line = screen_line + '|' + screen_line = color_func(c['MESSAGE']['me_frame'])(screen_line) + printNicely(screen_line) + printNicely(dotline) + # Format clock + date = parser.parse(m['created_at']) + date = arrow.get(date).to('local').datetime + clock_format = '%Y/%m/%d %H:%M:%S' + try: + clock_format = c['FORMAT']['MESSAGE']['CLOCK_FORMAT'] + except: + pass + clock = date.strftime(clock_format) + # Format id + if m['id'] not in c['message_dict']: + c['message_dict'].append(m['id']) + rid = len(c['message_dict']) - 1 + else: + rid = c['message_dict'].index(m['id']) + id = str(rid) + # Print meta + formater = '' + try: + virtual_meta = formater = c['THREAD_META_RIGHT'] + virtual_meta = clock.join(virtual_meta.split('#clock')) + virtual_meta = id.join(virtual_meta.split('#id')) + # Change clock word + word = [wo for wo in formater.split() if '#clock' in wo][0] + delimiter = color_func(c['MESSAGE']['clock'])( + clock.join(word.split('#clock'))) + formater = delimiter.join(formater.split(word)) + # Change id word + word = [wo for wo in formater.split() if '#id' in wo][0] + delimiter = color_func(c['MESSAGE']['id'])(id.join(word.split('#id'))) + formater = delimiter.join(formater.split(word)) + except Exception: + printNicely(red('Wrong format in config.')) + return + meta = formater + line = ' ' * (w - len(virtual_meta) - dg['frame_margin']) + meta + printNicely(line) + + +def print_left_message(m): + """ + Print a message on the left of screen + """ + h, w = os.popen('stty size', 'r').read().split() + w = int(w) + frame_width = w // 3 - dg['frame_margin'] + frame_width = max(c['THREAD_MIN_WIDTH'], frame_width) + step = frame_width - 2 * dg['frame_margin'] + slicing = textwrap.wrap(m['text'], step) + spaces = dg['frame_margin'] + dotline = ' ' * spaces + '-' * frame_width + dotline = color_func(c['MESSAGE']['partner_frame'])(dotline) + # Draw the frame + printNicely(dotline) + for line in slicing: + fill = step - len(line) + screen_line = ' ' + line + ' ' * fill + ' |' + if slicing[-1] == line: + screen_line = ' ' * (spaces - 1) + '< ' + screen_line + else: + screen_line = ' ' * spaces + '|' + screen_line + screen_line = color_func(c['MESSAGE']['partner_frame'])(screen_line) + printNicely(screen_line) + printNicely(dotline) + # Format clock + date = parser.parse(m['created_at']) + date = arrow.get(date).to('local').datetime + clock_format = '%Y/%m/%d %H:%M:%S' + try: + clock_format = c['FORMAT']['MESSAGE']['CLOCK_FORMAT'] + except: + pass + clock = date.strftime(clock_format) + # Format id + if m['id'] not in c['message_dict']: + c['message_dict'].append(m['id']) + rid = len(c['message_dict']) - 1 + else: + rid = c['message_dict'].index(m['id']) + id = str(rid) + # Print meta + formater = '' + try: + virtual_meta = formater = c['THREAD_META_LEFT'] + virtual_meta = clock.join(virtual_meta.split('#clock')) + virtual_meta = id.join(virtual_meta.split('#id')) + # Change clock word + word = [wo for wo in formater.split() if '#clock' in wo][0] + delimiter = color_func(c['MESSAGE']['clock'])( + clock.join(word.split('#clock'))) + formater = delimiter.join(formater.split(word)) + # Change id word + word = [wo for wo in formater.split() if '#id' in wo][0] + delimiter = color_func(c['MESSAGE']['id'])(id.join(word.split('#id'))) + formater = delimiter.join(formater.split(word)) + except Exception: + printNicely(red('Wrong format in config.')) + return + meta = formater + line = ' ' * dg['frame_margin'] + meta + printNicely(line) + + +def print_message(m): + """ + Print direct message + """ # Retrieve message sender_screen_name = '@' + m['sender_screen_name'] sender_name = m['sender']['name'] @@ -375,12 +581,12 @@ def print_message(m, check_semaphore=False): formater = recipient_nick.join(formater.split("#recipient_nick")) formater = text.join(formater.split("#message")) # Change clock word - word = [w for w in formater.split() if '#clock' in w][0] + word = [wo for wo in formater.split() if '#clock' in wo][0] delimiter = color_func(c['MESSAGE']['clock'])( clock.join(word.split('#clock'))) formater = delimiter.join(formater.split(word)) # Change id word - word = [w for w in formater.split() if '#id' in w][0] + word = [wo for wo in formater.split() if '#id' in wo][0] delimiter = color_func(c['MESSAGE']['id'])(id.join(word.split('#id'))) formater = delimiter.join(formater.split(word)) except: @@ -391,6 +597,279 @@ def print_message(m, check_semaphore=False): printNicely(formater) +def notify_retweet(t): + """ + Notify a retweet + """ + source = t['user'] + created_at = t['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'retweeted your tweet') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + draw(t=t['retweeted_status'], noti=True) + + +def notify_favorite(e): + """ + Notify a favorite event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = e['target_object'] + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'favorited your tweet') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + draw(t=target_object, noti=True) + + +def notify_unfavorite(e): + """ + Notify a unfavorite event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = e['target_object'] + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'unfavorited your tweet') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + draw(t=target_object, noti=True) + + +def notify_follow(e): + """ + Notify a follow event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'followed you') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + + +def notify_list_member_added(e): + """ + Notify a list_member_added event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = [e['target_object']] # list of Twitter list + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'added you to a list') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + print_list(target_object, noti=True) + + +def notify_list_member_removed(e): + """ + Notify a list_member_removed event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = [e['target_object']] # list of Twitter list + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'removed you from a list') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + print_list(target_object, noti=True) + + +def notify_list_user_subscribed(e): + """ + Notify a list_user_subscribed event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = [e['target_object']] # list of Twitter list + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'subscribed to your list') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + print_list(target_object, noti=True) + + +def notify_list_user_unsubscribed(e): + """ + Notify a list_user_unsubscribed event + """ + # Retrieve info + target = e['target'] + if target['screen_name'] != c['original_name']: + return + source = e['source'] + target_object = [e['target_object']] # list of Twitter list + created_at = e['created_at'] + # Format + source_user = cycle_color(source['name']) + \ + color_func(c['NOTIFICATION']['source_nick'])( + ' @' + source['screen_name']) + notify = color_func(c['NOTIFICATION']['notify'])( + 'unsubscribed from your list') + date = parser.parse(created_at) + date = arrow.get(date).to('local') + lang, encode = locale.getdefaultlocale() + clock = arrow.get(date).to('local').humanize(locale=lang) + clock = color_func(c['NOTIFICATION']['clock'])(clock) + meta = c['NOTIFY_FORMAT'] + meta = source_user.join(meta.split('#source_user')) + meta = notify.join(meta.split('#notify')) + meta = clock.join(meta.split('#clock')) + # Output + printNicely('') + printNicely(meta) + print_list(target_object, noti=True) + + +def nothing(e): + """ + Do nothing for other event + """ + return + + +def print_event(e): + """ + Notify an event + """ + event_dict = { + 'retweet': notify_retweet, + 'favorite': notify_favorite, + 'unfavorite': notify_unfavorite, + 'follow': notify_follow, + 'list_member_added': notify_list_member_added, + 'list_member_removed': notify_list_member_removed, + 'list_user_subscribed': notify_list_user_subscribed, + 'list_user_unsubscribed': notify_list_user_unsubscribed, + } + event_dict.get(e['event'], nothing)(e) + + def show_profile(u): """ Show a profile @@ -489,7 +968,7 @@ def print_trends(trends): printNicely('') -def print_list(group): +def print_list(group, noti=False): """ Display a list """ @@ -514,11 +993,15 @@ def print_list(group): clock = arrow.get(date).to('local').humanize(locale=lang) clock = 'Created at ' + color_func(c['GROUP']['clock'])(clock) + prefix = ' ' * 2 + # Add spaces in begining of line if this is inside a notification + if noti: + prefix = ' ' * 2 + prefix # Create lines - line1 = ' ' * 2 + name + member + ' ' + subscriber - line2 = ' ' * 4 + description - line3 = ' ' * 4 + mode - line4 = ' ' * 4 + clock + line1 = prefix + name + member + ' ' + subscriber + line2 = prefix + ' ' * 2 + description + line3 = prefix + ' ' * 2 + mode + line4 = prefix + ' ' * 2 + clock # Display printNicely('') @@ -527,7 +1010,8 @@ def print_list(group): printNicely(line3) printNicely(line4) - printNicely('') + if not noti: + printNicely('') def show_calendar(month, date, rel): @@ -574,31 +1058,31 @@ def format_quote(tweet): except: pass # Highlight like a tweet - formater = formater.split() - formater = lmap( + notice = formater.split() + notice = lmap( lambda x: light_green(x) if x == '#comment' else x, - formater) - formater = lmap( + notice) + notice = lmap( lambda x: color_func(c['TWEET']['rt'])(x) if x == 'RT' else x, - formater) - formater = lmap(lambda x: cycle_color(x) if x[0] == '@' else x, formater) - formater = lmap( + notice) + notice = lmap(lambda x: cycle_color(x) if x[0] == '@' else x, notice) + notice = lmap( lambda x: color_func(c['TWEET']['link'])(x) if x[0:4] == 'http' else x, - formater) - formater = lmap( + notice) + notice = lmap( lambda x: color_func(c['TWEET']['hashtag'])(x) if x.startswith('#') else x, - formater) - formater = ' '.join(formater) + notice) + notice = ' '.join(notice) # Notice - notice = light_magenta('Quoting: "') + formater + light_magenta('"') + notice = light_magenta('Quoting: "') + notice + light_magenta('"') printNicely(notice) return formater