v欣 发表于 2022-2-22 11:48:53

原神-祈愿模拟系统

本帖最后由 v欣 于 2022-2-23 08:52 编辑

我原神抽奖太非了,写来写代码骗骗自己

本人小白,还没过计算机二级{:10_250:}


文件解压后将两个 xlsx 文件放在 py 文件的工作目录下即可!

在window的cmd中怎么实现输出彩色字体, 让指教{:10_254:}


import random
import openpyxl
from openpyxl.styles import Font, PatternFill
import datetime
import os


WD = os.getcwd()
''' WD = '/home/vxin/桌面/python/games/yuanshen' '''
os.chdir(WD)

ResidentAwardPoolPath = './奖池.xlsx'
WinningProbabilityPath = './星级及概率.xlsx'
LotteryRecordPath = './记录.xlsx'

TitleBlockFont = Font(bold=True)
FiveStarFont = Font(bold=True, color='00FFFF00')
FourStarFont = Font(bold=True, color='00FF00FF')
AllFill = PatternFill(fill_type='solid', fgColor='00C0C0C0')


def outprint(star, type, name):
    """打印本次抽中的奖品

    Args:
      star (int): 星级
      type (str): 类型
      name (str): 名称
    """
    if star == 5:
      color = 33
    elif star == 4:
      color = 35
    elif star == 3:
      color = 34
    else:
      color = 31
    #print('\033[0;{};40m星级:{} 种类:{} 名称:{}\033[0m'.format(color, star, type, name))
    print('星级:{} 种类:{} 名称:{}'.format(star, type, name))

def query(project):
    """打印该抽奖项目的抽奖记录

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)
    """   
    if os.path.exists(LotteryRecordPath):
      wb = openpyxl.load_workbook(LotteryRecordPath)
      ws = wb
      for i in range(1, len()):
            if ws['D%d'%(i+1)].value == 5:
                color = 33
            elif ws['D%d'%(i+1)].value == 4:
                color = 35
            elif ws['D%d'%(i+1)].value == 3:
                color = 34
            else:
                color = 31
            print('\033[0;{};40m第{}次,抽到{}星{},名称:{}\033[0m'.format(
                  color, ws['E%d'%(i+1)].value, ws['D%d'%(i+1)].value,
                         ws['C%d'%(i+1)].value, ws['B%d'%(i+1)].value))
    else:
      print('未找到 %s 文件!'%(LotteryRecordPath))

def get_frequency(project):
    """查询抽各星级对应的概率(%)、保底次数、为本期物品概率(%)。加载当前工作目录下的文件 星级及概率.xlsx

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)

    Returns:
      dict: {星级:list(概率(%), 保底, 为本期物品概率(%))}
    """   
    wb = openpyxl.load_workbook(WinningProbabilityPath)
    ws = wb
    d = {}
    for y in (2, 3, 4):
      for x in ('A', 'B', 'C', 'D'):
            if x == 'A':
                star = ws['%s%d'%(x, y)].value
            elif x == 'B':
                frequency = ws['%s%d'%(x, y)].value
            elif x == 'C':
                minimum_times = ws['%s%d'%(x, y)].value
            else:
                ipota = ws['%s%d'%(x, y)].value# Item probability of this activity
      d.update({star:})
    return d

def get_jackpot(project):
    """导入奖池内容。加载当前工作目录下的文件 奖池.xlsx

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)

    Returns:
      tuple: 三个列表[星级, 类型, 名称]
    """   
    wb = openpyxl.load_workbook(ResidentAwardPoolPath)
    ws = wb
    five_star = []
    four_stat = []
    three_star = []
    t = 0
    for row in ws.values:
      t += 1
      if t == 1:
            pass
      else:
            tl = tuple(row)
            if tl == 5:
                five_star.append(tl)
            elif tl == 4:
                four_stat.append(tl)
            else:
                three_star.append(tl)
    return (five_star, four_stat, three_star)

def keep_records(project, star, type, name, remakes):
    """保存抽奖记录。保存文当前工作目录下的 记录.xlsx

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)
      star (int): 星级
      type (str): 类型
      name (str): 名称
      remakes (str)): 限定角色标记奖池备注
    """   
    if os.path.exists(LotteryRecordPath):
      wb = openpyxl.load_workbook(LotteryRecordPath)
      ws = wb
      p = len()# p为列表的行数
      ws['A%d'%(p+1)] = datetime.datetime.today()# A列为时间
      ws['A%d'%(p+1)].number_format = "yyyy-mm-dd hh:mm:ss"
      ws['B%d'%(p+1)] = name# B列为名称
      ws['C%d'%(p+1)] = type# C列为种类
      ws['D%d'%(p+1)] = int(star)# D列为星级
      if p == 1:
            ws['E%d'%(p+1)] = 1# E列为总次数
            ws['F%d'%(p+1)] = 1# F列为距离上次5星的次数
            ws['G%d'%(p+1)] = 1# G列为距离上次4星的次数
      else:
            ws['E%d'%(p+1)] = ws['E%d'%(p)].value + 1
            if ws['D%d'%(p)].value == 5:
                ws['F%d'%(p+1)] = 1
                ws['G%d'%(p+1)] = 1
            elif ws['D%d'%(p)].value == 4:
                ws['F%d'%(p+1)] = ws['F%d'%(p)].value + 1
                ws['G%d'%(p+1)] = 1
            else:
                ws['F%d'%(p+1)] = ws['F%d'%(p)].value + 1
                ws['G%d'%(p+1)] = ws['G%d'%(p)].value + 1
      ws['H%d'%(p+1)] = remakes# H列为备注
      for i in range(ord('A'), ord('I')):
            ws['%s%d'%(chr(i), p+1)].fill = AllFill
            if star == 5:
                ws['%s%d'%(chr(i), p+1)].font = FiveStarFont
            elif star == 4:
                ws['%s%d'%(chr(i), p+1)].font = FourStarFont
            else:
                pass
      wb.save(LotteryRecordPath)
    else:
      wb = openpyxl.Workbook()
      ws1 = wb.create_sheet("常驻祈愿")
      ws2 = wb.create_sheet("角色活动祈愿")
      ws3 = wb.create_sheet("武器活动祈愿")
      del wb['Sheet']
      for i in (1, 2, 3):
            exec('ws%d.column_dimensions["A"].width = 20'%(i))
            exec('ws%d.column_dimensions["B"].width = 15'%(i))
            exec('ws%d.column_dimensions["F"].width = 18'%(i))
            exec('ws%d.column_dimensions["G"].width = 18'%(i))
            exec('ws%d["A1"] = "时间"'%(i))
            exec('ws%d["B1"] = "名称"'%(i))
            exec('ws%d["C1"] = "类别"'%(i))
            exec('ws%d["D1"] = "星级"'%(i))
            exec('ws%d["E1"] = "总次数"'%(i))
            exec('ws%d["F1"] = "距离上次5星的次数"'%(i))
            exec('ws%d["G1"] = "距离上次4星的次数"'%(i))
            exec('ws%d["H1"] = "备注"'%(i))
            for j in range(ord('A'), ord('I')):
                exec('ws%d["%s1"].font = TitleBlockFont'%(i, chr(j)))
                exec('ws%d["%s1"].fill = AllFill'%(i, chr(j)))
            exec('ws%d.freeze_panes = "A2"'%(i))
      wb.save(LotteryRecordPath)
      keep_records(project, star, type, name, remakes)

def minimum_mark(project, x=0):
    """返回5星及4星的保底标志

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)
      x (int): 5星保底次数

    Returns:
      tuple: 5星的标记, 4星的标记[, 上次5星角色名称, 上次4星角色名称]
    """   
    if os.path.exists(LotteryRecordPath):
      wb = openpyxl.load_workbook(LotteryRecordPath)
      ws = wb
      p = len()
      five = ws['F%d'%(p)].value
      four = ws['G%d'%(p)].value
      if project == '常驻祈愿':
            return (five, four)
      else:
            def last(x):
                lfive = ['N']
                lfour = ['N']
                if p <= x:
                  for row in ws.values:
                        l = list(row)
                        if l == 5:
                            lfive.append(l)
                        elif l == 4:
                            lfour.append(l)
                else:
                  for row in list(ws.values):
                        l = list(row)
                        if l == 5:
                            lfive.append(l)
                        elif l == 4:
                            lfour.append(l)
                return (lfive[-1], lfour[-1])
            if project == '角色活动祈愿':
                (last_five, last_four) = last(x)
                return (five, four, last_five, last_four)
            elif project == '武器活动祈愿':
                wb1 = openpyxl.load_workbook(ResidentAwardPoolPath)
                ws1 = wb['武器活动祈愿']
                m = ws1['G1'].value
                (last_five, last_four) = last(x)
                return (five, four, last_five, last_four, m)
    else:
      if project == '常驻祈愿':
            return (0, 0)
      elif project == '角色活动祈愿':
            return (0, 0, 0, 0)
      elif project == '武器活动祈愿':
            wb1 = openpyxl.load_workbook(ResidentAwardPoolPath)
            ws1 = wb1['武器活动祈愿']
            m = ws1['G1'].value
            return (0, 0, 0, 0, m)
      else:
            pass

def get_jackpot2(project, remakes):
    (five_star1, four_star1, three_star1) = get_jackpot('常驻祈愿')
    if project == '常驻祈愿':
      return (five_star1, four_star1, three_star1)
    else:
      if project == '角色活动祈愿':
            for f in get_jackpot(project):
                if f == remakes[:-1]:
                  five_star2 =
            four_star2 = get_jackpot(project)
            for a in five_star1:
                if a == 5 and a == '武器':
                  five_star1.remove(a)
            for a in four_star2:
                if a in four_star1:
                  four_star1.remove(a)
      elif project == '武器活动祈愿':
            five_star2 = []
            for f in get_jackpot(project):
                five_star2.append(f)
            four_star2 = get_jackpot(project)
            for a in five_star1:
                if a == 5 and a == '角色':
                  five_star1.remove(a)
            for a in four_star2:
                if a in four_star1:
                  four_star1.remove(a)
      return (five_star1, four_star1, three_star1, five_star2, four_star2)

def ger_frequency2(project):
    d = get_frequency(project)# star:
    three_frequency = int(d*10)
    (five_frequency, five_minimun_times) = (int(d*10), int(d))
    (four_frequnecy, four_minimun_times) = (int(d*10), int(d))
    if project == '常驻祈愿':
      return (five_frequency, five_minimun_times, four_frequnecy, four_minimun_times, three_frequency)
    else:
      (five_frequency, five_minimun_times, five_ipota) = (int(d*10), int(d), int(d*10))
      (four_frequnecy, four_minimun_times, four_ipota) = (int(d*10), int(d), int(d*10))
      return (five_frequency, five_minimun_times, five_ipota, four_frequnecy, four_minimun_times, four_ipota, three_frequency, d)

def luck_draw(project, jackpot2, frequency2):
    """抽奖模块

    Args:
      project (str): 抽奖项目(常驻祈愿、角色活动祈愿、武器活动祈愿)
      jackpot2 (tupe): get_jackpot2的返回值
      frequency2 (tupe): get_frequency2的返回值

    Returns:
      list: 星级, 类别, 名称
    """   
    if project == '常驻祈愿':
      (five_star1, four_star1, three_star1) = jackpot2
      (five_mark, four_mark) = minimum_mark(project)
      (five_frequency, five_minimun_times, four_frequnecy, four_minimun_times, three_frequency) = frequency2
      if five_mark == five_minimun_times - 1:# 五星保底
            return random.choice(five_star1)
      else:
            if four_mark == four_minimun_times - 1:# 四星保底
                return random.choice(four_star1)
            else:# 常规
                x = random.randint(1, five_frequency+four_frequnecy+three_frequency)
                if x in range(1, five_frequency+1):
                  return random.choice(five_star1)
                elif x in range(five_frequency+1, five_frequency+four_frequnecy+1):
                  return random.choice(four_star1)
                elif x in range(five_frequency+four_frequnecy+1, five_frequency+four_frequnecy+three_frequency+1):
                  return random.choice(three_star1)
                else:
                  return ['!', '!', '!']
    else:
      (five_star1, four_star1, three_star1, five_star2, four_star2) = jackpot2
      (five_frequency, five_minimun_times, five_ipota, four_frequnecy, four_minimun_times, four_ipota, three_frequency, d) = frequency2
      if project == '角色活动祈愿':
            (five_mark, four_mark, last_five, last_four) = minimum_mark(project, d)
            if five_mark == five_minimun_times - 1:# 五星保底
                if last_five in for n in five_star1]:# 本期五星保底
                  return random.choice(five_star2)
                else:# 小保底
                  x = random.randint(1, 1000)
                  if x in range(1, five_ipota+1):
                        return random.choice(five_star2)
                  else:
                        return random.choice(five_star1)
            else:
                if four_mark == four_minimun_times - 1:# 四星保底
                  if last_four in for n in four_star1]:# 本期四星保底
                        return random.choice(four_star2)
                  else:# 常规四星保底
                        x = random.randint(1, 1000)
                        if x in range(1, four_ipota+1):
                            return random.choice(four_star2)
                        else:
                            return random.choice(four_star1)
                else:# 常规
                  x = random.randint(1, five_frequency+four_frequnecy+three_frequency)
                  if x in range(1, five_frequency+1):
                        i = random.randint(1, 1000)
                        if x in range(1, five_ipota+1):
                            return random.choice(five_star2)
                        else:
                            return random.choice(five_star1)
                  elif x in range(five_frequency+1, four_frequnecy+five_frequency+1):
                        i = random.randint(1, 1000)
                        if i in range(1, four_ipota):
                            return random.choice(four_star2)
                        else:
                            return random.choice(four_star1)
                  elif x in range(four_frequnecy+five_frequency+1, three_frequency+four_frequnecy+five_frequency+1):
                        return random.choice(three_star1)
                  else:
                        return['!', '!', '!']
      elif project == '武器活动祈愿':
            (five_mark, four_mark, last_five, last_four, m_value) = minimum_mark(project, d)
            wb = openpyxl.load_workbook(ResidentAwardPoolPath)
            ws = wb
            od_weapon = ws['E1'].value
            if five_mark == five_minimun_times - 1:# 五星保底
                if m_value == 2:# 定轨保底
                  revise_weapon(2)
                  return
                else:# 非定轨保底
                  if last_five in for n in five_star1]:# 本期五星保底
                        j = random.choice(five_star2)
                        if j == od_weapon:
                            revise_weapon(2)
                        return j
                  else:# 常规五星保底
                        x = random.randint(1, 1000)
                        if x in range(1, five_ipota+1):
                            j = random.choice(five_star2)
                            if j == od_weapon:
                              revise_weapon(2)
                            else:
                              revise_weapon(1)
                            return j
                        else:
                            revise_weapon(1)
                            return random.choice(five_star1)
            else:
                if four_mark == four_minimun_times - 1:# 四星保底
                  if last_four in for n in four_star1]:# 本期四星保底
                        return random.choice(four_star2)
                  else:# 常规四星保底
                        x = random.randint(1, 1000)
                        if x in range(1, four_ipota+1):
                            return random.choice(four_star2)
                        else:
                            return random.choice(four_star1)
                else:# 常规
                  x = random.randint(1, five_frequency+four_frequnecy+three_frequency)
                  if x in range(1, five_frequency+1):
                        i = random.randint(1, 1000)
                        if x in range(1, five_ipota+1):
                            j = random.choice(five_star2)
                            if j == od_weapon:
                              revise_weapon(2)
                            return j
                        else:
                            revise_weapon(1)
                            return random.choice(five_star1)
                  elif x in range(five_frequency+1, four_frequnecy+five_frequency+1):
                        i = random.randint(1, 1000)
                        if i in range(1, four_ipota):
                            return random.choice(four_star2)
                        else:
                            return random.choice(four_star1)
                  elif x in range(four_frequnecy+five_frequency+1, three_frequency+four_frequnecy+five_frequency+1):
                        return random.choice(three_star1)
                  else:
                        return['!', '!', '!']

def revise_weapon(mode):
    """武器定轨操作

    Args:
      mode (int): 0:取消定轨;1:正常;2.命值清零;3:修改定轨
    """   
    wb = openpyxl.load_workbook(ResidentAwardPoolPath)
    ws = wb['武器活动祈愿']
    if mode == 0:
      ws['E1'] = None
      ws['G1'] = None
    elif mode == 1:
      if ws['G1'].value != None and ws['E1'].value != None:
            ws['G1'] = ws['G1'].value + 1
      else:
            pass
    elif mode == 2:
      ws['G1'] = 0
    elif mode == 3:
      ws['G1'] = 0
      nl = []
      for row in ws.values:
            l = list(row)
            ifl == 5:
                nl.append(l)
      def ch():
            t = 1
            for n in nl:
                print('%d : %s'%(t, n))
                t += 1
            c = int(input('请选择:'))
            if c-1 in range(len(l)):
                ws['E1'] = nl
            else:
                ch()
      ch()
    wb.save(ResidentAwardPoolPath)

def fecondary_menu(project, remakes=None):# 二级菜单
    secondary_cycle = 1
    while secondary_cycle:
      lis = get_jackpot2(project, remakes)
      tup = ger_frequency2(project)
      print('1.来十发\n2.来一发\n3.查询抽奖记录\n4.返回上一级\n5.退出')
      choice = int(input('请选择:'))
      if choice == 1:
            for t in range(10):
                l = luck_draw(project, lis, tup)
                (star, typ, name) = (l, l, l)
                keep_records(project, star, typ, name, remakes)
                outprint(star, typ, name)
      elif choice == 2:
            l = luck_draw(project, lis, tup)
            (star, typ, name) = (l, l, l)
            keep_records(project, star, typ, name, remakes)
            outprint(star, typ, name)
      elif choice == 3:
            query(project)
      elif choice == 4:
            secondary_cycle = 0
      elif choice == 5:
            exit()
      else:
            continue

def main():# 主菜单
    print('!!!请将 "%s, %s" 文件移动在:%s!!!\n\n'%(
      ResidentAwardPoolPath, WinningProbabilityPath, WD))
    print('='*20 + '欢迎使用原神抽奖模拟系统' + '='*20)
    primary_cycle = 1
    while primary_cycle:
      print('1.常驻祈愿\n2.角色活动祈愿\n3.武器活动祈愿\n4.退出')
      a = int(input('请选择:'))
      if a in (1, 3):
            if a == 1:
                Project = '常驻祈愿'
            else:
                Project = '武器活动祈愿'
                while 1:
                  wb = openpyxl.load_workbook(ResidentAwardPoolPath)
                  ws = wb['武器活动祈愿']
                  od_weapon = ws['E1'].value
                  numerical_value = ws['G1'].value
                  if od_weapon == None and numerical_value == None:
                        print('当前无定轨!')
                  else:
                        print('当前定轨武器:%s\n定轨命值为:%d'%(od_weapon, numerical_value))
                  print('1.继续\n2.修改定轨\n3.取消定轨')
                  c = int(input('请选择:'))
                  if c == 1:
                        break
                  elif c == 2:
                        revise_weapon(3)
                        continue
                  elif c == 3:
                        revise_weapon(0)
                        continue
                  else:
                        continue
            fecondary_menu(Project)
      elif a == 2:
            Project = '角色活动祈愿'
            wb = openpyxl.load_workbook(ResidentAwardPoolPath)
            ws = wb
            nl = []
            for row in ws.values:
                l = list(row)
                ifl == 5:
                  nl.append(l)
            def choice():
                t = 1
                for n in nl:
                  print('%d : %s池'%(t, n))
                  t += 1
                c = int(input('请选择:'))
                if c-1 in range(len(l)):
                  fecondary_menu(Project, remakes='%s'%(nl+'池'))
                else:
                  choice()
            choice()
      elif a == 4:
            primary_cycle = 0
      else:
            continue


if __name__ == '__main__':
    main()

沐雨尘枫 发表于 2022-5-28 15:30:14

Windows的cmd中显色用color/?即可查询为甚无人看此贴发到csdn中吧那儿比较八卦

v欣 发表于 2022-5-31 10:46:28

沐雨尘枫 发表于 2022-5-28 15:30
Windows的cmd中显色用color/?即可查询为甚无人看此贴发到csdn中吧那儿比较八卦

在Ubuntu下没啥问题,到window10就没达到想要的效果
页: [1]
查看完整版本: 原神-祈愿模拟系统