Compare commits
2 Commits
v2.65.2.22
...
v2.65.2.24
| Author | SHA1 | Date | |
|---|---|---|---|
| bfd5bbb881 | |||
| c64991fb59 |
@@ -9,6 +9,46 @@ from gui.utils.numberFormatter import formatAmount
|
|||||||
|
|
||||||
_t = wx.GetTranslation
|
_t = wx.GetTranslation
|
||||||
|
|
||||||
|
# Mapping of repair/transfer amount attributes to their duration attribute and display name
|
||||||
|
PER_SECOND_ATTRIBUTES = {
|
||||||
|
"armorDamageAmount": {
|
||||||
|
"durationAttr": "duration",
|
||||||
|
"displayName": "Armor Hitpoints Repaired per second",
|
||||||
|
"unit": "HP/s"
|
||||||
|
},
|
||||||
|
"shieldBonus": {
|
||||||
|
"durationAttr": "duration",
|
||||||
|
"displayName": "Shield Hitpoints Repaired per second",
|
||||||
|
"unit": "HP/s"
|
||||||
|
},
|
||||||
|
"powerTransferAmount": {
|
||||||
|
"durationAttr": "duration",
|
||||||
|
"displayName": "Capacitor Transferred per second",
|
||||||
|
"unit": "GJ/s"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PerSecondAttributeInfo:
|
||||||
|
"""Helper class to store info about computed per-second attributes"""
|
||||||
|
def __init__(self, displayName, unit):
|
||||||
|
self.displayName = displayName
|
||||||
|
self.unit = PerSecondUnit(unit)
|
||||||
|
|
||||||
|
|
||||||
|
class PerSecondUnit:
|
||||||
|
"""Helper class to mimic the Unit class for per-second attributes"""
|
||||||
|
def __init__(self, displayName):
|
||||||
|
self.displayName = displayName
|
||||||
|
self.name = ""
|
||||||
|
|
||||||
|
|
||||||
|
class PerSecondAttributeValue:
|
||||||
|
"""Helper class to store computed per-second attribute values"""
|
||||||
|
def __init__(self, value):
|
||||||
|
self.value = value
|
||||||
|
self.info = None # Will be set when adding to attrs
|
||||||
|
|
||||||
|
|
||||||
def defaultSort(item):
|
def defaultSort(item):
|
||||||
return (item.metaLevel or 0, item.name)
|
return (item.metaLevel or 0, item.name)
|
||||||
@@ -36,8 +76,12 @@ class ItemCompare(wx.Panel):
|
|||||||
self.item = item
|
self.item = item
|
||||||
self.items = sorted(items, key=defaultSort)
|
self.items = sorted(items, key=defaultSort)
|
||||||
self.attrs = {}
|
self.attrs = {}
|
||||||
|
self.computedAttrs = {} # Store computed per-second attributes
|
||||||
self.HighlightOn = wx.Colour(255, 255, 0, wx.ALPHA_OPAQUE)
|
self.HighlightOn = wx.Colour(255, 255, 0, wx.ALPHA_OPAQUE)
|
||||||
self.highlightedNames = []
|
self.highlightedNames = []
|
||||||
|
self.bangBuckColumn = None # Store the column selected for bang/buck calculation
|
||||||
|
self.bangBuckColumnName = None # Store the display name of the selected column
|
||||||
|
self.columnHighlightColour = wx.Colour(173, 216, 230, wx.ALPHA_OPAQUE) # Light blue for column highlight
|
||||||
|
|
||||||
# get a dict of attrName: attrInfo of all unique attributes across all items
|
# get a dict of attrName: attrInfo of all unique attributes across all items
|
||||||
for item in self.items:
|
for item in self.items:
|
||||||
@@ -45,23 +89,66 @@ class ItemCompare(wx.Panel):
|
|||||||
if item.attributes[attr].info.displayName:
|
if item.attributes[attr].info.displayName:
|
||||||
self.attrs[attr] = item.attributes[attr].info
|
self.attrs[attr] = item.attributes[attr].info
|
||||||
|
|
||||||
|
# Compute per-second attributes for items that have both the amount and duration
|
||||||
|
for perSecondKey, config in PER_SECOND_ATTRIBUTES.items():
|
||||||
|
amountAttr = perSecondKey
|
||||||
|
durationAttr = config["durationAttr"]
|
||||||
|
perSecondAttrName = f"{perSecondKey}_per_second"
|
||||||
|
|
||||||
|
# Check if any item has both attributes
|
||||||
|
hasPerSecondAttr = False
|
||||||
|
for item in self.items:
|
||||||
|
if amountAttr in item.attributes and durationAttr in item.attributes:
|
||||||
|
hasPerSecondAttr = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if hasPerSecondAttr:
|
||||||
|
# Add the per-second attribute info to attrs
|
||||||
|
perSecondInfo = PerSecondAttributeInfo(config["displayName"], config["unit"])
|
||||||
|
self.attrs[perSecondAttrName] = perSecondInfo
|
||||||
|
self.computedAttrs[perSecondAttrName] = {
|
||||||
|
"amountAttr": amountAttr,
|
||||||
|
"durationAttr": durationAttr
|
||||||
|
}
|
||||||
|
|
||||||
# Process attributes for items and find ones that differ
|
# Process attributes for items and find ones that differ
|
||||||
for attr in list(self.attrs.keys()):
|
for attr in list(self.attrs.keys()):
|
||||||
value = None
|
value = None
|
||||||
|
|
||||||
for item in self.items:
|
for item in self.items:
|
||||||
# we can automatically break here if this item doesn't have the attribute,
|
# Check if this is a computed attribute
|
||||||
# as that means at least one item did
|
if attr in self.computedAttrs:
|
||||||
if attr not in item.attributes:
|
computed = self.computedAttrs[attr]
|
||||||
break
|
amountAttr = computed["amountAttr"]
|
||||||
|
durationAttr = computed["durationAttr"]
|
||||||
|
|
||||||
# this is the first attribute for the item set, set the initial value
|
# Item needs both attributes to compute per-second value
|
||||||
if value is None:
|
if amountAttr not in item.attributes or durationAttr not in item.attributes:
|
||||||
value = item.attributes[attr].value
|
break
|
||||||
continue
|
|
||||||
|
|
||||||
if attr not in item.attributes or item.attributes[attr].value != value:
|
# Calculate per-second value
|
||||||
break
|
amountValue = item.attributes[amountAttr].value
|
||||||
|
durationValue = item.attributes[durationAttr].value
|
||||||
|
# Duration is in milliseconds, convert to seconds
|
||||||
|
perSecondValue = amountValue / (durationValue / 1000.0) if durationValue > 0 else 0
|
||||||
|
|
||||||
|
if value is None:
|
||||||
|
value = perSecondValue
|
||||||
|
continue
|
||||||
|
|
||||||
|
if perSecondValue != value:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
# Regular attribute handling
|
||||||
|
if attr not in item.attributes:
|
||||||
|
break
|
||||||
|
|
||||||
|
if value is None:
|
||||||
|
value = item.attributes[attr].value
|
||||||
|
continue
|
||||||
|
|
||||||
|
if item.attributes[attr].value != value:
|
||||||
|
break
|
||||||
else:
|
else:
|
||||||
# attribute values were all the same, delete
|
# attribute values were all the same, delete
|
||||||
del self.attrs[attr]
|
del self.attrs[attr]
|
||||||
@@ -89,6 +176,7 @@ class ItemCompare(wx.Panel):
|
|||||||
|
|
||||||
self.toggleViewBtn.Bind(wx.EVT_TOGGLEBUTTON, self.ToggleViewMode)
|
self.toggleViewBtn.Bind(wx.EVT_TOGGLEBUTTON, self.ToggleViewMode)
|
||||||
self.Bind(wx.EVT_LIST_COL_CLICK, self.SortCompareCols)
|
self.Bind(wx.EVT_LIST_COL_CLICK, self.SortCompareCols)
|
||||||
|
self.Bind(wx.EVT_LIST_COL_RIGHT_CLICK, self.OnColumnRightClick)
|
||||||
|
|
||||||
self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.HighlightRow)
|
self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.HighlightRow)
|
||||||
|
|
||||||
@@ -105,6 +193,23 @@ class ItemCompare(wx.Panel):
|
|||||||
self.Thaw()
|
self.Thaw()
|
||||||
event.Skip()
|
event.Skip()
|
||||||
|
|
||||||
|
def OnColumnRightClick(self, event):
|
||||||
|
column = event.GetColumn()
|
||||||
|
# Column 0 is "Item", column len(self.attrs) + 1 is "Price", len(self.attrs) + 2 is "Buck/bang"
|
||||||
|
# Only allow selecting attribute columns (1 to len(self.attrs))
|
||||||
|
if 1 <= column <= len(self.attrs):
|
||||||
|
# If clicking the same column, deselect it
|
||||||
|
if self.bangBuckColumn == column:
|
||||||
|
self.bangBuckColumn = None
|
||||||
|
self.bangBuckColumnName = None
|
||||||
|
else:
|
||||||
|
self.bangBuckColumn = column
|
||||||
|
# Get the display name of the selected column
|
||||||
|
attr_key = list(self.attrs.keys())[column - 1]
|
||||||
|
self.bangBuckColumnName = self.attrs[attr_key].displayName if self.attrs[attr_key].displayName else attr_key
|
||||||
|
self.UpdateList()
|
||||||
|
event.Skip()
|
||||||
|
|
||||||
def SortCompareCols(self, event):
|
def SortCompareCols(self, event):
|
||||||
self.Freeze()
|
self.Freeze()
|
||||||
self.paramList.ClearAll()
|
self.paramList.ClearAll()
|
||||||
@@ -148,12 +253,32 @@ class ItemCompare(wx.Panel):
|
|||||||
# Remember to reduce by 1, because the attrs array
|
# Remember to reduce by 1, because the attrs array
|
||||||
# starts at 0 while the list has the item name as column 0.
|
# starts at 0 while the list has the item name as column 0.
|
||||||
attr = str(list(self.attrs.keys())[sort - 1])
|
attr = str(list(self.attrs.keys())[sort - 1])
|
||||||
func = lambda _val: _val.attributes[attr].value if attr in _val.attributes else 0.0
|
# Handle computed attributes for sorting
|
||||||
|
if attr in self.computedAttrs:
|
||||||
|
computed = self.computedAttrs[attr]
|
||||||
|
amountAttr = computed["amountAttr"]
|
||||||
|
durationAttr = computed["durationAttr"]
|
||||||
|
func = lambda _val: (_val.attributes[amountAttr].value / (_val.attributes[durationAttr].value / 1000.0)) if (amountAttr in _val.attributes and durationAttr in _val.attributes and _val.attributes[durationAttr].value > 0) else 0.0
|
||||||
|
else:
|
||||||
|
func = lambda _val: _val.attributes[attr].value if attr in _val.attributes else 0.0
|
||||||
# Clicked on a column that's not part of our array (price most likely)
|
# Clicked on a column that's not part of our array (price most likely)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
# Price
|
# Price
|
||||||
if sort == len(self.attrs) + 1:
|
if sort == len(self.attrs) + 1:
|
||||||
func = lambda i: i.price.price if i.price.price != 0 else float("Inf")
|
func = lambda i: i.price.price if i.price.price != 0 else float("Inf")
|
||||||
|
# Buck/bang
|
||||||
|
elif sort == len(self.attrs) + 2:
|
||||||
|
if self.bangBuckColumn is not None:
|
||||||
|
attr_key = list(self.attrs.keys())[self.bangBuckColumn - 1]
|
||||||
|
if attr_key in self.computedAttrs:
|
||||||
|
computed = self.computedAttrs[attr_key]
|
||||||
|
amountAttr = computed["amountAttr"]
|
||||||
|
durationAttr = computed["durationAttr"]
|
||||||
|
func = lambda i: (i.price.price / (i.attributes[amountAttr].value / (i.attributes[durationAttr].value / 1000.0)) if (amountAttr in i.attributes and durationAttr in i.attributes and i.attributes[durationAttr].value > 0 and (i.attributes[amountAttr].value / (i.attributes[durationAttr].value / 1000.0)) > 0) else float("Inf"))
|
||||||
|
else:
|
||||||
|
func = lambda i: (i.price.price / i.attributes[attr_key].value if (attr_key in i.attributes and i.attributes[attr_key].value > 0) else float("Inf"))
|
||||||
|
else:
|
||||||
|
func = defaultSort
|
||||||
# Something else
|
# Something else
|
||||||
else:
|
else:
|
||||||
self.sortReverse = False
|
self.sortReverse = False
|
||||||
@@ -166,18 +291,49 @@ class ItemCompare(wx.Panel):
|
|||||||
|
|
||||||
for i, attr in enumerate(self.attrs.keys()):
|
for i, attr in enumerate(self.attrs.keys()):
|
||||||
name = self.attrs[attr].displayName if self.attrs[attr].displayName else attr
|
name = self.attrs[attr].displayName if self.attrs[attr].displayName else attr
|
||||||
|
# Add indicator if this column is selected for bang/buck calculation
|
||||||
|
if self.bangBuckColumn == i + 1:
|
||||||
|
name = "► " + name
|
||||||
self.paramList.InsertColumn(i + 1, name)
|
self.paramList.InsertColumn(i + 1, name)
|
||||||
self.paramList.SetColumnWidth(i + 1, 120)
|
self.paramList.SetColumnWidth(i + 1, 120)
|
||||||
|
|
||||||
self.paramList.InsertColumn(len(self.attrs) + 1, _t("Price"))
|
self.paramList.InsertColumn(len(self.attrs) + 1, _t("Price"))
|
||||||
self.paramList.SetColumnWidth(len(self.attrs) + 1, 60)
|
self.paramList.SetColumnWidth(len(self.attrs) + 1, 60)
|
||||||
|
|
||||||
|
# Add Buck/bang column header
|
||||||
|
buckBangHeader = _t("Buck/bang")
|
||||||
|
if self.bangBuckColumnName:
|
||||||
|
buckBangHeader = _t("Buck/bang ({})").format(self.bangBuckColumnName)
|
||||||
|
self.paramList.InsertColumn(len(self.attrs) + 2, buckBangHeader)
|
||||||
|
self.paramList.SetColumnWidth(len(self.attrs) + 2, 80)
|
||||||
|
|
||||||
toHighlight = []
|
toHighlight = []
|
||||||
|
|
||||||
for item in self.items:
|
for item in self.items:
|
||||||
i = self.paramList.InsertItem(self.paramList.GetItemCount(), item.name)
|
i = self.paramList.InsertItem(self.paramList.GetItemCount(), item.name)
|
||||||
for x, attr in enumerate(self.attrs.keys()):
|
for x, attr in enumerate(self.attrs.keys()):
|
||||||
if attr in item.attributes:
|
# Handle computed attributes
|
||||||
|
if attr in self.computedAttrs:
|
||||||
|
computed = self.computedAttrs[attr]
|
||||||
|
amountAttr = computed["amountAttr"]
|
||||||
|
durationAttr = computed["durationAttr"]
|
||||||
|
|
||||||
|
# Item needs both attributes to display per-second value
|
||||||
|
if amountAttr in item.attributes and durationAttr in item.attributes:
|
||||||
|
amountValue = item.attributes[amountAttr].value
|
||||||
|
durationValue = item.attributes[durationAttr].value
|
||||||
|
# Duration is in milliseconds, convert to seconds
|
||||||
|
perSecondValue = amountValue / (durationValue / 1000.0) if durationValue > 0 else 0
|
||||||
|
|
||||||
|
info = self.attrs[attr]
|
||||||
|
if self.toggleView == 1:
|
||||||
|
valueUnit = formatAmount(perSecondValue, 3, 0, 0) + " " + info.unit.displayName
|
||||||
|
else:
|
||||||
|
valueUnit = str(perSecondValue)
|
||||||
|
|
||||||
|
self.paramList.SetItem(i, x + 1, valueUnit)
|
||||||
|
# else: leave cell empty
|
||||||
|
elif attr in item.attributes:
|
||||||
info = self.attrs[attr]
|
info = self.attrs[attr]
|
||||||
value = item.attributes[attr].value
|
value = item.attributes[attr].value
|
||||||
if self.toggleView != 1:
|
if self.toggleView != 1:
|
||||||
@@ -191,6 +347,27 @@ class ItemCompare(wx.Panel):
|
|||||||
|
|
||||||
# Add prices
|
# Add prices
|
||||||
self.paramList.SetItem(i, len(self.attrs) + 1, formatAmount(item.price.price, 3, 3, 9, currency=True) if item.price.price else "")
|
self.paramList.SetItem(i, len(self.attrs) + 1, formatAmount(item.price.price, 3, 3, 9, currency=True) if item.price.price else "")
|
||||||
|
|
||||||
|
# Add buck/bang values
|
||||||
|
if self.bangBuckColumn is not None and item.price.price and item.price.price > 0:
|
||||||
|
attr_key = list(self.attrs.keys())[self.bangBuckColumn - 1]
|
||||||
|
if attr_key in self.computedAttrs:
|
||||||
|
computed = self.computedAttrs[attr_key]
|
||||||
|
amountAttr = computed["amountAttr"]
|
||||||
|
durationAttr = computed["durationAttr"]
|
||||||
|
if amountAttr in item.attributes and durationAttr in item.attributes:
|
||||||
|
amountValue = item.attributes[amountAttr].value
|
||||||
|
durationValue = item.attributes[durationAttr].value
|
||||||
|
perSecondValue = amountValue / (durationValue / 1000.0) if durationValue > 0 else 0
|
||||||
|
if perSecondValue > 0:
|
||||||
|
buckBangValue = item.price.price / perSecondValue
|
||||||
|
self.paramList.SetItem(i, len(self.attrs) + 2, formatAmount(buckBangValue, 3, 3, 9, currency=True))
|
||||||
|
elif attr_key in item.attributes:
|
||||||
|
attrValue = item.attributes[attr_key].value
|
||||||
|
if attrValue > 0:
|
||||||
|
buckBangValue = item.price.price / attrValue
|
||||||
|
self.paramList.SetItem(i, len(self.attrs) + 2, formatAmount(buckBangValue, 3, 3, 9, currency=True))
|
||||||
|
|
||||||
if item.name in self.highlightedNames:
|
if item.name in self.highlightedNames:
|
||||||
toHighlight.append(i)
|
toHighlight.append(i)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user