-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcommon_utils.py
178 lines (159 loc) · 6.4 KB
/
common_utils.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
"""
Skinet (Segmentation of the Kidney through a Neural nETwork) Project
Common display/math methods
Copyright (c) 2021 Skinet Team
Licensed under the MIT License (see LICENSE for details)
Written by Adrien JAUGEY
"""
from datetime import datetime
import numpy as np
def progressBar(value, maxValue, prefix="", suffix="", forceNewLine=False, size=20, full='█', cursor='▒',
empty='░'):
"""
Prints a progress bar
Based on https://stackoverflow.com/questions/6169217/replace-console-output-in-python
:param value: the current progress value
:param maxValue: the maximum value that could be given
:param prefix: text to display before the progress bar
:param suffix: text to display after the progress bar
:param forceNewLine: False by default, if True a new line will be created even if not at the end
:param size: size of the bar itself
:param full: the character to use for the completed part of the bar
:param cursor: the character to use for the current position
:param empty: the character to use for the empty part of the bar
:return: None
"""
percent = float(value) / maxValue
nbFullChar = int(percent * size)
bar = full * nbFullChar + (cursor if percent > 0 and nbFullChar < size else "")
emptyBar = empty * (size - len(bar))
print(f'\r{prefix} {bar}{emptyBar} {percent: 6.2%} {suffix}',
end='\n' if value == maxValue or forceNewLine else "", flush=True)
def progressText(value, maxValue, onlyRaw=False, onlyPercent=False):
if onlyRaw and onlyPercent:
return None
elif onlyRaw:
return f"({value}/{maxValue})"
else:
percent = float(value) / maxValue
if onlyPercent:
return f"({percent:06.2%})"
else:
return f"({value}/{maxValue} | {percent:06.2%})"
def formatTime(seconds: int, minutes: int = 0, hours: int = 0):
"""
Returns a string representing the given time
:param seconds: number of seconds
:param minutes: if given, number of minutes adding to seconds
:param hours: if given, number of hours adding to seconds
:return: formated time as string of "hh:mm:ss" format
"""
seconds += minutes * 60 + hours * 3600
seconds = int(seconds)
h = seconds // 3600
m = (seconds % 3600) // 60
s = seconds % 60
hText = f"{h:02d}:" if h != 0 else ""
mText = f"{m:02d}:" if h + m != 0 else ""
sText = f"{s:02d}" if h + m != 0 else f"{s:02d}s"
return f"{hText}{mText}{sText}"
def formatDate(date: datetime = None, dateOnly=False, timeOnly=False):
"""
Returns date as string
:param date: datetime.datetime object to use specific date, current date and time by default
:param dateOnly: if True, will only display the date, not the time
:param timeOnly: if True, will only display the time, not the date
:return: formatted date as string
"""
if date is None:
date = datetime.now()
else:
assert type(date) is datetime, "Provide date as datetime.datetime object"
dateFormat = '%Y-%m-%d'
timeFormat = '%H-%M-%S'
if dateOnly and timeOnly:
outputFormat = ""
elif dateOnly:
outputFormat = dateFormat
elif timeOnly:
outputFormat = timeFormat
else:
outputFormat = f"{dateFormat}_{timeFormat}"
return date.strftime(outputFormat)
def format_number(num, maxLength=None):
"""
Formats a number using a metric prefix such as K (Kilo), M (Mega)... up to Y (Yotta)
:param num: the number to format
:param maxLength: maximum length of the formatted number, will reduce number of decimals from 2 to 0 depending on
the length but will at least return abs(num / 1000^N) followed by the metric prefix for 1000^N.
:return: the formatted number with up to 2 decimal digits
"""
# https://stackoverflow.com/questions/579310/formatting-long-numbers-as-strings-in-python
suffixes = ['', 'K', 'M', 'G', 'T', 'P', 'E', 'Z', 'Y']
magnitude = 0
isInt = np.issubdtype(type(num), np.integer)
while abs(num) >= 1000 and magnitude < len(suffixes) - 1:
magnitude += 1
num /= 1000.0
if magnitude == 0 and isInt:
return str(int(num))
if maxLength is not None:
possibleText = [f"{num:.2f}{suffixes[magnitude]}",
f"{num:.1f}{suffixes[magnitude]}",
f"{num:.0f}{suffixes[magnitude]}"]
for text in possibleText:
if len(text) <= maxLength:
return text
return possibleText[-1]
return f"{num:.1f}{suffixes[magnitude]}"
def combination(setSize, combinationSize):
"""
Computes the number of k-combinations in a set
Source : https://python.jpvweb.com/python/mesrecettespython/doku.php?id=combinaisons
:param setSize: number of elements in the set
:param combinationSize: number of elements in a combination
:return: number of k-combinations
"""
if combinationSize > setSize // 2:
combinationSize = setSize - combinationSize
x = 1
y = 1
i = setSize - combinationSize + 1
while i <= setSize:
x = (x * i) // y
y += 1
i += 1
return x
def rgb_to_hex(r, g=0, b=0, a=0, alpha=False):
"""
Returns the hexadecimal string of a color
:param r: red channel
:param g: green channel
:param b: blue channel
:param a: alpha channel
:param alpha: if True, alpha will be used
:return: color in a string format such as #abcdef
"""
if type(r) is not list:
color = [r, g, b, a]
else:
color = r
hex = "#"
for channel in color[:(4 if alpha else 3)]:
temp = channel if type(channel) is int else int(round(channel * 255))
hex += f'{temp:02x}'
return hex
def hex_to_rgb(hex_str: str, normalized=False):
"""
Returns a list of channel values of a hexadecimal string color
:param hex_str: color in hexadecimal string format (ex: '#abcdef')
:param normalized: if True, color will be float between 0 and 1
:return: list of all channels
"""
hex_str = hex_str.replace('#', '').replace('0x', '')
color = []
for idx in range(0, len(hex_str), 2):
color.append(int(hex_str[idx:idx + 2], 16))
if normalized:
color = [float(c) / 255 for c in color]
return color