OpenMD 3.2
Molecular Dynamics in the Open
Loading...
Searching...
No Matches
DipoleOrientation.cpp
1/*
2 * Copyright (c) 2004-present, The University of Notre Dame. All rights
3 * reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 *
11 * 2. Redistributions in binary form must reproduce the above copyright notice,
12 * this list of conditions and the following disclaimer in the documentation
13 * and/or other materials provided with the distribution.
14 *
15 * 3. Neither the name of the copyright holder nor the names of its
16 * contributors may be used to endorse or promote products derived from
17 * this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
23 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
30 *
31 * SUPPORT OPEN SCIENCE! If you use OpenMD or its source code in your
32 * research, please cite the following paper when you publish your work:
33 *
34 * [1] Drisko et al., J. Open Source Softw. 9, 7004 (2024).
35 *
36 * Good starting points for code and simulation methodology are:
37 *
38 * [2] Meineke, et al., J. Comp. Chem. 26, 252-271 (2005).
39 * [3] Fennell & Gezelter, J. Chem. Phys. 124, 234104 (2006).
40 * [4] Sun, Lin & Gezelter, J. Chem. Phys. 128, 234107 (2008).
41 * [5] Vardeman, Stocker & Gezelter, J. Chem. Theory Comput. 7, 834 (2011).
42 * [6] Kuang & Gezelter, Mol. Phys., 110, 691-701 (2012).
43 * [7] Lamichhane, Gezelter & Newman, J. Chem. Phys. 141, 134109 (2014).
44 * [8] Bhattarai, Newman & Gezelter, Phys. Rev. B 99, 094106 (2019).
45 * [9] Drisko & Gezelter, J. Chem. Theory Comput. 20, 4986-4997 (2024).
46 */
47
48#include "applications/staticProps/DipoleOrientation.hpp"
49
50#include <string>
51#include <vector>
52
53#include "applications/staticProps/SpatialStatistics.hpp"
54#include "brains/SimInfo.hpp"
56#include "math/Vector3.hpp"
58#include "utils/Accumulator.hpp"
59#include "utils/AccumulatorView.hpp"
60#include "utils/BaseAccumulator.hpp"
61#include "utils/StringUtils.hpp"
62
63using namespace OpenMD::Utils;
64
65namespace OpenMD {
66
67 DipoleOrientation::DipoleOrientation(
68 SimInfo* info, const std::string& filename, const std::string& sele,
69 const RealType dipoleX, const RealType dipoleY, const RealType dipoleZ,
70 int nzbins, int axis) :
71 SlabStatistics(info, filename, sele, nzbins, axis),
72 axis_(axis) {
73 switch (axis_) {
74 case 0:
75 axisLabel_ = "x";
76 refAxis_ = Vector3d(1, 0, 0);
77 break;
78 case 1:
79 axisLabel_ = "y";
80 refAxis_ = Vector3d(0, 1, 0);
81 break;
82 case 2:
83 default:
84 axisLabel_ = "z";
85 refAxis_ = Vector3d(0, 0, 1);
86 break;
87 }
88 setOutputName(getPrefix(filename) + ".Sz");
89
90 dipoleVector_ = Vector3d(dipoleX, dipoleY, dipoleZ);
91 dipoleVector_.normalize();
92
93 // Pre-load the OutputData
94 data_.resize(DipoleOrientation::ENDINDEX);
95
96 OutputData z;
97 z.units = "Angstroms";
98 z.title = axisLabel_;
99 z.dataHandling = DataHandling::Average;
100 for (unsigned int i = 0; i < nBins_; i++)
101 z.accumulator.push_back(
102 std::make_unique<AccumulatorView<RealAccumulator>>());
103 data_[Z] = std::move(z);
104
105 OutputData orderS;
106 orderS.units = "";
107 orderS.title = "Orientational Order parameter";
108 orderS.dataHandling = DataHandling::Average;
109 for (unsigned int i = 0; i < nBins_; i++)
110 orderS.accumulator.push_back(
111 std::make_unique<AccumulatorView<RealAccumulator>>());
112 data_[ORDERS] = std::move(orderS);
113
114 OutputData orderSCos;
115 orderSCos.units = "";
116 orderSCos.title = "Orientational Order parameter cosine Theta";
117 orderSCos.dataHandling = DataHandling::Average;
118 for (unsigned int i = 0; i < nBins_; i++)
119 orderSCos.accumulator.push_back(
120 std::make_unique<AccumulatorView<RealAccumulator>>());
121 data_[ORDERSCOS] = std::move(orderSCos);
122 }
123
124 void DipoleOrientation::processFrame(int) {
125 RealType z;
126
127 hmat_ = currentSnapshot_->getHmat();
128
129 for (unsigned int i = 0; i < nBins_; i++) {
130 z = (((RealType)i + 0.5) / (RealType)nBins_) * hmat_(axis_, axis_);
131 data_[Z].accumulator[i]->add(z);
132 }
133
134 volume_ = currentSnapshot_->getVolume();
135
136 StuntDouble* sd;
137 int i;
138
139 std::vector<RealType> binS(nBins_, 0.0);
140 std::vector<RealType> binSCos(nBins_, 0.0);
141
142 std::vector<int> count(nBins_, 0.0);
143
144 if (evaluator_.isDynamic()) {
145 seleMan_.setSelectionSet(evaluator_.evaluate());
146 }
147
148 // loop over the selected atoms:
150 Vector3d rotatedDipoleVector;
151 RealType ctheta;
152 RealType orderParameter;
153
154 for (sd = seleMan_.beginSelected(i); sd != NULL;
155 sd = seleMan_.nextSelected(i)) {
156 // figure out where that object is:
157 Vector3d pos = sd->getPos();
158
159 int bin = getBin(pos);
160
161 if (sd->isDirectional() || sd->isRigidBody()) {
162 rotMat = sd->getA();
163 rotatedDipoleVector = rotMat * dipoleVector_;
164 rotatedDipoleVector.normalize();
165 ctheta = dot(rotatedDipoleVector, refAxis_);
166
167 orderParameter = (3 * (ctheta * ctheta) - 1) / 2;
168
169 binS[bin] += orderParameter;
170 binSCos[bin] += ctheta;
171
172 count[bin] += 1;
173 }
174 }
175
176 for (unsigned int i = 0; i < nBins_; i++) {
177 count[i] != 0 ? data_[ORDERS].accumulator[i]->add(binS[i] / count[i]) :
178 data_[ORDERS].accumulator[i]->add(binS[i]);
179 count[i] != 0 ?
180 data_[ORDERSCOS].accumulator[i]->add(binSCos[i] / count[i]) :
181 data_[ORDERSCOS].accumulator[i]->add(binSCos[i]);
182 }
183 }
184} // namespace OpenMD
One of the heavy-weight classes of OpenMD, SimInfo maintains objects and variables relating to the cu...
Definition SimInfo.hpp:96
"Don't move, or you're dead! Stand up! Captain, we've got them!"
RotMat3x3d getA()
Returns the current rotation matrix of this stuntDouble.
Vector3d getPos()
Returns the current position of this stuntDouble.
bool isRigidBody()
Tests if this stuntDouble is a rigid body.
bool isDirectional()
Tests if this stuntDouble is a directional one.
void normalize()
Normalizes this vector in place.
Definition Vector.hpp:406
This basic Periodic Table class was originally taken from the data.cpp file in OpenBabel.
Real dot(const DynamicVector< Real > &v1, const DynamicVector< Real > &v2)
Returns the dot product of two DynamicVectors.
std::string getPrefix(const std::string &str)