diff --git a/src/aiidalab_qe/app/configuration/advanced/advanced.py b/src/aiidalab_qe/app/configuration/advanced/advanced.py
index 563c1377f..4a43bc40a 100644
--- a/src/aiidalab_qe/app/configuration/advanced/advanced.py
+++ b/src/aiidalab_qe/app/configuration/advanced/advanced.py
@@ -50,6 +50,11 @@ def __init__(self, model: AdvancedConfigurationSettingsModel, **kwargs):
"kpoints_distance",
)
+ # NOTE connect pseudos first, as some settings depend on it
+ pseudos_model = PseudosConfigurationSettingsModel()
+ self.pseudos = PseudosConfigurationSettingsPanel(model=pseudos_model)
+ model.add_model("pseudos", pseudos_model)
+
smearing_model = SmearingConfigurationSettingsModel()
self.smearing = SmearingConfigurationSettingsPanel(model=smearing_model)
model.add_model("smearing", smearing_model)
@@ -64,10 +69,6 @@ def __init__(self, model: AdvancedConfigurationSettingsModel, **kwargs):
self.hubbard = HubbardConfigurationSettingsPanel(model=hubbard_model)
model.add_model("hubbard", hubbard_model)
- pseudos_model = PseudosConfigurationSettingsModel()
- self.pseudos = PseudosConfigurationSettingsPanel(model=pseudos_model)
- model.add_model("pseudos", pseudos_model)
-
def render(self):
if self.rendered:
return
diff --git a/src/aiidalab_qe/app/configuration/advanced/magnetization/magnetization.py b/src/aiidalab_qe/app/configuration/advanced/magnetization/magnetization.py
index 7e4cd75d6..681d8ad9e 100644
--- a/src/aiidalab_qe/app/configuration/advanced/magnetization/magnetization.py
+++ b/src/aiidalab_qe/app/configuration/advanced/magnetization/magnetization.py
@@ -8,10 +8,10 @@ class MagnetizationConfigurationSettingsPanel(
AdvancedConfigurationSubSettingsPanel[MagnetizationConfigurationSettingsModel],
):
"""Widget to set the type of magnetization used in the calculation:
- 1) Tot_magnetization: Total majority spin charge - minority spin charge.
- 2) Starting magnetization: Starting spin polarization on atomic type 'i' in a spin polarized (LSDA or noncollinear/spin-orbit) calculation.
+ 1) Total magnetization: Total majority spin charge - minority spin charge.
+ 2) Magnetic moments: Starting spin polarization on atomic type 'i' in a spin polarized (LSDA or noncollinear/spin-orbit) calculation.
- For Starting magnetization you can set each kind names defined in the StructureData (StructureData.get_kind_names())
+ For Magnetic moments you can set each kind names defined in the StructureData (StructureData.get_kind_names())
Usually these are the names of the elements in the StructureData
(For example 'C' , 'N' , 'Fe' . However the StructureData can have defined kinds like 'Fe1' and 'Fe2')
The widget generate a dictionary that can be used to set initial_magnetic_moments in the builder of PwBaseWorkChain
@@ -39,18 +39,34 @@ def __init__(self, model: MagnetizationConfigurationSettingsModel, **kwargs):
self._on_magnetization_type_change,
"type",
)
+ self._model.observe(
+ self._on_family_change,
+ "family",
+ )
def render(self):
if self.rendered:
return
- self.description = ipw.HTML("Magnetization:")
+ self.header = ipw.HTML("Magnetization:")
+
+ self.unit = ipw.HTML(
+ value="ยตB",
+ layout=ipw.Layout(margin="2px 2px 5px"),
+ )
+
+ self.magnetization_type_help = ipw.HTML()
+ ipw.dlink(
+ (self._model, "type_help"),
+ (self.magnetization_type_help, "value"),
+ )
self.magnetization_type = ipw.ToggleButtons(
style={
"description_width": "initial",
"button_width": "initial",
},
+ layout=ipw.Layout(margin="0 0 10px 0"),
)
ipw.dlink(
(self._model, "type_options"),
@@ -73,11 +89,19 @@ def render(self):
(self.tot_magnetization, "value"),
)
+ self.tot_magnetization_with_unit = ipw.HBox(
+ children=[
+ self.tot_magnetization,
+ self.unit,
+ ],
+ layout=ipw.Layout(align_items="center"),
+ )
+
self.kind_moment_widgets = ipw.VBox()
self.container = ipw.VBox(
children=[
- self.tot_magnetization,
+ self.tot_magnetization_with_unit,
]
)
@@ -92,12 +116,17 @@ def _on_input_structure_change(self, _):
def _on_electronic_type_change(self, _):
self._switch_widgets()
+ self._model.update_type_help()
def _on_spin_type_change(self, _):
self.refresh(specific="spin")
def _on_magnetization_type_change(self, _):
self._toggle_widgets()
+ self._model.update_type_help()
+
+ def _on_family_change(self, _):
+ self._model.update_default_starting_magnetization()
def _update(self, specific=""):
if self.updated:
@@ -129,8 +158,8 @@ def _build_kinds_widget(self):
for kind_name in kind_names:
kind_moment_widget = ipw.BoundedFloatText(
description=kind_name,
- min=-4,
- max=4,
+ min=-7,
+ max=7,
step=0.1,
)
link = ipw.link(
@@ -145,7 +174,15 @@ def _build_kinds_widget(self):
],
)
self.links.append(link)
- children.append(kind_moment_widget)
+ children.append(
+ ipw.HBox(
+ children=[
+ kind_moment_widget,
+ self.unit,
+ ],
+ layout=ipw.Layout(align_items="center"),
+ )
+ )
self.kind_moment_widgets.children = children
@@ -155,23 +192,29 @@ def _switch_widgets(self):
if self._model.spin_type == "none":
children = []
else:
- children = [self.description]
+ children = [self.header]
if self._model.electronic_type == "metal":
children.extend(
[
self.magnetization_type,
+ self.magnetization_type_help,
self.container,
]
)
else:
- children.append(self.tot_magnetization)
+ children.extend(
+ [
+ self.magnetization_type_help,
+ self.tot_magnetization_with_unit,
+ ],
+ )
self.children = children
def _toggle_widgets(self):
if self._model.spin_type == "none" or not self.rendered:
return
self.container.children = [
- self.tot_magnetization
+ self.tot_magnetization_with_unit
if self._model.type == "tot_magnetization"
else self.kind_moment_widgets
]
diff --git a/src/aiidalab_qe/app/configuration/advanced/magnetization/model.py b/src/aiidalab_qe/app/configuration/advanced/magnetization/model.py
index fafd44ff5..dcf3ef704 100644
--- a/src/aiidalab_qe/app/configuration/advanced/magnetization/model.py
+++ b/src/aiidalab_qe/app/configuration/advanced/magnetization/model.py
@@ -1,8 +1,14 @@
from copy import deepcopy
import traitlets as tl
+from aiida_pseudo.groups.family import PseudoPotentialFamily
+from aiida_quantumespresso.workflows.protocols.utils import (
+ get_magnetization_parameters,
+ get_starting_magnetization,
+)
from aiidalab_qe.common.mixins import HasInputStructure
+from aiidalab_qe.utils import fetch_pseudo_family_by_label
from ..subsettings import AdvancedCalculationSubSettingsModel
@@ -17,19 +23,22 @@ class MagnetizationConfigurationSettingsModel(
"input_structure",
"electronic_type",
"spin_type",
+ "pseudos.family",
]
electronic_type = tl.Unicode()
spin_type = tl.Unicode()
+ family = tl.Unicode()
type_options = tl.List(
trait=tl.List(tl.Unicode()),
default_value=[
- ["Starting magnetization", "starting_magnetization"],
+ ["Initial magnetic moments", "starting_magnetization"],
["Total magnetization", "tot_magnetization"],
],
)
type = tl.Unicode("starting_magnetization")
+ type_help = tl.Unicode("")
total = tl.Float(0.0)
moments = tl.Dict(
key_trait=tl.Unicode(), # kind name
@@ -37,21 +46,86 @@ class MagnetizationConfigurationSettingsModel(
default_value={},
)
+ _TYPE_HELP_TEMPLATE = """
+
+ {content}
+
+ """
+
+ _DEFAULT_MOMENTS = get_magnetization_parameters()
+
+ def __init__(self, *args, **kwargs):
+ super().__init__(*args, **kwargs)
+ self._default_starting_magnetization = {}
+
def update(self, specific=""): # noqa: ARG002
if self.spin_type == "none" or not self.has_structure:
self._defaults["moments"] = {}
else:
- self._defaults["moments"] = {
- kind_name: 0.0 for kind_name in self.input_structure.get_kind_names()
- }
+ self.update_type_help()
+ self.update_default_starting_magnetization()
+ self._update_default_moments()
with self.hold_trait_notifications():
self.moments = self._get_default_moments()
+ def update_type_help(self):
+ """Update the type field help text w.r.t the current model state."""
+ if self.electronic_type == "insulator" or self.type == "tot_magnetization":
+ self.type_help = self._TYPE_HELP_TEMPLATE.format(
+ content="""
+ Constrain the desired total electronic magnetization (difference
+ between majority and minority spin charge).
+ """
+ )
+ else:
+ self.type_help = self._TYPE_HELP_TEMPLATE.format(
+ content="""
+ If a nonzero ground-state magnetization is expected, you
+ must assign a nonzero value to at least one atomic
+ type (note that the app already provide tentative initial values).
+ To simulate an antiferromagnetic state, first, if you have not
+ done so already, please use the atom tag editor (Select
+ structure -> Edit structure -> Edit atom tags) to mark atoms of
+ the species of interest as distinct by assigning each a different
+ integer tag. Once tagged, assign each an initial magnetic moments
+ of opposite sign.
+ """
+ )
+
+ def update_default_starting_magnetization(self):
+ """Update the default starting magnetization based on the structure and
+ pseudopotential family.
+ """
+ if not self.has_structure:
+ # TODO this guard shouldn't be here! It IS here only because in the present
+ # implementation, an update is called on app start. This breaks lazy loading
+ # and should be carefully checked!
+ return
+ family = fetch_pseudo_family_by_label(self.family)
+ initial_guess = get_starting_magnetization(self.input_structure, family)
+ self._default_starting_magnetization = initial_guess
+
def reset(self):
with self.hold_trait_notifications():
self.type = self.traits()["type"].default_value
self.total = self.traits()["total"].default_value
self.moments = self._get_default_moments()
+ def _update_default_moments(self):
+ family = fetch_pseudo_family_by_label(self.family)
+ self._defaults["moments"] = {
+ kind.name: self._to_moment(symbol=kind.symbol, family=family)
+ for kind in self.input_structure.kinds
+ }
+
+ def _to_moment(self, symbol: str, family: PseudoPotentialFamily) -> float:
+ """Convert the default magnetization to an initial magnetic moment."""
+ magnetization = (
+ self._default_starting_magnetization.get(symbol, 0.1)
+ if self._DEFAULT_MOMENTS.get(symbol, {}).get("magmom")
+ else 0.1
+ )
+ return round(magnetization * family.get_pseudo(symbol).z_valence, 3)
+
def _get_default_moments(self):
return deepcopy(self._defaults.get("moments", {}))
diff --git a/src/aiidalab_qe/app/configuration/advanced/model.py b/src/aiidalab_qe/app/configuration/advanced/model.py
index 9bc75a63c..a09253701 100644
--- a/src/aiidalab_qe/app/configuration/advanced/model.py
+++ b/src/aiidalab_qe/app/configuration/advanced/model.py
@@ -255,11 +255,7 @@ def _link_model(self, model: AdvancedCalculationSubSettingsModel):
self._on_any_change,
tl.All,
)
- for trait in model.dependencies:
- ipw.dlink(
- (self, trait),
- (model, trait),
- )
+ super()._link_model(model)
def _update_kpoints_mesh(self, _=None):
if not self.has_structure:
diff --git a/src/aiidalab_qe/app/configuration/advanced/pseudos/model.py b/src/aiidalab_qe/app/configuration/advanced/pseudos/model.py
index bf9231fd2..725b3c230 100644
--- a/src/aiidalab_qe/app/configuration/advanced/pseudos/model.py
+++ b/src/aiidalab_qe/app/configuration/advanced/pseudos/model.py
@@ -5,13 +5,13 @@
import traitlets as tl
from aiida_pseudo.common.units import U
-from aiida import orm
from aiida.common import exceptions
from aiida.plugins import GroupFactory
from aiida_quantumespresso.workflows.pw.base import PwBaseWorkChain
from aiidalab_qe.app.parameters import DEFAULT_PARAMETERS
from aiidalab_qe.common.mixins import HasInputStructure
from aiidalab_qe.setup.pseudos import PSEUDODOJO_VERSION, SSSP_VERSION, PseudoFamily
+from aiidalab_qe.utils import fetch_pseudo_family_by_label
from ..subsettings import AdvancedCalculationSubSettingsModel
@@ -132,7 +132,7 @@ def update_default_pseudos(self):
self.status_message = ""
try:
- pseudo_family = self._get_pseudo_family_from_database()
+ pseudo_family = fetch_pseudo_family_by_label(self.family)
pseudos = pseudo_family.get_pseudos(structure=self.input_structure)
except ValueError as exception:
self.status_message = f"""
@@ -156,7 +156,7 @@ def update_default_cutoffs(self):
self.status_message = ""
try:
- pseudo_family = self._get_pseudo_family_from_database()
+ pseudo_family = fetch_pseudo_family_by_label(self.family)
current_unit = pseudo_family.get_cutoffs_unit()
cutoff_dict = pseudo_family.get_cutoffs()
except exceptions.NotExistent:
@@ -301,21 +301,6 @@ def _get_default(self, trait):
)
return self._defaults.get(trait, self.traits()[trait].default_value)
- def _get_pseudo_family_from_database(self):
- """Get the pseudo family from the database."""
- return (
- orm.QueryBuilder()
- .append(
- (
- PseudoDojoFamily,
- SsspFamily,
- CutoffsPseudoPotentialFamily,
- ),
- filters={"label": self.family},
- )
- .one()[0]
- )
-
def _get_default_dictionary(self):
return deepcopy(self._defaults["dictionary"])
diff --git a/src/aiidalab_qe/app/configuration/basic/basic.py b/src/aiidalab_qe/app/configuration/basic/basic.py
index a92c88fd7..9d393d695 100644
--- a/src/aiidalab_qe/app/configuration/basic/basic.py
+++ b/src/aiidalab_qe/app/configuration/basic/basic.py
@@ -45,17 +45,18 @@ def render(self):
(self._model, "spin_type"),
(self.spin_type, "value"),
)
+ self.spin_type.observe(
+ self._on_spin_type_change,
+ "value",
+ )
+
self.magnetization_info = ipw.HTML(
value="""
Set the desired magnetic configuration in advanced settings
""",
- layout=ipw.Layout(visibility="hidden"),
- )
- self.spin_type.observe(
- self._on_spin_type_change,
- "value",
+ layout=ipw.Layout(display="none"),
)
# Spin-Orbit calculation
@@ -80,6 +81,29 @@ def render(self):
(self.protocol, "value"),
)
+ self.warning = ipw.HTML(
+ value="""
+
+
+ Warning: detected multiples atoms with different tags.
+ You may be interested in an antiferromagnetic system. Note that
+ default starting magnetic moments do not distinguish tagged
+ atoms and are set to the same value.
+
+
+ Please go to Advanced settings and override the default
+ values, specifying appropriate magnetic moments for each
+ species (e.g. with different signs for an antiferromagnetic
+ configuration).
+
+
+ """,
+ layout=ipw.Layout(display="none"),
+ )
+
self.children = [
InAppGuide(identifier="basic-settings"),
ipw.HTML("""
@@ -153,6 +177,7 @@ def render(self):
(at the price of longer/costlier calculations).
"""),
+ self.warning,
]
self.rendered = True
@@ -161,7 +186,10 @@ def _on_input_structure_change(self, _):
self.refresh(specific="structure")
def _on_spin_type_change(self, _):
- if self.spin_type.value == "none":
- self.magnetization_info.layout.visibility = "hidden"
+ if self._model.spin_type == "collinear":
+ self.magnetization_info.layout.display = "block"
+ if self._model.has_tags:
+ self.warning.layout.display = "flex"
else:
- self.magnetization_info.layout.visibility = "visible"
+ self.magnetization_info.layout.display = "none"
+ self.warning.layout.display = "none"
diff --git a/src/aiidalab_qe/app/configuration/basic/model.py b/src/aiidalab_qe/app/configuration/basic/model.py
index e9e826741..f526e547d 100644
--- a/src/aiidalab_qe/app/configuration/basic/model.py
+++ b/src/aiidalab_qe/app/configuration/basic/model.py
@@ -1,13 +1,16 @@
import traitlets as tl
-from aiida import orm
from aiidalab_qe.app.parameters import DEFAULT_PARAMETERS
+from aiidalab_qe.common.mixins import HasInputStructure
from aiidalab_qe.common.panel import ConfigurationSettingsModel
DEFAULT: dict = DEFAULT_PARAMETERS # type: ignore
-class BasicConfigurationSettingsModel(ConfigurationSettingsModel):
+class BasicConfigurationSettingsModel(
+ ConfigurationSettingsModel,
+ HasInputStructure,
+):
title = "Basic settings"
identifier = "workchain"
@@ -15,8 +18,6 @@ class BasicConfigurationSettingsModel(ConfigurationSettingsModel):
"input_structure",
]
- input_structure = tl.Union([tl.Instance(orm.StructureData)], allow_none=True)
-
protocol_options = tl.List(
trait=tl.Tuple(tl.Unicode(), tl.Unicode()),
default_value=[
diff --git a/src/aiidalab_qe/app/configuration/model.py b/src/aiidalab_qe/app/configuration/model.py
index 36a9c3e84..ceb19dc07 100644
--- a/src/aiidalab_qe/app/configuration/model.py
+++ b/src/aiidalab_qe/app/configuration/model.py
@@ -129,18 +129,7 @@ def _link_model(self, model: ConfigurationSettingsModel):
(self, "confirmed"),
(model, "confirmed"),
)
- for dependency in model.dependencies:
- dependency_parts = dependency.split(".")
- if len(dependency_parts) == 1: # from parent, e.g. input_structure
- target_model = self
- trait = dependency
- else: # from sibling, e.g. workchain.protocol
- sibling, trait = dependency_parts
- target_model = self.get_model(sibling)
- ipw.dlink(
- (target_model, trait),
- (model, trait),
- )
+ super()._link_model(model)
def _get_properties(self):
properties = []
diff --git a/src/aiidalab_qe/common/mixins.py b/src/aiidalab_qe/common/mixins.py
index 7dbe5df80..e6629a3e4 100644
--- a/src/aiidalab_qe/common/mixins.py
+++ b/src/aiidalab_qe/common/mixins.py
@@ -26,6 +26,13 @@ def has_structure(self):
def has_pbc(self):
return not self.has_structure or any(self.input_structure.pbc)
+ @property
+ def has_tags(self):
+ return any(
+ not kind_name.isalpha()
+ for kind_name in self.input_structure.get_kind_names()
+ )
+
class HasModels(t.Generic[T]):
def __init__(self):
@@ -51,7 +58,20 @@ def get_models(self) -> t.Iterable[tuple[str, T]]:
return self._models.items()
def _link_model(self, model: T):
- pass
+ if not hasattr(model, "dependencies"):
+ return
+ for dependency in model.dependencies:
+ dependency_parts = dependency.split(".")
+ if len(dependency_parts) == 1: # from parent
+ target_model = self
+ trait = dependency
+ else: # from sibling
+ sibling, trait = dependency_parts
+ target_model = self.get_model(sibling)
+ tl.dlink(
+ (target_model, trait),
+ (model, trait),
+ )
class HasProcess(tl.HasTraits):
diff --git a/src/aiidalab_qe/utils.py b/src/aiidalab_qe/utils.py
index 17e8013f4..398782ca3 100644
--- a/src/aiidalab_qe/utils.py
+++ b/src/aiidalab_qe/utils.py
@@ -1,3 +1,5 @@
+from aiida_pseudo.groups.family import PseudoPotentialFamily
+
from aiida import orm
@@ -30,3 +32,8 @@ def enable_pencil_decomposition(component):
"""Enable the pencil decomposition for the given component."""
component.settings = orm.Dict({"CMDLINE": ["-pd", ".true."]})
+
+
+def fetch_pseudo_family_by_label(label) -> PseudoPotentialFamily:
+ """Fetch the pseudo family by label."""
+ return orm.Group.collection.get(label=label) # type: ignore