Fix depsgraph to compute more accurate links for collision & force.
[blender.git] / source / blender / modifiers / intern / MOD_smoke.c
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software  Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * The Original Code is Copyright (C) 2005 by the Blender Foundation.
19  * All rights reserved.
20  *
21  * Contributor(s): Daniel Dunbar
22  *                 Ton Roosendaal,
23  *                 Ben Batt,
24  *                 Brecht Van Lommel,
25  *                 Campbell Barton
26  *
27  * ***** END GPL LICENSE BLOCK *****
28  *
29  */
30
31 /** \file blender/modifiers/intern/MOD_smoke.c
32  *  \ingroup modifiers
33  */
34
35
36 #include <stddef.h>
37
38 #include "MEM_guardedalloc.h"
39
40 #include "DNA_group_types.h"
41 #include "DNA_object_types.h"
42 #include "DNA_scene_types.h"
43 #include "DNA_smoke_types.h"
44 #include "DNA_object_force.h"
45
46 #include "BLI_utildefines.h"
47
48
49 #include "BKE_cdderivedmesh.h"
50 #include "BKE_library.h"
51 #include "BKE_library_query.h"
52 #include "BKE_main.h"
53 #include "BKE_modifier.h"
54 #include "BKE_smoke.h"
55
56 #include "depsgraph_private.h"
57 #include "DEG_depsgraph_build.h"
58
59 static void initData(ModifierData *md) 
60 {
61         SmokeModifierData *smd = (SmokeModifierData *) md;
62         
63         smd->domain = NULL;
64         smd->flow = NULL;
65         smd->coll = NULL;
66         smd->type = 0;
67         smd->time = -1;
68 }
69
70 static void copyData(ModifierData *md, ModifierData *target)
71 {
72         SmokeModifierData *smd  = (SmokeModifierData *)md;
73         SmokeModifierData *tsmd = (SmokeModifierData *)target;
74         
75         smokeModifier_copy(smd, tsmd);
76 }
77
78 static void freeData(ModifierData *md)
79 {
80         SmokeModifierData *smd = (SmokeModifierData *) md;
81         
82         smokeModifier_free(smd);
83 }
84
85 static CustomDataMask requiredDataMask(Object *UNUSED(ob), ModifierData *md)
86 {
87         SmokeModifierData *smd  = (SmokeModifierData *)md;
88         CustomDataMask dataMask = 0;
89
90         if (smd && (smd->type & MOD_SMOKE_TYPE_FLOW) && smd->flow) {
91                 if (smd->flow->source == MOD_SMOKE_FLOW_SOURCE_MESH) {
92                         /* vertex groups */
93                         if (smd->flow->vgroup_density)
94                                 dataMask |= CD_MASK_MDEFORMVERT;
95                         /* uv layer */
96                         if (smd->flow->texture_type == MOD_SMOKE_FLOW_TEXTURE_MAP_UV)
97                                 dataMask |= CD_MASK_MTFACE;
98                 }
99         }
100         return dataMask;
101 }
102
103 static DerivedMesh *applyModifier(ModifierData *md, Object *ob, 
104                                   DerivedMesh *dm,
105                                   ModifierApplyFlag flag)
106 {
107         SmokeModifierData *smd = (SmokeModifierData *) md;
108
109         if (flag & MOD_APPLY_ORCO)
110                 return dm;
111
112         return smokeModifier_do(smd, md->scene, ob, dm);
113 }
114
115 static bool dependsOnTime(ModifierData *UNUSED(md))
116 {
117         return true;
118 }
119
120 static bool is_flow_cb(Object *UNUSED(ob), ModifierData *md)
121 {
122         SmokeModifierData *smd = (SmokeModifierData *) md;
123         return (smd->type & MOD_SMOKE_TYPE_FLOW) && smd->flow;
124 }
125
126 static bool is_coll_cb(Object *UNUSED(ob), ModifierData *md)
127 {
128         SmokeModifierData *smd = (SmokeModifierData *) md;
129         return (smd->type & MOD_SMOKE_TYPE_COLL) && smd->coll;
130 }
131
132 static void updateDepgraph(ModifierData *md, DagForest *forest,
133                            struct Main *UNUSED(bmain),
134                            struct Scene *scene, struct Object *ob,
135                            DagNode *obNode)
136 {
137         SmokeModifierData *smd = (SmokeModifierData *) md;
138
139         if (smd && (smd->type & MOD_SMOKE_TYPE_DOMAIN) && smd->domain) {
140                 /* Actual code uses get_collisionobjects */
141                 dag_add_collision_relations(forest, scene, ob, obNode, smd->domain->fluid_group, ob->lay|scene->lay, eModifierType_Smoke, is_flow_cb, true, "Smoke Flow");
142                 dag_add_collision_relations(forest, scene, ob, obNode, smd->domain->coll_group, ob->lay|scene->lay, eModifierType_Smoke, is_coll_cb, true, "Smoke Coll");
143
144                 dag_add_forcefield_relations(forest, scene, ob, obNode, smd->domain->effector_weights, true, PFIELD_SMOKEFLOW, "Smoke Force Field");
145         }
146 }
147
148 static void updateDepsgraph(ModifierData *md,
149                             struct Main *UNUSED(bmain),
150                             struct Scene *scene,
151                             Object *ob,
152                             struct DepsNodeHandle *node)
153 {
154         SmokeModifierData *smd = (SmokeModifierData *)md;
155
156         if (smd && (smd->type & MOD_SMOKE_TYPE_DOMAIN) && smd->domain) {
157                 /* Actual code uses get_collisionobjects */
158                 DEG_add_collision_relations(node, scene, ob, smd->domain->fluid_group, ob->lay|scene->lay, eModifierType_Smoke, is_flow_cb, true, "Smoke Flow");
159                 DEG_add_collision_relations(node, scene, ob, smd->domain->coll_group, ob->lay|scene->lay, eModifierType_Smoke, is_coll_cb, true, "Smoke Coll");
160
161                 DEG_add_forcefield_relations(node, scene, ob, smd->domain->effector_weights, true, PFIELD_SMOKEFLOW, "Smoke Force Field");
162         }
163 }
164
165 static void foreachIDLink(ModifierData *md, Object *ob,
166                           IDWalkFunc walk, void *userData)
167 {
168         SmokeModifierData *smd = (SmokeModifierData *) md;
169
170         if (smd->type == MOD_SMOKE_TYPE_DOMAIN && smd->domain) {
171                 walk(userData, ob, (ID **)&smd->domain->coll_group, IDWALK_NOP);
172                 walk(userData, ob, (ID **)&smd->domain->fluid_group, IDWALK_NOP);
173                 walk(userData, ob, (ID **)&smd->domain->eff_group, IDWALK_NOP);
174
175                 if (smd->domain->effector_weights) {
176                         walk(userData, ob, (ID **)&smd->domain->effector_weights->group, IDWALK_NOP);
177                 }
178         }
179
180         if (smd->type == MOD_SMOKE_TYPE_FLOW && smd->flow) {
181                 walk(userData, ob, (ID **)&smd->flow->noise_texture, IDWALK_USER);
182         }
183 }
184
185 ModifierTypeInfo modifierType_Smoke = {
186         /* name */              "Smoke",
187         /* structName */        "SmokeModifierData",
188         /* structSize */        sizeof(SmokeModifierData),
189         /* type */              eModifierTypeType_Constructive,
190         /* flags */             eModifierTypeFlag_AcceptsMesh |
191                                 eModifierTypeFlag_UsesPointCache |
192                                 eModifierTypeFlag_Single,
193
194         /* copyData */          copyData,
195         /* deformVerts */       NULL,
196         /* deformMatrices */    NULL,
197         /* deformVertsEM */     NULL,
198         /* deformMatricesEM */  NULL,
199         /* applyModifier */     applyModifier,
200         /* applyModifierEM */   NULL,
201         /* initData */          initData,
202         /* requiredDataMask */  requiredDataMask,
203         /* freeData */          freeData,
204         /* isDisabled */        NULL,
205         /* updateDepgraph */    updateDepgraph,
206         /* updateDepsgraph */   updateDepsgraph,
207         /* dependsOnTime */     dependsOnTime,
208         /* dependsOnNormals */  NULL,
209         /* foreachObjectLink */ NULL,
210         /* foreachIDLink */     foreachIDLink,
211         /* foreachTexLink */    NULL
212 };