Macro Pinger

From FreeCAD Documentation
Revision as of 13:39, 7 May 2022 by Kunda1 (talk | contribs) (typo and refinements)
(diff) ← Older revision | Latest revision (diff) | Newer revision → (diff)
Other languages:

Macro Pinger

Description
Use this macro to easily ping users on the FreeCAD forum. Select the user from the combo box, click Ping, paste code into the forum post. You may add new users.

Macro version: 1.81
Last modified: 2022.03.28
FreeCAD version: 0.19
Download: ToolBar Icon
Author: TheMarkster
Author
TheMarkster
Download
ToolBar Icon
Links
Macro Version
1.81
Date last modified
2022.03.28
FreeCAD Version(s)
0.19
Default shortcut
None
See also
Macro Snip

Description

Use this macro to ping another user on the forum so that user gets a notification. Pinging requires knowing the username exactly and the user id number. You also must know the syntax required to produce the ping. It's very user-unfriendly, to say the least. In comes the Pinger macro to save the day. Run the macro, select the user you wish to ping, click the Ping button, copy-paste the code into the forum post.

Pinger screenshot

Parameters

The Macro supports user parameters, which are used for the added users list. It is in User parameter:Plugins/Pinger as "AddedUsers", a string containing the string representation of a Python dictionary.

Script

ToolBar icon

Macro_Snip.FCMacro

# -*- coding: utf-8 -*-
## Pinger macro v1.8
## By: TheMarkster
#  2022.03.28
## Aids in pinging users on the forum
## Select the user from the list and paste clipboard contents into the forum
__version__ = "1.81"
from PySide import QtGui,QtCore
import FreeCADGui as Gui
import FreeCAD
import json

#werner's code to get about info
class AboutInfo(QtCore.QObject):
    def __init__(self):
        super(AboutInfo, self).__init__()

    def eventFilter(self, obj, ev):
        if obj.metaObject().className() == "Gui::Dialog::AboutDialog" and\
        ev.type() == ev.ChildPolished and\
        hasattr(obj, 'on_copyButton_clicked'):
            obj.on_copyButton_clicked()
            QtGui.QApplication.instance().postEvent(obj, QtGui.QCloseEvent())
        return False

class PingerDlg:
    def __init__(self,parent=Gui.getMainWindow()):
        self.dict1 = {}
        self.setupDict1()
        self.pg = FreeCAD.ParamGet("User parameter:Plugins/Pinger")
        self.addedDict = {}
        self.fetchAdded() #updates self.addedDict
        self.fullDict = {}
        self.recentDict = {}
        self.cbUsers = QtGui.QComboBox()
        self.cbUsers.setEditable(True)
        self.cbUsers.lineEdit().returnPressed.connect(self.onReturnPressed)
        self.cbUsers.setToolTip("Type in a new name to add a new user, then click Ping/Add or press Enter.")
        self.form = QtGui.QDialog(parent, QtCore.Qt.Tool)
        self.form.setAttribute(QtCore.Qt.WA_DeleteOnClose, True)
        self.form.setWindowTitle("Pinger v"+__version__)
        self.form.setWindowIcon(QIconFromXPMString(__icon__))
        self.layout = QtGui.QVBoxLayout()
        self.form.setLayout(self.layout)
        topLine = QtGui.QHBoxLayout()
        iconBtn = QtGui.QPushButton()
        iconBtn.setIcon(QIconFromXPMString(__icon__))
        iconBtn.clicked.connect(self.information)
        topLine.addWidget(iconBtn)
        iconBtn.setMaximumWidth(48)
        self.layout.addLayout(topLine)
        usersLabel = QtGui.QLabel("Users: ")
        usersLabel.setMaximumWidth(100)
        topLine.addWidget(usersLabel)
        topLine.addWidget(self.cbUsers)
        self.pingBtn = QtGui.QPushButton("Ping/Add")
        self.pingBtn.setToolTip("Ping this user or add the user if not already in database.")
        topLine.addWidget(self.pingBtn)
        self.pingBtn.clicked.connect(self.onPingBtnClicked)
        secondLine = QtGui.QHBoxLayout()
        self.layout.addLayout(secondLine)
        self.aboutBtn = QtGui.QPushButton("About info")
        self.aboutBtn.setToolTip("Copy FreeCAD about information to clipboard")
        self.aboutBtn.clicked.connect(self.getAboutInfo)
        secondLine.addWidget(self.aboutBtn)
        self.editBtn = QtGui.QPushButton("Edit user id")
        self.editBtn.clicked.connect(self.onEditBtnClicked)
        secondLine.addWidget(self.editBtn)
        self.delBtn = QtGui.QPushButton("Remove user")
        self.delBtn.setToolTip("Remove user from added user list")
        self.delBtn.clicked.connect(self.onDelBtnClicked)
        secondLine.addWidget(self.delBtn)
        self.sortCheckBox = QtGui.QCheckBox("Sorted")
        secondLine.addWidget(self.sortCheckBox)
        self.sortCheckBox.stateChanged.connect(self.populateUsers)
        self.msgBox = QtGui.QPlainTextEdit()
        self.layout.addWidget(self.msgBox)
        recentLine = QtGui.QHBoxLayout()
        self.recentList = QtGui.QComboBox()
        self.recentList.setMinimumWidth(175)
        recentLine.addWidget(QtGui.QLabel("Recents:"))
        recentLine.addWidget(self.recentList)
        self.layout.addLayout(recentLine)
        self.recentPingBtn = QtGui.QPushButton("Ping")
        self.recentPingBtn.clicked.connect(self.onRecentPingBtnClicked)
        recentLine.addWidget(self.recentPingBtn)
        self.removeUserFromRecentBtn = QtGui.QPushButton("Remove user")
        self.removeUserFromRecentBtn.clicked.connect(self.onRemoveUserFromRecentBtnClicked)
        self.removeUserFromRecentBtn.setToolTip("Remove this user from recent list")
        recentLine.addWidget(self.removeUserFromRecentBtn)
        self.removeAllUsersFromRecentBtn = QtGui.QPushButton("Remove all")
        self.removeAllUsersFromRecentBtn.setToolTip("Remove all users from recent list")
        self.removeAllUsersFromRecentBtn.clicked.connect(self.onRemoveAllUsersFromRecentBtnClicked)
        recentLine.addWidget(self.removeAllUsersFromRecentBtn)
        self.populateUsers()

    def onRecentPingBtnClicked(self, arg1):
        curRecent = self.recentList.currentText()
        if not curRecent:
            self.msg("Recent users is empty","Error")
            return
        self.cbUsers.setCurrentText(curRecent)
        self.onPingBtnClicked(True)

    def onRemoveUserFromRecentBtnClicked(self, arg1):
        curRecent = self.recentList.currentText()
        if not curRecent:
            self.msg("Recent users is empty","Error")
            return
        self.recentDict.pop(curRecent)
        jstring = json.dumps(self.recentDict)
        self.pg.SetString("RecentUsers",jstring)
        self.populateUsers()

    def onRemoveAllUsersFromRecentBtnClicked(self, arg1):
        self.recentDict = {}
        self.pg.SetString("RecentUsers","{}")
        self.populateUsers()

    def msg(self, txt, msgType="Message"):
        if msgType == "Message":
            self.msgBox.setStyleSheet("color:black")
        elif msgType == "Error":
            self.msgBox.setStyleSheet("color:red")
        elif msgType == "Warning":
            self.msgBox.setStyleSheet("color:yellow")
        self.msgBox.setPlainText(txt)

    def buildFullDict(self):
        self.fullDict = {}
        defaultUsers = [k for k in self.dict1.keys()]
        self.fetchAdded()
        self.fetchRecent()
        addedUsers = [k for k in self.addedDict.keys()]
        for k,v in self.addedDict.items():
            self.fullDict[k] = v
        for k,v in self.dict1.items():
            self.fullDict[k] = v



    def populateUsers(self):
        self.buildFullDict()
        self.cbUsers.clear()
        self.cbUsers.addItem('--select user--')
        if self.sortCheckBox.checkState():
            self.cbUsers.addItems(sorted(self.fullDict))
        else:
            self.cbUsers.addItems(list(self.fullDict))
        self.fetchRecent()
        self.recentList.clear()
        self.recentList.addItems(list(self.recentDict))

    def fetchAdded(self):
        addedStr = self.pg.GetString("AddedUsers","{}")
        self.addedDict = json.loads(addedStr)

    def fetchRecent(self):
        recentStr = self.pg.GetString("RecentUsers","{}")
        self.recentDict = json.loads(recentStr)

    def addUser(self, username):
        self.fetchAdded()
        self.fetchRecent()
        curId =""
        if username in self.addedDict:
            curId = self.addedDict[username]
        id,ok = QtGui.QInputDialog.getText(self.form,"Add/Edit user",f"Enter user id for {username}:",text=curId)
        if not ok:
            return False
        try:
            id_int = int(id)
        except ValueError as ve:
            self.msg(f"{ve}","Error")
            return False
        self.addedDict[username] = id
        self.recentDict[username] = id
        jstring = json.dumps(self.addedDict)
        self.pg.SetString("AddedUsers",jstring)
        jstring = json.dumps(self.recentDict)
        self.pg.SetString("RecentUsers",jstring)
        self.buildFullDict()
        self.populateUsers()
        return True

    def information(self):
        self.msg("Pinger macro is to help in pinging users on the FreeCAD forum.\n\
Usage: select a user, click Ping/Add button, paste text from clipboard into forum\n\n\
To add a new user: type in the new name, click Ping/Add button.\n\n\
Click About info button to copy your FreeCAD version information to the clipboard.\n\n\
Only added users may be edited or removed.\n\n\
Sorted checkbox sorts the users alphabetically.  \
Added users are at the top of the list when not sorted, then according to number of posts.\n\n\
Only users with 100+ posts (at the time the database was updated) are guaranteed to appear in the list.\n\
")

    def onDelBtnClicked(self, arg1):
        username = self.cbUsers.currentText()
        if not username in self.addedDict:
            self.msg(f"{username} is not an added user, cannot be deleted","Error")
            return
        self.addedDict.pop(username)
        jstring = json.dumps(self.addedDict)
        self.pg.SetString("AddedUsers",jstring)
        if username in self.recentDict:
            self.recentDict.pop(username)
            jstring = json.dumps(self.recentDict)
            self.pg.SetString("RecentUsers",jstring)
        self.cbUsers.clear()
        self.populateUsers()
        self.msg(f"{username} has been removed from added users","Message")

    def onReturnPressed(self):
        self.onPingBtnClicked(True)

    def onEditBtnClicked(self, arg1):
        username = self.cbUsers.currentText()
        if not username in self.dict1: #adding edits addedusers only
            if not self.addUser(username):
                return
        else:
            self.msg(f"{username} is in default dictionary, cannot be edited","Error")
            return

    def onPingBtnClicked(self, arg1):
        username = self.cbUsers.currentText()
        if username == '--select user--':
            return
        if not username in self.fullDict:
            if not self.addUser(username):
                return
        ping = "[quote="+username+" user_id="+self.fullDict[username]+"]\npinged by pinger macro\n[/quote]"
        self.setClipText(ping)
        self.msg("Success!\n\nNow copied to clipboard: \n\n"+ping)
        self.form.setFocus()
        self.fetchRecent()
        if not username in self.recentDict:
            self.recentDict[username] = self.fullDict[username]
            jstring = json.dumps(self.recentDict)
            self.pg.SetString("RecentUsers",jstring)
            self.populateUsers()
        #self.form.close()

    def getClipText(self):
        clip = QtGui.QClipboard()
        return clip.text(QtGui.QClipboard.Clipboard)

    def setClipText(self,txt):
        clip = QtGui.QClipboard()
        clip.setText(txt)

    def getAboutInfo(self):
        ai=AboutInfo()
        QtGui.QApplication.instance().installEventFilter(ai)
        Gui.runCommand("Std_About")
        QtGui.QApplication.instance().removeEventFilter(ai)
        del ai
        clipText = self.getClipText()
        self.msg(f"Success!  Following now copied to clipboard:\n\n{clipText}")
        self.form.setFocus()

    def setupDict1(self):
        self.dict1 = {'chrisb': '5646', 'NormandC': '202', 'wmayer': '69', 'yorik': '68', 'bernd': '2069', 'Kunda1': '12229', 'jmaustpc': '611', 'triplus': '782', 'DeepSOIC': '3888', 'freecad-heini-1': '2598', 'openBrain': '22265', 'sgrogan': '4252', 'bejant': '1940', 'thomas-neemann': '29293', 'vocx': '21943', 'drmacro': '2867', 'microelly2': '2364', 'Roy_043': '22936', 'abdullah': '3232', 'papyblaise': '25808', 'wandererfan': '1375', 'TheMarkster': '19292', 'paullee': '8738', 'kisolre': '22435', 'mario52': '1058',
'quick61': '2030', 'looo': '2349', 'Vincen': '5772', 'uwestoehr': '23505', 'Shalmeneser': '38754', 'PrzemoF': '3666', 'jriegel': '67', 'GeneFC': '8180', 'ickby': '686', 'easyw-fc': '6387', 'meme2704': '14145', 'shoogen': '765', 'sliptonic': '708', 'renatorivo': '918', 'thschrader': '15166', 'kkremitzki': '7997', 'freedman': '19098', 'Bance': '5418', 'herbk': '4353', 'keithsloan52': '930', 'Jee-Bee': '6234', 'r-frank': '1529', 'realthunder': '12167', 'carlopav': '23005', 'Chris_G': '2561',
'onekk': '5245', 'ulrich1a': '1928', 'Joel_graff': '14673', 'chennes': '11959', 'Forthman': '19652', 'rockn': '681', 'freman': '22524', 'adrianinsaval': '19302', 'saso': '3305', 'Zolko': '22794', 'Willem': '9760', 'mlampert': '10163', 'tanderson69': '208', 'bitacovir': '3136', 'Syres': '21041', 'hammax': '12483', 'HakanSeven12': '23711', 'roerich_64': '6075', 'UR_': '12184', 'Dirk.B': '23630', 'HoWil': '6222', 'user1234': '9411', '-alex-': '23825', 'HarryGeier': '15774', 'freecadjam': '20736',
'flachyjoe': '984', 'bambuko': '27936', 'jeno': '3508', 'HarryvL': '18062', 'kbwbe': '19380', 'peterl94': '1819', 'RatonLaveur': '24547', 'johnwang': '23504', 'cox': '4523', 'M4x': '13711', 'reox': '9769', 'pablogil': '4517', 'domad': '33963', 'david69': '2566', 'edwilliams16': '36484', 'leoheck': '19003', 'Russ4262': '20523', 'heda': '7350', 'ppemawm': '1807', 'ArminF': '11953', 'jp-willm': '7776', 'Pauvres_honteux': '2803', 'regis': '6401', 'vejmarie': '7506', 'Gift': '6611',
'ian.rees': '3449', 'pperisin': '356', 'emills2': '5929', 'dubstar-04': '1642', 'ezzieyguywuf': '6058', 'mrlukeparry': '607', 'fc_tofu': '29038', 'logari81': '270', 'FaDa3D': '16107', 'jpg87': '13810', 'NewJoker': '36858', 'eivindkvedalen': '1546', 'damian': '6134', 'oliveroxtoby': '11950', 'Vagulus': '35139', 'jtm2020hyo': '35626', 'Cobraschock': '25890', 'jrheinlaender': '997', 'cappy0815': '4130', 'kaktus': '26846', 'Jimidi': '15956', 'vanuan': '22024', 'davidosterberg': '36295', 'ebrahi': '21433', 'kwahoo': '2441',
'Joyas': '3582', 'r.tec': '4264', 'dcapeletti': '3651', 'drei': '3278', 'schupin': '18259', 'mariwan': '39136', 'agryson': '11337', 'JoshM': '16808', 'jnxd': '5734', 'Cekuhnen': '43243', 'manuelkrause': '20592', 'Mar': '14191', 'aapo': '22095', 'blue0cean': '23920', 'fandaL': '3658', 'Fran': '16427', 'teobo': '2833', 'paddle': '29478', 'OficineRobotica': '23938', 'Giuli': '9935', 'danielfalck': '689', 'makkemal': '5932', 'apeltauer': '16146', 'shaise': '6188', 'bavariaSHAPE': '3425',
'ragohix769': '35209', 'galou_breizh': '334', 'garya': '22407', 'fcaduser': '2823', 'jaisejames': '10269', 'bill': '5185', 'blacey': '7328', 'fosselius': '8591', 'herrdeh': '3918', 'jruiz': '4299', 'edi': '29214', 'Sudhanshu': '21878', 'fran6t': '3600', 'wsteffe': '3846', 'nemesis': '2986', 'holdi': '11560', 'piffpoof': '4556', 'mnesarco': '30314', 'f3nix': '6125', 'furti': '17491', 'balrobs': '30989', 'amrit3701': '9146', 'jbe': '2330', 'Roland': '6638', 'rynn': '20919',
'jreinhardt': '2072', 'koluna': '21576', 'chakkree': '6331', 'RedBaron': '36427', 'memfis': '11177', 'Evgeniy': '43393', 'plgarcia': '6249', 'arturromarr': '15598', 'Sura': '21412', 'Lauri': '13429', 'joha2': '10551', 'cad1234': '22955', 'lemonbug': '4228', 'C4e': '49784', 'MRx': '35035', 'Claud': '29138', 'clintonsam75': '3470', 'Repman': '3536', 'Moult': '23100', 'OldDraftsman': '16332', 'LHC': '36255', 'usbhub': '24632', 'efyx': '4074', 'catman': '23210', 'JMG': '2538',
'polymer': '3974', 'Miq58': '46857', 'hobbes1069': '725', 'dxp.dev': '22713', 'wega': '2347', 'mendy': '19994', 'Konstantin': '3649', 'Renat': '3315', 'davecoventry': '2481', 'gsandy': '23521', 'blonblon': '10279', 'Drederwisch': '11726', 'Aleks': '30029', 'oldmachine': '15343', 'salp': '2409', 'FBXL5': '26761', 'Marco_T': '7564', 'bgoodr': '3447', 'dimitar': '25801', 'CharlieMAC': '3179', 'Markymark': '28078', 'HBC0': '6833', 'obelisk79': '36480', 'ceremcem': '18069', 'arcol': '2320',
'qingfeng.xia': '6825', 'MaurinoWeb': '15573', 'hhassey': '6158', 'Brutha': '5971', 'lainegates': '1295', 'FC-Architecter': '19079', 'papy': '21427', 'chrisf': '2578', 'nahshon': '1973', 'MSOlsen65': '29727', 'ediloren': '1783', 'dan-miel': '21478', 'jean.thil': '6498', 'pl7i92LCNC': '24265', 'falviani': '25270', 'foxint': '7589', 'project4': '1944', 'j-dowsett': '652', 'un1corn': '7434', 'kwahooo': '254', 'qwerty_f': '36471', 'drum22': '40830', 'Luixx': '18322', 'paul18': '3622', 'serrepattes': '15169',
'cram': '11546', 'a3bksll47': '19044', 'joel': '12631', 'yoshimitsuspeed': '1337', 'heron': '30887', 'HartmutG': '10744', 'Smiling_user': '35383', 'josegegas': '13017', 'gdo35': '858', 'spanner888': '25590', 'dprojects': '13594', 'ProBowlUk': '4171', 'C_h_o_p_i_n': '25037', 'JussyAD': '41661', 'rentlau_64': '4246', 'Do': '11781', 'Piero69': '26052', 'tonyaimer': '27612', 'AR795': '20058', 'heilo': '12525', 'iplayfast': '27301', 'hardeeprai': '255', 'Turro75': '9800', 'Wsk8': '22665', 'Fat-Zer': '4325',
'Grub': '37710', 'Sam': '8192', 'TomB19': '25136', 'agima2': '12024', 'jmarie3D': '31172', 'Norus': '6964', 'S.N.A.L': '3080', 'PeterSD': '48559', 'Olav': '11423', 'Renato': '5905', 'kanagan': '5980', 'Routerworks': '19363', 'tom': '5727', 'Spindoctor': '6848', 'm42kus': '3911', 'freecc': '18704', 'ikua': '14240', 'grandcross': '6949', 'JellyPalm': '30355', 'TopDown': '20410', 'babaroga': '9780', 'Leatherman': '12789', 'cblt2l': '251', 'psicofil': '789', 'japie': '3493',
'turn211': '39797', 'crobar': '3893', 'christi': '22009', 'gbroques': '29296', 'industromatic': '2997', 'clytle374': '2207', 'nyholku': '12058', 'doia': '42412', 'ajoeiam': '24132', 'ABeton': '27254', 'CrashedAgain': '29187', 'wvmarle': '4214', 'run_the_race': '39789', 'derschutzhund': '3592', 'ToniTen': '36030', 'sanguinariojoe': '574', 'perot': '16372', '1D_Inc': '23696', 'etrombly': '28581', 'cadcam': '30470', 'TedM': '30824', 'crashfridh': '1483', 'Anderl': '17746', 'Chaospilot': '22486', 'Eric': '25272',
'ektus': '1283', 'Eric': '25272', 'xbit': '36362', 'bhs67': '5322', 'hyarion': '34430', 'Mongrel_Shark': '10595', 'zbigg': '17804', '2cv001': '28971', 'oddtopus': '10205', 'exsolvespacer': '39536', 'jakob': '3595', 'FemUser': '16291', 'luggw1': '13046', 'Cyril': '16289', 'jmcornil': '37409', 'kcleung': '512', 'm0n5t3r': '12818', 'Galahad': '20671', 'schnebeck': '15579', 'Crossleyuk': '20393', 'Hannu': '6761', 'HardRock': '18990', 'IzzY': '14602', 'maker': '19242', 'waebbl': '21119',
'bmsaus4ax': '37689', 'rebeltaz': '24648', 'Koemi': '17917', 'nic': '24865', 'xibinke': '5703', 'ifohancroft': '25644', 'albertdela': '16540', 'lot': '25410', 'jonasthomas': '873', 'Jos': '31258', 'jonasb': '38748', 'Ukacor': '24411', 'jmplonka': '11876', 'JiPe38': '28975', 'mfasano': '19405', 'KAP': '28935', 'scrungy_doolittle': '20416', 'doubters': '8165', 'smktec': '29984', 'eason': '5803', 'OakLD': '18151', 'brjhaverkamp': '6838', 'zohozer': '3684', 'roivai': '12794', 'prrvchr': '4119',
'hds': '13102', 'hholgi': '21563', 'bbear123': '2465', 'xf3qc': '16073', 'kreso-t': '20975', 'more11': '4079', 'alex::freecad': '3892', 'aerospaceweeb': '41416', 'cnirbhay': '9824', 'IanP': '7957', 'heideachim': '18226', 'felixlee868': '26041', 'seppelw': '18203', 'aguseguedre': '535', 'frecd': '5116', 'R23D': '29332', 'F_Rosa': '7885', 'Petrikas': '38333', 'freecadlzh': '30003', 'mrdic': '14954', 'Petert': '7290', 'duckmug': '42568', 'Gauthier': '3532', 'czinehuba': '21877', 'm.cavallerin': '20120',
'Gauthier': '3532', 'wagnerlip': '29985', 'abasel': '21318', 'NC3D': '10606', 'sibelius': '29023', 'mack5': '15924', 'Linden': '6627', 'manos': '37638', 'EkaitzEsteban': '21473', 's-light': '5425', 'NateM': '2390', 'kaih': '28962', 'julieng': '22492', 'Sidemountyucatan': '24853', 'django013': '5218', 'Bertel': '37383', 'lhagan': '108', 'PMac': '16406', 'MrRossi': '11492', 'spikey': '7624', 'Daniel84': '13510', 'Andr': '7655', 'PunyTune': '23110', 'bzb.dev001': '25224', 'godblessfq': '6436',
'godblessfq': '6436', 'DrBwts': '4281', 'neondata': '1573', 'mikeprice99': '34491', 'watsug': '36528', 'balazs': '4211', 'atzensepp': '5514', 'foadsf': '5587', 'onesz': '729', 'Gregor': '30570', 'Rost': '37043', 'oldestfox': '1910', 'Magnum56': '19253', 'hinckel': '4480', 'jmh': '4317', 'Serchu': '5411', 'miniellipse': '19613', 'Enyalios': '13094', 'Quaoar': '15966', 'Xav-83': '3750', 'A_3': '7639', 'andre': '2370', 'iogui': '24401', 'dulouie': '16780', 'j9lemmon': '2018',
        #add new users and userids here
        #some examples:
        #'Enyalios': '13094',
        #'agima2': '12024'

        }
def QIconFromXPMString(xpm_string):
    xpm = xpm_string.replace("\"","").replace(',','').splitlines()[4:-1]
    pixmap = QtGui.QPixmap(xpm)
    icon = QtGui.QIcon(pixmap)
    return icon

__icon__ = """
/* XPM */
static char *_648433055688[] = {
/* columns rows colors chars-per-pixel */
"64 64 4 1 ",
"  c black",
". c #FD8711",
"X c #FFC90E",
"o c None",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooXXXXXXXoooooooooooooooooooooooooooooo",
"ooooooooooooooooooooooooXXXXXXXXXXXXoooooooooooooooooooooooooooo",
"oooooooooooooooooooooooXXXXX     XXXXooooooooooooooooooooooooooo",
"ooooooooooooooooooooooXXXX         XXXXooooooooooooooooooooooooo",
"oooooooooooooooooooooXXX            XXXXoooooooooooooooooooooooo",
"oooooooooooooooooooooXX      XXX      XXXooooooooooooooooooooooo",
"ooooooooooooooooooooXXX     XXXXX      XXXoooooooooooooooooooooo",
"oooooooooooooooooooXXX    XXXXXXXXX     XXoooooooooooooooooooooo",
"ooooooooooooooooooXXX    XXXXXXXXXXX    XXoooooooooooooooooooooo",
"oooooooooooooooooXXX    XXXXXoooXXXXX   XXXooooooooooooooooooooo",
"ooooooooooooooooXXX     XXXXoooooXXXX    XXXoooooooooooooooooooo",
"ooooooooooooooooXX     XXXXooo.oooXXXX    XXoooooooooooooooooooo",
"oooooooooooooooXXX    XXXXooo...oooXXXX   XXoooooooooooooooooooo",
"oooooooooooooooXXX    XXXXoo.....oooXXX   XXXooooooooooooooooooo",
"oooooooooooooooXXX   XXXXoo.......ooXXXX   XXooooooooooooooooooo",
"oooooooooooooooXXX   XXXXoo.......ooXXXX   XXooooooooooooooooooo",
"oooooooooooooooXX    XXXXoo.......ooXXXX   XXXoooooooooooooooooo",
"ooooooooooooooXXX    XXXooo.......oooXXX    XXoooooooooooooooooo",
"oooooooooooooXXX     XXXoo.........ooXXX    XXXooooooooooooooooo",
"oooooooooooooXXX    XXXXoo.........ooXXXX    XXooooooooooooooooo",
"oooooooooooooXXX   XXXXooo.........oooXXX    XXooooooooooooooooo",
"oooooooooooooXX    XXXXoo...........ooXXX    XXXoooooooooooooooo",
"oooooooooooooXX    XXXooo...........ooXXXX   XXXoooooooooooooooo",
"ooooooooooooXXX    XXXoo............ooXXXX    XXXooooooooooooooo",
"ooooooooooooXX    XXXooo............ooXXXXX    XXooooooooooooooo",
"ooooooooooooXX    XXXoo.............ooXXXXX    XXooooooooooooooo",
"oooooooooooXXX    XXXoo.............oooXXXX    XXooooooooooooooo",
"ooooooooooXXXX    XXooo..............oooXXX    XXooooooooooooooo",
"oooooooooXXX     XXXoo................oooXXX   XXXXooooooooooooo",
"ooooooooXXX      XXXoooooooooooooooooooooXXX    XXXXoooooooooooo",
"oooooooXXXX     XXXXXoooooooooooooooooooooXXX     XXXooooooooooo",
"oXXXXXXXX     XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX    XXXXXXXXXXooo",
"oXXXXXXX      XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX     XXXXXXXXXooo",
"oXX                                                        XXooo",
"oXX                                                        XXooo",
"oXX                                                        XXooo",
"oXX                                                        XXooo",
"oXX                                                        XXooo",
"oXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXooo",
"oXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXooo",
"oXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo",
"oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo"
};
"""
#pinger = Pinger()
dlg = PingerDlg()

if __name__ == "__main__":
#    pinger.run()
    dlg.form.show()

Link

The forum discussion Pinger macro