bitmask.py: make operations more concise

This commit is contained in:
dogeystamp 2022-08-02 20:48:11 -04:00
parent c12660ba06
commit a90d767595
Signed by: dogeystamp
GPG Key ID: 7225FE3592EFFA38

View File

@ -101,9 +101,9 @@ class Bitmask:
def __repr__(self): def __repr__(self):
enum_name = fullname(self.AllFlags(0)) enum_name = fullname(self.AllFlags(0))
args = ", ".join([enum_name] + [f"{enum_name}.{flag.name}" for flag in self]) args = [enum_name] + [f"{enum_name}.{flag.name}" for flag in self]
return f"{fullname(self)}({args})" return f"{fullname(self)}({', '.join(args)})"
def __eq__(self, other): def __eq__(self, other):
"""Check equality.""" """Check equality."""
@ -114,35 +114,22 @@ class Bitmask:
else: else:
return other.value == self.value return other.value == self.value
def _flag_op(self, flag, op):
"""Apply a single flag to the bitmask.
Args:
flag (enum value): flag to apply to the bitmask.
op (function): function that returns new numerical value for the
bitmask, given the initial value and the flag.
"""
if not issubclass(type(flag), self.AllFlags):
raise TypeError(
f"can only apply {type_name(self.AllFlags)} to {type_name(self)}"
)
self.value = op(self.value, flag)
def __mask_op(self, other, op): def __mask_op(self, other, op):
"""Run operations on two bitmasks/bitmask and flag. """Run operations on two bitmasks/bitmask and flag.
Will run `__flag_op` with each flag in the other bitmask.
Args: Args:
other (Bitmask or Enum value): bitmask to run operation with. other (Bitmask or Enum value): bitmask to run operation with.
op (function): operation passed to `__flag_op` for applying individual flags. op (function): operation to run with the two bitmask values.
Returns:
Resulting Bitmask object of the operation.
""" """
new_bitmask = self.__class__(self.AllFlags, *(flag for flag in self)) new_bitmask = self.__class__(self._AllFlags)
if issubclass(type(other), type(self)): if issubclass(type(other), type(self)):
new_bitmask.value = op(self.value, other.value) new_bitmask.value = op(self.value, other.value)
elif issubclass(type(other), self.AllFlags): elif issubclass(type(other), self.AllFlags):
new_bitmask._flag_op(other, op) new_bitmask.value = op(self.value, other)
else: else:
raise TypeError( raise TypeError(
f"can only apply {type_name(self)} or {type_name(self.AllFlags)} to {type_name(self)}" f"can only apply {type_name(self)} or {type_name(self.AllFlags)} to {type_name(self)}"
@ -150,10 +137,6 @@ class Bitmask:
return new_bitmask return new_bitmask
def add(self, other):
"""Add single flag to the bitmask."""
self._flag_op(other, lambda a, b: a | b)
def __add__(self, other): def __add__(self, other):
"""Implement + operator.""" """Implement + operator."""
return self.__mask_op(other, lambda a, b: a | b) return self.__mask_op(other, lambda a, b: a | b)
@ -162,6 +145,10 @@ class Bitmask:
"""Alias the + operator in reverse.""" """Alias the + operator in reverse."""
return self.__add__(other) return self.__add__(other)
def add(self, other):
"""Add flag to the bitmask."""
self.value = (self + other).value
def __or__(self, other): def __or__(self, other):
"""Implement | operator.""" """Implement | operator."""
return self + other return self + other
@ -203,7 +190,7 @@ class Bitmask:
f"can only discard {type_name(self._AllFlags)} from {type_name(self)}" f"can only discard {type_name(self._AllFlags)} from {type_name(self)}"
) )
return self._flag_op(flag, lambda a, b: a & ~b) self.value = self.__mask_op(flag, lambda a, b: a & ~b).value
def remove(self, flag): def remove(self, flag):
"""Remove `flag` from the bitmask. """Remove `flag` from the bitmask.