| 
									
										
										
										
											2008-01-01 15:53:38 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup blenloader | 
					
						
							| 
									
										
										
										
											2011-02-27 20:35:41 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * FILE FORMAT | 
					
						
							|  |  |  |  * =========== | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * IFF-style structure (but not IFF compatible!) | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * Start file: | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  * <pre> | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * `BLENDER_V100`  `12` bytes  (version 1.00 is just an example). | 
					
						
							|  |  |  |  *                 `V` = big endian, `v` = little endian. | 
					
						
							|  |  |  |  *                 `_` = 4 byte pointer, `-` = 8 byte pointer. | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  * </pre> | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-06-12 09:04:10 +10:00
										 |  |  |  * data-blocks: (also see struct #BHead). | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  * <pre> | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * `bh.code`       `char[4]` see `BLO_blend_defs.h` for a list of known types. | 
					
						
							|  |  |  |  * `bh.len`        `int32` length data after #BHead in bytes. | 
					
						
							|  |  |  |  * `bh.old`        `void *` old pointer (the address at the time of writing the file). | 
					
						
							|  |  |  |  * `bh.SDNAnr`     `int32` struct index of structs stored in #DNA1 data. | 
					
						
							|  |  |  |  * `bh.nr`         `int32` in case of array: number of structs. | 
					
						
							|  |  |  |  * data | 
					
						
							|  |  |  |  * ... | 
					
						
							|  |  |  |  * ... | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  * </pre> | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Almost all data in Blender are structures. Each struct saved | 
					
						
							|  |  |  |  * gets a BHead header.  With BHead the struct can be linked again | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * and compared with #StructDNA. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * WRITE | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  * ===== | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Preferred writing order: (not really a must, but why would you do it random?) | 
					
						
							| 
									
										
										
										
											2019-10-15 12:02:40 +11:00
										 |  |  |  * Any case: direct data is ALWAYS after the lib block. | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * (Local file data) | 
					
						
							|  |  |  |  * - for each LibBlock | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  *   - write LibBlock | 
					
						
							|  |  |  |  *   - write associated direct data | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  * (External file data) | 
					
						
							|  |  |  |  * - per library | 
					
						
							| 
									
										
										
										
											2016-06-27 11:21:03 +10:00
										 |  |  |  *   - write library block | 
					
						
							|  |  |  |  *   - per LibBlock | 
					
						
							|  |  |  |  *     - write the ID of LibBlock | 
					
						
							|  |  |  |  * - write #TEST (#RenderInfo struct. 128x128 blend file preview is optional). | 
					
						
							|  |  |  |  * - write #GLOB (#FileGlobal struct) (some global vars). | 
					
						
							|  |  |  |  * - write #DNA1 (#SDNA struct) | 
					
						
							| 
									
										
										
										
											2017-02-15 14:10:42 +11:00
										 |  |  |  * - write #USER (#UserDef struct) if filename is ``~/.config/blender/X.XX/config/startup.blend``. | 
					
						
							| 
									
										
										
										
											2012-10-04 13:26:15 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2003-04-26 18:01:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-06 13:20:05 +00:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | #include <limits.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-06 13:20:05 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-06 13:20:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-02 11:43:54 +02:00
										 |  |  | #ifdef WIN32
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #  include "BLI_winstuff.h"
 | 
					
						
							| 
									
										
										
										
											2012-04-15 07:54:07 +00:00
										 |  |  | #  include "winsock2.h"
 | 
					
						
							|  |  |  | #  include <io.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #  include <zlib.h> /* odd include order-issue */
 | 
					
						
							| 
									
										
										
										
											2014-04-03 09:20:04 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #  include <unistd.h> /* FreeBSD, for write() and close(). */
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-22 13:35:32 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-21 11:15:01 +00:00
										 |  |  | /* allow writefile to use deprecated functionality (for forward compatibility code) */ | 
					
						
							|  |  |  | #define DNA_DEPRECATED_ALLOW
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 10:41:45 +00:00
										 |  |  | #include "DNA_anim_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_armature_types.h"
 | 
					
						
							| 
									
										
										
											
												
Brush Datablock:
- Added a new Brush datablock, only used by image paint, but intended
  to be used in texture paint, vertex paint, weight paint and sculpt
  mode also.
- Being a datablock, these brushes can be saved, appended and linked.
  They have a fake user by default, to make sure they are saved even if
  not selected.
Image Painting:
- Replaced the img module with C code in imagepaint.c
- Airbrush is no longer a separate tool, but rather an option that can
  be used for soften, smear and clone also.
- Blend modes mix, add, subtract, multiply, darken and lighten have been
  added, code taken directly from vertex paint.
Note to project files maintainers:
- The img module was removed from SCons and Makefiles, and this should
  be done in other build systems also. I'll wait to remove the module
  from cvs, to not break compilation.
											
										 
											2006-07-26 22:29:23 +00:00
										 |  |  | #include "DNA_brush_types.h"
 | 
					
						
							| 
									
										
										
											
												Basic Alembic support
All in all, this patch adds an Alembic importer, an Alembic exporter,
and a new CacheFile data block which, for now, wraps around an Alembic
archive. This data block is made available through a new modifier ("Mesh
Sequence Cache") as well as a new constraint ("Transform Cache") to
somewhat properly support respectively geometric and transformation data
streaming from alembic caches.
A more in-depth documentation is to be found on the wiki, as well as a
 guide to compile alembic: https://wiki.blender.org/index.php/
User:Kevindietrich/AlembicBasicIo.
Many thanks to everyone involved in this little project, and huge shout
out to "cgstrive" for the thorough testings with Maya, 3ds Max, Houdini
and Realflow as well as @fjuhec, @jensverwiebe and @jasperge for the
custom builds and compile fixes.
Reviewers: sergey, campbellbarton, mont29
Reviewed By: sergey, campbellbarton, mont29
Differential Revision: https://developer.blender.org/D2060
											
										 
											2016-08-06 06:20:37 +02:00
										 |  |  | #include "DNA_cachefile_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_camera_types.h"
 | 
					
						
							| 
									
										
										
										
											2008-01-29 21:01:12 +00:00
										 |  |  | #include "DNA_cloth_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-29 15:32:50 +02:00
										 |  |  | #include "DNA_collection_types.h"
 | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | #include "DNA_constraint_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_curveprofile_types.h"
 | 
					
						
							| 
									
										
										
										
											2011-05-24 07:08:58 +00:00
										 |  |  | #include "DNA_dynamicpaint_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_fileglobal_types.h"
 | 
					
						
							|  |  |  | #include "DNA_fluid_types.h"
 | 
					
						
							| 
									
										
										
										
											2008-10-31 23:50:02 +00:00
										 |  |  | #include "DNA_genfile.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | #include "DNA_gpencil_modifier_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_gpencil_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | #include "DNA_hair_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_key_types.h"
 | 
					
						
							|  |  |  | #include "DNA_lattice_types.h"
 | 
					
						
							| 
									
										
											  
											
												Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
  Option to add objects based on their names
* Collection Manager operators
  The existing buttons  are placeholders
* Collection Manager drawing
  The editor main region is empty
* Collection Override
* Per-Collection engine settings
  This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
  We are replacing Base by a new extended Base while keeping it backward
  compatible with some legacy settings (i.e., lay, flag_legacy).
  Renamed all Base to BaseLegacy to make it clear the areas of code that
  still need to be converted
  Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
  - read, write, add/copy/remove objects, copy scene, collection
  link/unlinking, context)
* New Editor: Collection Manager
  Based on patch by Julian Eisel
  This is extracted from the layer-manager branch. With the following changes:
    - Renamed references of layer manager to collections manager
    - I doesn't include the editors/space_collections/ draw and util files
    - The drawing code itself will be implemented separately by Julian
* Base / Object:
  A little note about them. Original Blender code would try to keep them
  in sync through the code, juggling flags back and forth. This will now
  be handled by Depsgraph, keeping Object and Bases more separated
  throughout the non-rendering code.
  Scene.base is being cleared in doversion, and the old viewport drawing
  code was poorly converted to use the new bases while the new viewport
  code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
											
										 
											2017-02-07 10:18:38 +01:00
										 |  |  | #include "DNA_layer_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_light_types.h"
 | 
					
						
							|  |  |  | #include "DNA_lightprobe_types.h"
 | 
					
						
							| 
									
										
										
										
											2013-03-23 03:00:37 +00:00
										 |  |  | #include "DNA_linestyle_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_mask_types.h"
 | 
					
						
							|  |  |  | #include "DNA_material_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							| 
									
										
										
										
											2004-03-20 22:55:42 +00:00
										 |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_meta_types.h"
 | 
					
						
							|  |  |  | #include "DNA_movieclip_types.h"
 | 
					
						
							| 
									
										
											  
											
												Christmas coding work!
********* Node editor work:
- To enable Nodes for Materials, you have to set the "Use Nodes"
  button, in the new Material buttons "Nodes" Panel or in header
  of the Node editor. Doing this will disable Material-Layers.
- Nodes now execute materials ("shaders"), but still only using the
  previewrender code.
- Nodes have (optional) previews for rendered images.
- Node headers allow to hide buttons and/or preview image
- Nodes can be dragged larger/smaller (right-bottom corner)
- Nodes can be hidden (minimized) with hotkey H
- CTRL+click on an Input Socket gives a popup with default values.
- Changing Material/Texture or Mix node will adjust Node title.
- Click-drag outside of a Node changes cursor to "Knife' and allows to
  draw a rect where to cut Links.
- Added new node types RGBtoBW, Texture, In/Output, ColorRamp
- Material Nodes have options to ouput diffuse or specular, or to use
  a negative normal. The input socket 'Normal' will force the material
  to use that normal, otherwise it uses the normal from the Material
  that has the node tree.
- When drawing a link between two not-matching sockets, Blender inserts
  a converting node (now only for value/rgb combos)
- When drawing a link to an input socket that's already in use, the
  old link will either disappear or flip to another unused socket.
- A click on a Material Node will activate it, and show all its settings
  in the Material Buttons. Active Material Nodes draw the material icon
  in red.
- A click on any node will show its options in the Node Panel in the
  Material buttons.
- Multiple Output Nodes can be used, to sample contents of a tree, but
  only one Output is the real one, which is indicated in a different
  color and red material icon.
- Added ThemeColors for node types
- ALT+C will convert existing Material-Layers to Node... this currently
  only adds the material/mix nodes and connects them. Dunno if this is
  worth a lot of coding work to make perfect?
- Press C to call another "Solve order", which will show all possible
  cyclic conflicts (if there are).
- Technical: nodes now use "Type" structs which define the
  structure of nodes and in/output sockets. The Type structs store all
  fixed info, callbacks, and allow to reconstruct saved Nodes to match
  what is required by Blender.
- Defining (new) nodes now is as simple as filling in a fixed
  Type struct, plus code some callbacks. A doc will be made!
- Node preview images are by default float
********* Icon drawing:
- Cleanup of how old icons were implemented in new system, making
  them 16x16 too, correctly centered *and* scaled.
- Made drawing Icons use float coordinates
- Moved BIF_calcpreview_image() into interface_icons.c, renamed it
  icon_from_image(). Removed a lot of unneeded Imbuf magic here! :)
- Skipped scaling and imbuf copying when icons are OK size
********* Preview render:
- Huge cleanup of code....
- renaming BIF_xxx calls that only were used internally
- BIF_previewrender() now accepts an argument for rendering method,
  so it supports icons, buttonwindow previewrender and node editor
- Only a single BIF_preview_changed() call now exists, supporting all
  signals as needed for buttos and node editor
********* More stuff:
- glutil.c, glaDrawPixelsSafe() and glaDrawPixelsTex() now accept format
  argument for GL_FLOAT rects
- Made the ColorBand become a built-in button for interface.c
  Was a load of cleanup work in buttons_shading.c...
- removed a load of unneeded glBlendFunc() calls
- Fixed bug in calculating text length for buttons (ancient!)
											
										 
											2005-12-28 15:42:51 +00:00
										 |  |  | #include "DNA_node_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-07 11:14:08 +11:00
										 |  |  | #include "DNA_object_force_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_packedFile_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | #include "DNA_particle_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | #include "DNA_pointcloud_types.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-23 05:56:22 +00:00
										 |  |  | #include "DNA_rigidbody_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_screen_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_sdna_types.h"
 | 
					
						
							|  |  |  | #include "DNA_sequence_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_shader_fx_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-20 10:37:38 +02:00
										 |  |  | #include "DNA_simulation_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_sound_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_space_types.h"
 | 
					
						
							| 
									
										
										
										
											2011-08-01 11:44:20 +00:00
										 |  |  | #include "DNA_speaker_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_text_types.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-02 13:28:13 +00:00
										 |  |  | #include "DNA_vfont_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_view3d_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | #include "DNA_volume_types.h"
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:53:37 +00:00
										 |  |  | #include "DNA_windowmanager_types.h"
 | 
					
						
							| 
									
										
											  
											
												Main Workspace Integration
This commit does the main integration of workspaces, which is a design we agreed on during the 2.8 UI workshop (see https://wiki.blender.org/index.php/Dev:2.8/UI/Workshop_Writeup)
Workspaces should generally be stable, I'm not aware of any remaining bugs (or I've forgotten them :) ). If you find any, let me know!
(Exception: mode switching button might get out of sync with actual mode in some cases, would consider that a limitation/ToDo. Needs to be resolved at some point.)
== Main Changes/Features
* Introduces the new Workspaces as data-blocks.
* Allow storing a number of custom workspaces as part of the user configuration. Needs further work to allow adding and deleting individual workspaces.
* Bundle a default workspace configuration with Blender (current screen-layouts converted to workspaces).
* Pressing button to add a workspace spawns a menu to select between "Duplicate Current" and the workspaces from the user configuration. If no workspaces are stored in the user configuration, the default workspaces are listed instead.
* Store screen-layouts (`bScreen`) per workspace.
* Store an active screen-layout per workspace. Changing the workspace will enable this layout.
* Store active mode in workspace. Changing the workspace will also enter the mode of the new workspace. (Note that we still store the active mode in the object, moving this completely to workspaces is a separate project.)
* Store an active render layer per workspace.
* Moved mode switch from 3D View header to Info Editor header.
* Store active scene in window (not directly workspace related, but overlaps quite a bit).
* Removed 'Use Global Scene' User Preference option.
* Compatibility with old files - a new workspace is created for every screen-layout of old files. Old Blender versions should be able to read files saved with workspace support as well.
* Default .blend only contains one workspace ("General").
* Support appending workspaces.
Opening files without UI and commandline rendering should work fine.
Note that the UI is temporary! We plan to introduce a new global topbar
that contains the workspace options and tabs for switching workspaces.
== Technical Notes
* Workspaces are data-blocks.
* Adding and removing `bScreen`s should be done through `ED_workspace_layout` API now.
* A workspace can be active in multiple windows at the same time.
* The mode menu (which is now in the Info Editor header) doesn't display "Grease Pencil Edit" mode anymore since its availability depends on the active editor. Will be fixed by making Grease Pencil an own object type (as planned).
* The button to change the active workspace object mode may get out of sync with the mode of the active object. Will either be resolved by moving mode out of object data, or we'll disable workspace modes again (there's a `#define USE_WORKSPACE_MODE` for that).
* Screen-layouts (`bScreen`) are IDs and thus stored in a main list-base. Had to add a wrapper `WorkSpaceLayout` so we can store them in a list-base within workspaces, too. On the long run we could completely replace `bScreen` by workspace structs.
* `WorkSpace` types use some special compiler trickery to allow marking structs and struct members as private. BKE_workspace API should be used for accessing those.
* Added scene operators `SCENE_OT_`. Was previously done through screen operators.
== BPY API Changes
* Removed `Screen.scene`, added `Window.scene`
* Removed `UserPreferencesView.use_global_scene`
* Added `Context.workspace`, `Window.workspace` and `BlendData.workspaces`
* Added `bpy.types.WorkSpace` containing `screens`, `object_mode` and `render_layer`
* Added Screen.layout_name for the layout name that'll be displayed in the UI (may differ from internal name)
== What's left?
* There are a few open design questions (T50521). We should find the needed answers and implement them.
* Allow adding and removing individual workspaces from workspace configuration (needs UI design).
* Get the override system ready and support overrides per workspace.
* Support custom UI setups as part of workspaces (hidden panels, hidden buttons, customizable toolbars, etc).
* Allow enabling add-ons per workspace.
* Support custom workspace keymaps.
* Remove special exception for workspaces in linking code (so they're always appended, never linked). Depends on a few things, so best to solve later.
* Get the topbar done.
* Workspaces need a proper icon, current one is just a placeholder :)
Reviewed By: campbellbarton, mont29
Tags: #user_interface, #bf_blender_2.8
Maniphest Tasks: T50521
Differential Revision: https://developer.blender.org/D2451
											
										 
											2017-06-01 19:56:58 +02:00
										 |  |  | #include "DNA_workspace_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_world_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:31:38 +00:00
										 |  |  | #include "BLI_bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BLI_blenlib.h"
 | 
					
						
							| 
									
										
										
										
											2013-08-03 11:35:09 +00:00
										 |  |  | #include "BLI_mempool.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "MEM_guardedalloc.h"  // MEM_freeN
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_action.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:42:41 +10:00
										 |  |  | #include "BKE_blender_version.h"
 | 
					
						
							| 
									
										
										
										
											2013-08-03 11:35:09 +00:00
										 |  |  | #include "BKE_bpath.h"
 | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | #include "BKE_collection.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_constraint.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-07 18:00:24 +01:00
										 |  |  | #include "BKE_curve.h"
 | 
					
						
							|  |  |  | #include "BKE_fcurve.h"
 | 
					
						
							| 
									
										
										
										
											2020-05-01 12:43:12 +02:00
										 |  |  | #include "BKE_fcurve_driver.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #include "BKE_global.h"  // for G
 | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | #include "BKE_gpencil_modifier.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 19:37:00 +01:00
										 |  |  | #include "BKE_idtype.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-04 13:00:46 +02:00
										 |  |  | #include "BKE_layer.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  | #include "BKE_lib_id.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 12:58:59 +01:00
										 |  |  | #include "BKE_lib_override.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-19 16:36:15 +00:00
										 |  |  | #include "BKE_main.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-07 18:00:24 +01:00
										 |  |  | #include "BKE_modifier.h"
 | 
					
						
							| 
									
										
											  
											
												Christmas coding work!
********* Node editor work:
- To enable Nodes for Materials, you have to set the "Use Nodes"
  button, in the new Material buttons "Nodes" Panel or in header
  of the Node editor. Doing this will disable Material-Layers.
- Nodes now execute materials ("shaders"), but still only using the
  previewrender code.
- Nodes have (optional) previews for rendered images.
- Node headers allow to hide buttons and/or preview image
- Nodes can be dragged larger/smaller (right-bottom corner)
- Nodes can be hidden (minimized) with hotkey H
- CTRL+click on an Input Socket gives a popup with default values.
- Changing Material/Texture or Mix node will adjust Node title.
- Click-drag outside of a Node changes cursor to "Knife' and allows to
  draw a rect where to cut Links.
- Added new node types RGBtoBW, Texture, In/Output, ColorRamp
- Material Nodes have options to ouput diffuse or specular, or to use
  a negative normal. The input socket 'Normal' will force the material
  to use that normal, otherwise it uses the normal from the Material
  that has the node tree.
- When drawing a link between two not-matching sockets, Blender inserts
  a converting node (now only for value/rgb combos)
- When drawing a link to an input socket that's already in use, the
  old link will either disappear or flip to another unused socket.
- A click on a Material Node will activate it, and show all its settings
  in the Material Buttons. Active Material Nodes draw the material icon
  in red.
- A click on any node will show its options in the Node Panel in the
  Material buttons.
- Multiple Output Nodes can be used, to sample contents of a tree, but
  only one Output is the real one, which is indicated in a different
  color and red material icon.
- Added ThemeColors for node types
- ALT+C will convert existing Material-Layers to Node... this currently
  only adds the material/mix nodes and connects them. Dunno if this is
  worth a lot of coding work to make perfect?
- Press C to call another "Solve order", which will show all possible
  cyclic conflicts (if there are).
- Technical: nodes now use "Type" structs which define the
  structure of nodes and in/output sockets. The Type structs store all
  fixed info, callbacks, and allow to reconstruct saved Nodes to match
  what is required by Blender.
- Defining (new) nodes now is as simple as filling in a fixed
  Type struct, plus code some callbacks. A doc will be made!
- Node preview images are by default float
********* Icon drawing:
- Cleanup of how old icons were implemented in new system, making
  them 16x16 too, correctly centered *and* scaled.
- Made drawing Icons use float coordinates
- Moved BIF_calcpreview_image() into interface_icons.c, renamed it
  icon_from_image(). Removed a lot of unneeded Imbuf magic here! :)
- Skipped scaling and imbuf copying when icons are OK size
********* Preview render:
- Huge cleanup of code....
- renaming BIF_xxx calls that only were used internally
- BIF_previewrender() now accepts an argument for rendering method,
  so it supports icons, buttonwindow previewrender and node editor
- Only a single BIF_preview_changed() call now exists, supporting all
  signals as needed for buttos and node editor
********* More stuff:
- glutil.c, glaDrawPixelsSafe() and glaDrawPixelsTex() now accept format
  argument for GL_FLOAT rects
- Made the ColorBand become a built-in button for interface.c
  Was a load of cleanup work in buttons_shading.c...
- removed a load of unneeded glBlendFunc() calls
- Fixed bug in calculating text length for buttons (ancient!)
											
										 
											2005-12-28 15:42:51 +00:00
										 |  |  | #include "BKE_node.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:58:37 +02:00
										 |  |  | #include "BKE_object.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-07 18:00:24 +01:00
										 |  |  | #include "BKE_pointcache.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-19 00:50:21 +00:00
										 |  |  | #include "BKE_report.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-13 14:56:45 +00:00
										 |  |  | #include "BKE_sequencer.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | #include "BKE_shader_fx.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:24 +00:00
										 |  |  | #include "BKE_subsurf.h"
 | 
					
						
							| 
									
										
											  
											
												Main Workspace Integration
This commit does the main integration of workspaces, which is a design we agreed on during the 2.8 UI workshop (see https://wiki.blender.org/index.php/Dev:2.8/UI/Workshop_Writeup)
Workspaces should generally be stable, I'm not aware of any remaining bugs (or I've forgotten them :) ). If you find any, let me know!
(Exception: mode switching button might get out of sync with actual mode in some cases, would consider that a limitation/ToDo. Needs to be resolved at some point.)
== Main Changes/Features
* Introduces the new Workspaces as data-blocks.
* Allow storing a number of custom workspaces as part of the user configuration. Needs further work to allow adding and deleting individual workspaces.
* Bundle a default workspace configuration with Blender (current screen-layouts converted to workspaces).
* Pressing button to add a workspace spawns a menu to select between "Duplicate Current" and the workspaces from the user configuration. If no workspaces are stored in the user configuration, the default workspaces are listed instead.
* Store screen-layouts (`bScreen`) per workspace.
* Store an active screen-layout per workspace. Changing the workspace will enable this layout.
* Store active mode in workspace. Changing the workspace will also enter the mode of the new workspace. (Note that we still store the active mode in the object, moving this completely to workspaces is a separate project.)
* Store an active render layer per workspace.
* Moved mode switch from 3D View header to Info Editor header.
* Store active scene in window (not directly workspace related, but overlaps quite a bit).
* Removed 'Use Global Scene' User Preference option.
* Compatibility with old files - a new workspace is created for every screen-layout of old files. Old Blender versions should be able to read files saved with workspace support as well.
* Default .blend only contains one workspace ("General").
* Support appending workspaces.
Opening files without UI and commandline rendering should work fine.
Note that the UI is temporary! We plan to introduce a new global topbar
that contains the workspace options and tabs for switching workspaces.
== Technical Notes
* Workspaces are data-blocks.
* Adding and removing `bScreen`s should be done through `ED_workspace_layout` API now.
* A workspace can be active in multiple windows at the same time.
* The mode menu (which is now in the Info Editor header) doesn't display "Grease Pencil Edit" mode anymore since its availability depends on the active editor. Will be fixed by making Grease Pencil an own object type (as planned).
* The button to change the active workspace object mode may get out of sync with the mode of the active object. Will either be resolved by moving mode out of object data, or we'll disable workspace modes again (there's a `#define USE_WORKSPACE_MODE` for that).
* Screen-layouts (`bScreen`) are IDs and thus stored in a main list-base. Had to add a wrapper `WorkSpaceLayout` so we can store them in a list-base within workspaces, too. On the long run we could completely replace `bScreen` by workspace structs.
* `WorkSpace` types use some special compiler trickery to allow marking structs and struct members as private. BKE_workspace API should be used for accessing those.
* Added scene operators `SCENE_OT_`. Was previously done through screen operators.
== BPY API Changes
* Removed `Screen.scene`, added `Window.scene`
* Removed `UserPreferencesView.use_global_scene`
* Added `Context.workspace`, `Window.workspace` and `BlendData.workspaces`
* Added `bpy.types.WorkSpace` containing `screens`, `object_mode` and `render_layer`
* Added Screen.layout_name for the layout name that'll be displayed in the UI (may differ from internal name)
== What's left?
* There are a few open design questions (T50521). We should find the needed answers and implement them.
* Allow adding and removing individual workspaces from workspace configuration (needs UI design).
* Get the override system ready and support overrides per workspace.
* Support custom UI setups as part of workspaces (hidden panels, hidden buttons, customizable toolbars, etc).
* Allow enabling add-ons per workspace.
* Support custom workspace keymaps.
* Remove special exception for workspaces in linking code (so they're always appended, never linked). Depends on a few things, so best to solve later.
* Get the topbar done.
* Workspaces need a proper icon, current one is just a placeholder :)
Reviewed By: campbellbarton, mont29
Tags: #user_interface, #bf_blender_2.8
Maniphest Tasks: T50521
Differential Revision: https://developer.blender.org/D2451
											
										 
											2017-06-01 19:56:58 +02:00
										 |  |  | #include "BKE_workspace.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-25 17:31:32 +01:00
										 |  |  | #include "BLO_blend_defs.h"
 | 
					
						
							|  |  |  | #include "BLO_blend_validate.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 11:44:36 +02:00
										 |  |  | #include "BLO_read_write.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BLO_readfile.h"
 | 
					
						
							| 
									
										
										
										
											2004-09-05 13:43:51 +00:00
										 |  |  | #include "BLO_undofile.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-25 17:31:32 +01:00
										 |  |  | #include "BLO_writefile.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "readfile.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | /* for SDNA_TYPE_FROM_STRUCT() macro */ | 
					
						
							|  |  |  | #include "dna_type_offsets.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Patch provided by Shaul Kedem: Compressed files are back!
He even made a nice doc in wiki:
http://wiki.blender.org/bin/view.pl/Blenderdev/Blendgz
Usage: set the option "Compress File" in the main "File" pulldown menu.
This setting is a user-def, meaning it is not changed on reading files.
If you want it default, save it with CTRL+U.
The longest debate went over the file naming convention. Shaul started
with .blend.gz files, which gave issues in Blender because of the code
hanging out everywhere that detects blender files, and that appends the
.blend extension if needed.
Daniel Dunbar proposed to just save it as .blend, and not bother users
with such details. This is indeed the most elegant solution, with as
only drawback that old Blender executables cannot read it.
This drawback isn't very relevant at the moment, since we're heading
towards a release that isn't upward compatible anyway... the recode
going on on Meshes, Modfiers, Armatures, Poses, Actions, NLA already
have upward compatibility issues.
We might check - during the next month(s) - on a builtin system to
warn users in the future when we change things that make a file risky
to read in an older release.
											
										 
											2005-07-27 19:46:06 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2004-09-05 13:43:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-24 22:54:51 +11:00
										 |  |  | /* Make preferences read-only. */ | 
					
						
							|  |  |  | #define U (*((const UserDef *)&U))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-05 15:13:41 +00:00
										 |  |  | /* ********* my write, buffered writing with minimum size chunks ************ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-08 14:32:29 +10:00
										 |  |  | /* Use optimal allocation since blocks of this size are kept in memory for undo. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define MYWRITE_BUFFER_SIZE (MEM_SIZE_OPTIMAL(1 << 17)) /* 128kb */
 | 
					
						
							|  |  |  | #define MYWRITE_MAX_CHUNK (MEM_SIZE_OPTIMAL(1 << 15))   /* ~32kb */
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** Use if we want to store how many bytes have been written to the file. */ | 
					
						
							|  |  |  | // #define USE_WRITE_DATA_LEN
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Internal Write Wrapper's (Abstracts Compression)
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef enum { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WW_WRAP_NONE = 1, | 
					
						
							|  |  |  |   WW_WRAP_ZLIB, | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } eWriteWrapType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct WriteWrap WriteWrap; | 
					
						
							|  |  |  | struct WriteWrap { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* callbacks */ | 
					
						
							|  |  |  |   bool (*open)(WriteWrap *ww, const char *filepath); | 
					
						
							|  |  |  |   bool (*close)(WriteWrap *ww); | 
					
						
							|  |  |  |   size_t (*write)(WriteWrap *ww, const char *data, size_t data_len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Buffer output (we only want when output isn't already buffered). */ | 
					
						
							|  |  |  |   bool use_buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* internal */ | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     int file_handle; | 
					
						
							|  |  |  |     gzFile gz_handle; | 
					
						
							|  |  |  |   } _user_data; | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* none */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define FILE_HANDLE(ww) (ww)->_user_data.file_handle
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | static bool ww_open_none(WriteWrap *ww, const char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int file; | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   file = BLI_open(filepath, O_BINARY + O_WRONLY + O_CREAT + O_TRUNC, 0666); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (file != -1) { | 
					
						
							|  |  |  |     FILE_HANDLE(ww) = file; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | static bool ww_close_none(WriteWrap *ww) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (close(FILE_HANDLE(ww)) != -1); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | static size_t ww_write_none(WriteWrap *ww, const char *buf, size_t buf_len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return write(FILE_HANDLE(ww), buf, buf_len); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | #undef FILE_HANDLE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* zlib */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define FILE_HANDLE(ww) (ww)->_user_data.gz_handle
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | static bool ww_open_zlib(WriteWrap *ww, const char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   gzFile file; | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   file = BLI_gzopen(filepath, "wb1"); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (file != Z_NULL) { | 
					
						
							|  |  |  |     FILE_HANDLE(ww) = file; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | static bool ww_close_zlib(WriteWrap *ww) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (gzclose(FILE_HANDLE(ww)) == Z_OK); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | static size_t ww_write_zlib(WriteWrap *ww, const char *buf, size_t buf_len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return gzwrite(FILE_HANDLE(ww), buf, buf_len); | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | #undef FILE_HANDLE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* --- end compression types --- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ww_handle_init(eWriteWrapType ww_type, WriteWrap *r_ww) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   memset(r_ww, 0, sizeof(*r_ww)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (ww_type) { | 
					
						
							|  |  |  |     case WW_WRAP_ZLIB: { | 
					
						
							|  |  |  |       r_ww->open = ww_open_zlib; | 
					
						
							|  |  |  |       r_ww->close = ww_close_zlib; | 
					
						
							|  |  |  |       r_ww->write = ww_write_zlib; | 
					
						
							|  |  |  |       r_ww->use_buf = false; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: { | 
					
						
							|  |  |  |       r_ww->open = ww_open_none; | 
					
						
							|  |  |  |       r_ww->close = ww_close_none; | 
					
						
							|  |  |  |       r_ww->write = ww_write_none; | 
					
						
							|  |  |  |       r_ww->use_buf = true; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Write Data Type & Functions
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const struct SDNA *sdna; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /** Use for file and memory writing (fixed size of #MYWRITE_BUFFER_SIZE). */ | 
					
						
							|  |  |  |   uchar *buf; | 
					
						
							|  |  |  |   /** Number of bytes used in #WriteData.buf (flushed when exceeded). */ | 
					
						
							|  |  |  |   int buf_used_len; | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef USE_WRITE_DATA_LEN
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /** Total number of bytes written. */ | 
					
						
							|  |  |  |   size_t write_len; | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /** Set on unlikely case of an error (ignores further file writing).  */ | 
					
						
							|  |  |  |   bool error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** #MemFile writing (used for undo). */ | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |   MemFileWriteData mem; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /** When true, write to #WriteData.current, could also call 'is_undo'. */ | 
					
						
							|  |  |  |   bool use_memfile; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Wrap writing, so we can use zlib or | 
					
						
							|  |  |  |    * other compression types later, see: G_FILE_COMPRESS | 
					
						
							|  |  |  |    * Will be NULL for UNDO. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   WriteWrap *ww; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } WriteData; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 11:44:36 +02:00
										 |  |  | typedef struct BlendWriter { | 
					
						
							|  |  |  |   WriteData *wd; | 
					
						
							|  |  |  | } BlendWriter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-04 21:48:36 +10:00
										 |  |  | static WriteData *writedata_new(WriteWrap *ww) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WriteData *wd = MEM_callocN(sizeof(*wd), "writedata"); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wd->sdna = DNA_sdna_current_get(); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wd->ww = ww; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((ww == NULL) || (ww->use_buf)) { | 
					
						
							|  |  |  |     wd->buf = MEM_mallocN(MYWRITE_BUFFER_SIZE, "wd->buf"); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return wd; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-17 19:51:40 +00:00
										 |  |  | static void writedata_do_write(WriteData *wd, const void *mem, int memlen) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((wd == NULL) || wd->error || (mem == NULL) || memlen < 1) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (UNLIKELY(wd->error)) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* memory based save */ | 
					
						
							|  |  |  |   if (wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |     BLO_memfile_chunk_add(&wd->mem, mem, memlen); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     if (wd->ww->write(wd->ww, mem, memlen) != memlen) { | 
					
						
							|  |  |  |       wd->error = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | static void writedata_free(WriteData *wd) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wd->buf) { | 
					
						
							|  |  |  |     MEM_freeN(wd->buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   MEM_freeN(wd); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Local Writing API 'mywrite'
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 16:02:45 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Flush helps the de-duplicating memory for undo-save by logically segmenting data, | 
					
						
							|  |  |  |  * so differences in one part of memory won't cause unrelated data to be duplicated. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void mywrite_flush(WriteData *wd) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wd->buf_used_len) { | 
					
						
							|  |  |  |     writedata_do_write(wd, wd->buf, wd->buf_used_len); | 
					
						
							|  |  |  |     wd->buf_used_len = 0; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-07-07 16:02:45 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Low level WRITE(2) wrapper that buffers data | 
					
						
							| 
									
										
										
										
											2018-12-12 12:50:58 +11:00
										 |  |  |  * \param adr: Pointer to new chunk of data | 
					
						
							|  |  |  |  * \param len: Length of new chunk of data | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-04-29 17:11:40 +00:00
										 |  |  | static void mywrite(WriteData *wd, const void *adr, int len) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (UNLIKELY(wd->error)) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (UNLIKELY(adr == NULL)) { | 
					
						
							|  |  |  |     BLI_assert(0); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | #ifdef USE_WRITE_DATA_LEN
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wd->write_len += len; | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wd->buf == NULL) { | 
					
						
							|  |  |  |     writedata_do_write(wd, adr, len); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* if we have a single big chunk, write existing data in
 | 
					
						
							|  |  |  |      * buffer and write out big chunk in smaller pieces */ | 
					
						
							|  |  |  |     if (len > MYWRITE_MAX_CHUNK) { | 
					
						
							|  |  |  |       if (wd->buf_used_len) { | 
					
						
							|  |  |  |         writedata_do_write(wd, wd->buf, wd->buf_used_len); | 
					
						
							|  |  |  |         wd->buf_used_len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       do { | 
					
						
							|  |  |  |         int writelen = MIN2(len, MYWRITE_MAX_CHUNK); | 
					
						
							|  |  |  |         writedata_do_write(wd, adr, writelen); | 
					
						
							|  |  |  |         adr = (const char *)adr + writelen; | 
					
						
							|  |  |  |         len -= writelen; | 
					
						
							|  |  |  |       } while (len > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if data would overflow buffer, write out the buffer */ | 
					
						
							|  |  |  |     if (len + wd->buf_used_len > MYWRITE_BUFFER_SIZE - 1) { | 
					
						
							|  |  |  |       writedata_do_write(wd, wd->buf, wd->buf_used_len); | 
					
						
							|  |  |  |       wd->buf_used_len = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* append data at end of buffer */ | 
					
						
							|  |  |  |     memcpy(&wd->buf[wd->buf_used_len], adr, len); | 
					
						
							|  |  |  |     wd->buf_used_len += len; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * BeGiN initializer for mywrite | 
					
						
							| 
									
										
										
										
											2015-05-20 12:54:45 +10:00
										 |  |  |  * \param ww: File write wrapper. | 
					
						
							| 
									
										
										
										
											2018-12-12 12:50:58 +11:00
										 |  |  |  * \param compare: Previous memory file (can be NULL). | 
					
						
							|  |  |  |  * \param current: The current memory file (can be NULL). | 
					
						
							| 
									
										
										
										
											2012-03-02 16:05:54 +00:00
										 |  |  |  * \warning Talks to other functions with global parameters | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | static WriteData *mywrite_begin(WriteWrap *ww, MemFile *compare, MemFile *current) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WriteData *wd = writedata_new(ww); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (current != NULL) { | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |     BLO_memfile_write_init(&wd->mem, current, compare); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     wd->use_memfile = true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2006-10-27 18:24:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return wd; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * END the mywrite wrapper | 
					
						
							| 
									
										
										
										
											2012-03-02 16:05:54 +00:00
										 |  |  |  * \return 1 if write failed | 
					
						
							|  |  |  |  * \return unknown global variable otherwise | 
					
						
							|  |  |  |  * \warning Talks to other functions with global parameters | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | static bool mywrite_end(WriteData *wd) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wd->buf_used_len) { | 
					
						
							|  |  |  |     writedata_do_write(wd, wd->buf, wd->buf_used_len); | 
					
						
							|  |  |  |     wd->buf_used_len = 0; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |   if (wd->use_memfile) { | 
					
						
							|  |  |  |     BLO_memfile_write_finalize(&wd->mem); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const bool err = wd->error; | 
					
						
							|  |  |  |   writedata_free(wd); | 
					
						
							| 
									
										
										
										
											2007-02-14 11:00:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return err; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Start writing of data related to a single ID. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Only does something when storing an undo step. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void mywrite_id_begin(WriteData *wd, ID *id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (wd->use_memfile) { | 
					
						
							|  |  |  |     wd->mem.current_id_session_uuid = id->session_uuid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* If current next memchunk does not match the ID we are about to write, try to find the
 | 
					
						
							|  |  |  |      * correct memchunk in the mapping using ID's session_uuid. */ | 
					
						
							|  |  |  |     if (wd->mem.id_session_uuid_mapping != NULL && | 
					
						
							|  |  |  |         (wd->mem.reference_current_chunk == NULL || | 
					
						
							|  |  |  |          wd->mem.reference_current_chunk->id_session_uuid != id->session_uuid)) { | 
					
						
							|  |  |  |       void *ref = BLI_ghash_lookup(wd->mem.id_session_uuid_mapping, | 
					
						
							|  |  |  |                                    POINTER_FROM_UINT(id->session_uuid)); | 
					
						
							|  |  |  |       if (ref != NULL) { | 
					
						
							|  |  |  |         wd->mem.reference_current_chunk = ref; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       /* Else, no existing memchunk found, i.e. this is supposed to be a new ID. */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Otherwise, we try with the current memchunk in any case, whether it is matching current
 | 
					
						
							|  |  |  |      * ID's session_uuid or not. */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Start writing of data related to a single ID. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Only does something when storing an undo step. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-04 01:18:14 +10:00
										 |  |  | static void mywrite_id_end(WriteData *wd, ID *UNUSED(id)) | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   if (wd->use_memfile) { | 
					
						
							|  |  |  |     /* Very important to do it after every ID write now, otherwise we cannot know whether a
 | 
					
						
							|  |  |  |      * specific ID changed or not. */ | 
					
						
							|  |  |  |     mywrite_flush(wd); | 
					
						
							|  |  |  |     wd->mem.current_id_session_uuid = MAIN_ID_SESSION_UUID_UNSET; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Generic DNA File Writing
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | static void writestruct_at_address_nr( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WriteData *wd, int filecode, const int struct_nr, int nr, const void *adr, const void *data) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BHead bh; | 
					
						
							|  |  |  |   const short *sp; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(struct_nr > 0 && struct_nr < SDNA_TYPE_MAX); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (adr == NULL || data == NULL || nr == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* init BHead */ | 
					
						
							|  |  |  |   bh.code = filecode; | 
					
						
							|  |  |  |   bh.old = adr; | 
					
						
							|  |  |  |   bh.nr = nr; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bh.SDNAnr = struct_nr; | 
					
						
							|  |  |  |   sp = wd->sdna->structs[bh.SDNAnr]; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bh.len = nr * wd->sdna->types_size[sp[0]]; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (bh.len == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mywrite(wd, &bh, sizeof(BHead)); | 
					
						
							|  |  |  |   mywrite(wd, data, bh.len); | 
					
						
							| 
									
										
										
										
											2013-04-12 15:33:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | static void writestruct_at_address_id( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WriteData *wd, int filecode, const char *structname, int nr, const void *adr, const void *data) | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (adr == NULL || data == NULL || nr == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int SDNAnr = DNA_struct_find_nr(wd->sdna, structname); | 
					
						
							|  |  |  |   if (UNLIKELY(SDNAnr == -1)) { | 
					
						
							|  |  |  |     printf("error: can't find SDNA code <%s>\n", structname); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct_at_address_nr(wd, filecode, SDNAnr, nr, adr, data); | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void writestruct_nr( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WriteData *wd, int filecode, const int struct_nr, int nr, const void *adr) | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct_at_address_nr(wd, filecode, struct_nr, nr, adr, adr); | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void writestruct_id( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WriteData *wd, int filecode, const char *structname, int nr, const void *adr) | 
					
						
							| 
									
										
										
										
											2013-04-12 15:33:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct_at_address_id(wd, filecode, structname, nr, adr, adr); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 14:41:19 +10:00
										 |  |  | /* do not use for structs */ | 
					
						
							|  |  |  | static void writedata(WriteData *wd, int filecode, int len, const void *adr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BHead bh; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (adr == NULL || len == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* align to 4 (writes uninitialized bytes in some cases) */ | 
					
						
							|  |  |  |   len = (len + 3) & ~3; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* init BHead */ | 
					
						
							|  |  |  |   bh.code = filecode; | 
					
						
							|  |  |  |   bh.old = adr; | 
					
						
							|  |  |  |   bh.nr = 1; | 
					
						
							|  |  |  |   bh.SDNAnr = 0; | 
					
						
							|  |  |  |   bh.len = len; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mywrite(wd, &bh, sizeof(BHead)); | 
					
						
							|  |  |  |   mywrite(wd, adr, len); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-07 16:57:53 +00:00
										 |  |  | /* use this to force writing of lists in same order as reading (using link_list) */ | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | static void writelist_nr(WriteData *wd, int filecode, const int struct_nr, const ListBase *lb) | 
					
						
							| 
									
										
										
										
											2013-03-07 16:57:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const Link *link = lb->first; | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   while (link) { | 
					
						
							|  |  |  |     writestruct_nr(wd, filecode, struct_nr, 1, link); | 
					
						
							|  |  |  |     link = link->next; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-03-07 16:57:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | static void writelist_id(WriteData *wd, int filecode, const char *structname, const ListBase *lb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const Link *link = lb->first; | 
					
						
							|  |  |  |   if (link) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int struct_nr = DNA_struct_find_nr(wd->sdna, structname); | 
					
						
							|  |  |  |     if (struct_nr == -1) { | 
					
						
							|  |  |  |       printf("error: can't find SDNA code <%s>\n", structname); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (link) { | 
					
						
							|  |  |  |       writestruct_nr(wd, filecode, struct_nr, 1, link); | 
					
						
							|  |  |  |       link = link->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define writestruct_at_address(wd, filecode, struct_id, nr, adr, data) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct_at_address_nr(wd, filecode, SDNA_TYPE_FROM_STRUCT(struct_id), nr, adr, data) | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define writestruct(wd, filecode, struct_id, nr, adr) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct_nr(wd, filecode, SDNA_TYPE_FROM_STRUCT(struct_id), nr, adr) | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define writelist(wd, filecode, struct_id, lb) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writelist_nr(wd, filecode, SDNA_TYPE_FROM_STRUCT(struct_id), lb) | 
					
						
							| 
									
										
										
										
											2016-06-28 20:05:42 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Typed DNA File Writing
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * These functions are used by blender's .blend system for file saving/loading. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | void IDP_WriteProperty_OnlyData(const IDProperty *prop, BlendWriter *writer); | 
					
						
							|  |  |  | void IDP_WriteProperty(const IDProperty *prop, WriteData *wd); | 
					
						
							|  |  |  | void IDP_WriteProperty_new_api(const IDProperty *prop, BlendWriter *writer); | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | static void IDP_WriteArray(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /*REMEMBER to set totalen to len in the linking code!!*/ | 
					
						
							|  |  |  |   if (prop->data.pointer) { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |     BLO_write_raw(writer, MEM_allocN_len(prop->data.pointer), prop->data.pointer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (prop->subtype == IDP_GROUP) { | 
					
						
							|  |  |  |       IDProperty **array = prop->data.pointer; | 
					
						
							|  |  |  |       int a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (a = 0; a < prop->len; a++) { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |         IDP_WriteProperty_new_api(array[a], writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | static void IDP_WriteIDPArray(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2008-12-31 13:16:37 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /*REMEMBER to set totalen to len in the linking code!!*/ | 
					
						
							|  |  |  |   if (prop->data.pointer) { | 
					
						
							|  |  |  |     const IDProperty *array = prop->data.pointer; | 
					
						
							|  |  |  |     int a; | 
					
						
							| 
									
										
										
										
											2008-12-31 13:16:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |     BLO_write_struct_array(writer, IDProperty, prop->len, array); | 
					
						
							| 
									
										
										
										
											2008-12-31 13:16:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     for (a = 0; a < prop->len; a++) { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |       IDP_WriteProperty_OnlyData(&array[a], writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-31 13:16:37 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | static void IDP_WriteString(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /*REMEMBER to set totalen to len in the linking code!!*/ | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |   BLO_write_raw(writer, prop->len, prop->data.pointer); | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | static void IDP_WriteGroup(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   IDProperty *loop; | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (loop = prop->data.group.first; loop; loop = loop->next) { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |     IDP_WriteProperty_new_api(loop, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Functions to read/write ID Properties */ | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | void IDP_WriteProperty_OnlyData(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   switch (prop->type) { | 
					
						
							|  |  |  |     case IDP_GROUP: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |       IDP_WriteGroup(prop, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case IDP_STRING: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |       IDP_WriteString(prop, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case IDP_ARRAY: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |       IDP_WriteArray(prop, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case IDP_IDPARRAY: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |       IDP_WriteIDPArray(prop, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  | void IDP_WriteProperty_new_api(const IDProperty *prop, BlendWriter *writer) | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:21:07 +02:00
										 |  |  |   BLO_write_struct(writer, IDProperty, prop); | 
					
						
							|  |  |  |   IDP_WriteProperty_OnlyData(prop, writer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void IDP_WriteProperty(const IDProperty *prop, WriteData *wd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BlendWriter writer = {wd}; | 
					
						
							|  |  |  |   IDP_WriteProperty_new_api(prop, &writer); | 
					
						
							| 
									
										
										
										
											2006-11-17 04:46:48 +00:00
										 |  |  | } | 
					
						
							| 
									
										
											  
											
												Orange:
- New UI element: the "Curve Button".
For mapping ranges (like 0 - 1) to another range, the curve button can be
used for proportional falloff, bone influences, painting density, etc.
Most evident use is of course to map RGB color with curves.
To be able to use it, you have to allocate a CurveMapping struct and pass
this on to the button. The CurveMapping API is in the new C file
blenkernel/intern/colortools.c
It's as simple as calling:
   curvemap= curvemapping_add(3, 0, 0, 1, 1)
Which will create 3 curves, and sets a default 0-1 range. The current code
only supports up to 4 curves maximum per mapping struct.
The CurveMap button in Blender than handles allmost all  editing.
Evaluating a single channel:
   float newvalue= curvemapping_evaluateF(curvemap, 0, oldval);
Where the second argument is the channel index, here 0-1-2 are possible.
Or mapping a vector:
   curvemapping_evaluate3F(curvemap, newvec, oldvec);
Optimized versions for byte or short mapping is possible too, not done yet.
In butspace.c I've added a template wrapper for buttons around the curve, to
reveil settings or show tools; check this screenie:
http://www.blender.org/bf/curves.jpg
- Buttons R, G, B: select channel
- icons + and -: zoom in, out
- icon 'wrench': menu with tools, like clear curve, set handle type
- icon 'clipping': menu with clip values, and to dis/enable clipping
- icon 'x': delete selection
In the curve button itself, only LMB clicks are handled (like all UI elements
in Blender).
- click on point: select
- shift+click on point: swap select
- click on point + drag: select point (if not selected) and move it
- click outside point + drag: translate view
- CTRL+click: add new point
- hold SHIFT while dragging to snap to grid
  (Yes I know... either one of these can be Blender compliant, not both!)
- if you drag a point exactly on top of another, it merges them
Other fixes:
- Icons now draw using "Safe RasterPos", so they align with pixel boundary.
  the old code made ints from the raster pos coordinate, which doesn't work
  well for zoom in/out situations
- bug in Node editing: buttons could not get freed, causing in memory error
  prints at end of a Blender session. That one was a very simple, but nasty
  error causing me all evening last night to find!
  (Hint; check diff of editnode.c, where uiDoButtons is called)
Last note: this adds 3 new files in our tree, I did scons, but not MSVC!
											
										 
											2006-01-08 11:41:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-17 12:09:06 +01:00
										 |  |  | static void write_iddata(WriteData *wd, ID *id) | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ID_WM's id->properties are considered runtime only, and never written in .blend file. */ | 
					
						
							|  |  |  |   if (id->properties && !ELEM(GS(id->name), ID_WM)) { | 
					
						
							|  |  |  |     IDP_WriteProperty(id->properties, wd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:16:04 +02:00
										 |  |  |   if (id->override_library) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, IDOverrideLibrary, 1, id->override_library); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:16:04 +02:00
										 |  |  |     writelist(wd, DATA, IDOverrideLibraryProperty, &id->override_library->properties); | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (IDOverrideLibraryProperty *, op, &id->override_library->properties) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writedata(wd, DATA, strlen(op->rna_path) + 1, op->rna_path); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:16:04 +02:00
										 |  |  |       writelist(wd, DATA, IDOverrideLibraryPropertyOperation, &op->operations); | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |       LISTBASE_FOREACH (IDOverrideLibraryPropertyOperation *, opop, &op->operations) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         if (opop->subitem_reference_name) { | 
					
						
							|  |  |  |           writedata( | 
					
						
							|  |  |  |               wd, DATA, strlen(opop->subitem_reference_name) + 1, opop->subitem_reference_name); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (opop->subitem_local_name) { | 
					
						
							|  |  |  |           writedata(wd, DATA, strlen(opop->subitem_local_name) + 1, opop->subitem_local_name); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-22 08:44:26 +10:00
										 |  |  | static void write_previews(WriteData *wd, const PreviewImage *prv_orig) | 
					
						
							| 
									
										
										
										
											2015-08-10 15:41:28 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Note we write previews also for undo steps. It takes up some memory,
 | 
					
						
							|  |  |  |    * but not doing so would causes all previews to be re-rendered after | 
					
						
							|  |  |  |    * undo which is too expensive. */ | 
					
						
							|  |  |  |   if (prv_orig) { | 
					
						
							|  |  |  |     PreviewImage prv = *prv_orig; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* don't write out large previews if not requested */ | 
					
						
							|  |  |  |     if (!(U.flag & USER_SAVE_PREVIEWS)) { | 
					
						
							|  |  |  |       prv.w[1] = 0; | 
					
						
							|  |  |  |       prv.h[1] = 0; | 
					
						
							|  |  |  |       prv.rect[1] = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct_at_address(wd, DATA, PreviewImage, 1, prv_orig, &prv); | 
					
						
							|  |  |  |     if (prv.rect[0]) { | 
					
						
							|  |  |  |       writedata(wd, DATA, prv.w[0] * prv.h[0] * sizeof(uint), prv.rect[0]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (prv.rect[1]) { | 
					
						
							|  |  |  |       writedata(wd, DATA, prv.w[1] * prv.h[1] * sizeof(uint), prv.rect[1]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-10 15:41:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | static void write_fmodifiers(WriteData *wd, ListBase *fmodifiers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   FModifier *fcm; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write all modifiers first (for faster reloading) */ | 
					
						
							|  |  |  |   writelist(wd, DATA, FModifier, fmodifiers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Modifiers */ | 
					
						
							|  |  |  |   for (fcm = fmodifiers->first; fcm; fcm = fcm->next) { | 
					
						
							|  |  |  |     const FModifierTypeInfo *fmi = fmodifier_get_typeinfo(fcm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write the specific data */ | 
					
						
							|  |  |  |     if (fmi && fcm->data) { | 
					
						
							|  |  |  |       /* firstly, just write the plain fmi->data struct */ | 
					
						
							|  |  |  |       writestruct_id(wd, DATA, fmi->structName, 1, fcm->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* do any modifier specific stuff */ | 
					
						
							|  |  |  |       switch (fcm->type) { | 
					
						
							|  |  |  |         case FMODIFIER_TYPE_GENERATOR: { | 
					
						
							|  |  |  |           FMod_Generator *data = fcm->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* write coefficients array */ | 
					
						
							|  |  |  |           if (data->coefficients) { | 
					
						
							|  |  |  |             writedata(wd, DATA, sizeof(float) * (data->arraysize), data->coefficients); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case FMODIFIER_TYPE_ENVELOPE: { | 
					
						
							|  |  |  |           FMod_Envelope *data = fcm->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* write envelope data */ | 
					
						
							|  |  |  |           if (data->data) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, FCM_EnvelopeData, data->totvert, data->data); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case FMODIFIER_TYPE_PYTHON: { | 
					
						
							|  |  |  |           FMod_Python *data = fcm->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* Write ID Properties -- and copy this comment EXACTLY for easy finding
 | 
					
						
							|  |  |  |            * of library blocks that implement this.*/ | 
					
						
							|  |  |  |           IDP_WriteProperty(data->prop, wd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fcurves(WriteData *wd, ListBase *fcurves) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   FCurve *fcu; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   writelist(wd, DATA, FCurve, fcurves); | 
					
						
							|  |  |  |   for (fcu = fcurves->first; fcu; fcu = fcu->next) { | 
					
						
							|  |  |  |     /* curve data */ | 
					
						
							|  |  |  |     if (fcu->bezt) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, BezTriple, fcu->totvert, fcu->bezt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (fcu->fpt) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, FPoint, fcu->totvert, fcu->fpt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fcu->rna_path) { | 
					
						
							|  |  |  |       writedata(wd, DATA, strlen(fcu->rna_path) + 1, fcu->rna_path); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* driver data */ | 
					
						
							|  |  |  |     if (fcu->driver) { | 
					
						
							|  |  |  |       ChannelDriver *driver = fcu->driver; | 
					
						
							|  |  |  |       DriverVar *dvar; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, ChannelDriver, 1, driver); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* variables */ | 
					
						
							|  |  |  |       writelist(wd, DATA, DriverVar, &driver->variables); | 
					
						
							|  |  |  |       for (dvar = driver->variables.first; dvar; dvar = dvar->next) { | 
					
						
							|  |  |  |         DRIVER_TARGETS_USED_LOOPER_BEGIN (dvar) { | 
					
						
							|  |  |  |           if (dtar->rna_path) { | 
					
						
							|  |  |  |             writedata(wd, DATA, strlen(dtar->rna_path) + 1, dtar->rna_path); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         DRIVER_TARGETS_LOOPER_END; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* write F-Modifiers */ | 
					
						
							|  |  |  |     write_fmodifiers(wd, &fcu->modifiers); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_action(WriteData *wd, bAction *act, const void *id_address) | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (act->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_AC, bAction, 1, id_address, act); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &act->id); | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_fcurves(wd, &act->curves); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (bActionGroup *, grp, &act->groups) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, bActionGroup, 1, grp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (TimeMarker *, marker, &act->markers) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, TimeMarker, 1, marker); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_keyingsets(WriteData *wd, ListBase *list) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   KeyingSet *ks; | 
					
						
							|  |  |  |   KS_Path *ksp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (ks = list->first; ks; ks = ks->next) { | 
					
						
							|  |  |  |     /* KeyingSet */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, KeyingSet, 1, ks); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Paths */ | 
					
						
							|  |  |  |     for (ksp = ks->paths.first; ksp; ksp = ksp->next) { | 
					
						
							|  |  |  |       /* Path */ | 
					
						
							|  |  |  |       writestruct(wd, DATA, KS_Path, 1, ksp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (ksp->rna_path) { | 
					
						
							|  |  |  |         writedata(wd, DATA, strlen(ksp->rna_path) + 1, ksp->rna_path); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_nlastrips(WriteData *wd, ListBase *strips) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   NlaStrip *strip; | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writelist(wd, DATA, NlaStrip, strips); | 
					
						
							|  |  |  |   for (strip = strips->first; strip; strip = strip->next) { | 
					
						
							|  |  |  |     /* write the strip's F-Curves and modifiers */ | 
					
						
							|  |  |  |     write_fcurves(wd, &strip->fcurves); | 
					
						
							|  |  |  |     write_fmodifiers(wd, &strip->modifiers); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write the strip's children */ | 
					
						
							|  |  |  |     write_nlastrips(wd, &strip->strips); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_nladata(WriteData *wd, ListBase *nlabase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   NlaTrack *nlt; | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write all the tracks */ | 
					
						
							|  |  |  |   for (nlt = nlabase->first; nlt; nlt = nlt->next) { | 
					
						
							|  |  |  |     /* write the track first */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, NlaTrack, 1, nlt); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write the track's strips */ | 
					
						
							|  |  |  |     write_nlastrips(wd, &nlt->strips); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_animdata(WriteData *wd, AnimData *adt) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   AnimOverride *aor; | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* firstly, just write the AnimData block */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, AnimData, 1, adt); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write drivers */ | 
					
						
							|  |  |  |   write_fcurves(wd, &adt->drivers); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write overrides */ | 
					
						
							|  |  |  |   // FIXME: are these needed?
 | 
					
						
							|  |  |  |   for (aor = adt->overrides.first; aor; aor = aor->next) { | 
					
						
							|  |  |  |     /* overrides consist of base data + rna_path */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, AnimOverride, 1, aor); | 
					
						
							|  |  |  |     writedata(wd, DATA, strlen(aor->rna_path) + 1, aor->rna_path); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   // TODO write the remaps (if they are needed)
 | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write NLA data */ | 
					
						
							|  |  |  |   write_nladata(wd, &adt->nla_tracks); | 
					
						
							| 
									
										
										
										
											2010-12-13 08:08:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | static void write_curvemapping_curves(WriteData *wd, CurveMapping *cumap) | 
					
						
							| 
									
										
											  
											
												Orange:
- New UI element: the "Curve Button".
For mapping ranges (like 0 - 1) to another range, the curve button can be
used for proportional falloff, bone influences, painting density, etc.
Most evident use is of course to map RGB color with curves.
To be able to use it, you have to allocate a CurveMapping struct and pass
this on to the button. The CurveMapping API is in the new C file
blenkernel/intern/colortools.c
It's as simple as calling:
   curvemap= curvemapping_add(3, 0, 0, 1, 1)
Which will create 3 curves, and sets a default 0-1 range. The current code
only supports up to 4 curves maximum per mapping struct.
The CurveMap button in Blender than handles allmost all  editing.
Evaluating a single channel:
   float newvalue= curvemapping_evaluateF(curvemap, 0, oldval);
Where the second argument is the channel index, here 0-1-2 are possible.
Or mapping a vector:
   curvemapping_evaluate3F(curvemap, newvec, oldvec);
Optimized versions for byte or short mapping is possible too, not done yet.
In butspace.c I've added a template wrapper for buttons around the curve, to
reveil settings or show tools; check this screenie:
http://www.blender.org/bf/curves.jpg
- Buttons R, G, B: select channel
- icons + and -: zoom in, out
- icon 'wrench': menu with tools, like clear curve, set handle type
- icon 'clipping': menu with clip values, and to dis/enable clipping
- icon 'x': delete selection
In the curve button itself, only LMB clicks are handled (like all UI elements
in Blender).
- click on point: select
- shift+click on point: swap select
- click on point + drag: select point (if not selected) and move it
- click outside point + drag: translate view
- CTRL+click: add new point
- hold SHIFT while dragging to snap to grid
  (Yes I know... either one of these can be Blender compliant, not both!)
- if you drag a point exactly on top of another, it merges them
Other fixes:
- Icons now draw using "Safe RasterPos", so they align with pixel boundary.
  the old code made ints from the raster pos coordinate, which doesn't work
  well for zoom in/out situations
- bug in Node editing: buttons could not get freed, causing in memory error
  prints at end of a Blender session. That one was a very simple, but nasty
  error causing me all evening last night to find!
  (Hint; check diff of editnode.c, where uiDoButtons is called)
Last note: this adds 3 new files in our tree, I did scons, but not MSVC!
											
										 
											2006-01-08 11:41:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (int a = 0; a < CM_TOT; a++) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, CurveMapPoint, cumap->cm[a].totpoint, cumap->cm[a].curve); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Orange:
- New UI element: the "Curve Button".
For mapping ranges (like 0 - 1) to another range, the curve button can be
used for proportional falloff, bone influences, painting density, etc.
Most evident use is of course to map RGB color with curves.
To be able to use it, you have to allocate a CurveMapping struct and pass
this on to the button. The CurveMapping API is in the new C file
blenkernel/intern/colortools.c
It's as simple as calling:
   curvemap= curvemapping_add(3, 0, 0, 1, 1)
Which will create 3 curves, and sets a default 0-1 range. The current code
only supports up to 4 curves maximum per mapping struct.
The CurveMap button in Blender than handles allmost all  editing.
Evaluating a single channel:
   float newvalue= curvemapping_evaluateF(curvemap, 0, oldval);
Where the second argument is the channel index, here 0-1-2 are possible.
Or mapping a vector:
   curvemapping_evaluate3F(curvemap, newvec, oldvec);
Optimized versions for byte or short mapping is possible too, not done yet.
In butspace.c I've added a template wrapper for buttons around the curve, to
reveil settings or show tools; check this screenie:
http://www.blender.org/bf/curves.jpg
- Buttons R, G, B: select channel
- icons + and -: zoom in, out
- icon 'wrench': menu with tools, like clear curve, set handle type
- icon 'clipping': menu with clip values, and to dis/enable clipping
- icon 'x': delete selection
In the curve button itself, only LMB clicks are handled (like all UI elements
in Blender).
- click on point: select
- shift+click on point: swap select
- click on point + drag: select point (if not selected) and move it
- click outside point + drag: translate view
- CTRL+click: add new point
- hold SHIFT while dragging to snap to grid
  (Yes I know... either one of these can be Blender compliant, not both!)
- if you drag a point exactly on top of another, it merges them
Other fixes:
- Icons now draw using "Safe RasterPos", so they align with pixel boundary.
  the old code made ints from the raster pos coordinate, which doesn't work
  well for zoom in/out situations
- bug in Node editing: buttons could not get freed, causing in memory error
  prints at end of a Blender session. That one was a very simple, but nasty
  error causing me all evening last night to find!
  (Hint; check diff of editnode.c, where uiDoButtons is called)
Last note: this adds 3 new files in our tree, I did scons, but not MSVC!
											
										 
											2006-01-08 11:41:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | static void write_curvemapping(WriteData *wd, CurveMapping *cumap) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, DATA, CurveMapping, 1, cumap); | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_curvemapping_curves(wd, cumap); | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 16:12:32 -05:00
										 |  |  | static void write_CurveProfile(WriteData *wd, CurveProfile *profile) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writestruct(wd, DATA, CurveProfile, 1, profile); | 
					
						
							|  |  |  |   writestruct(wd, DATA, CurveProfilePoint, profile->path_len, profile->path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 13:29:58 +01:00
										 |  |  | static void write_node_socket_default_value(WriteData *wd, bNodeSocket *sock) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (sock->default_value == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch ((eNodeSocketDatatype)sock->type) { | 
					
						
							|  |  |  |     case SOCK_FLOAT: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueFloat, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_VECTOR: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueVector, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_RGBA: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueRGBA, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_BOOLEAN: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueBoolean, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_INT: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueInt, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_STRING: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueString, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-04-20 13:22:20 +02:00
										 |  |  |     case SOCK_OBJECT: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueObject, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SOCK_IMAGE: | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeSocketValueImage, 1, sock->default_value); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-03-06 13:29:58 +01:00
										 |  |  |     case __SOCK_MESH: | 
					
						
							|  |  |  |     case SOCK_CUSTOM: | 
					
						
							|  |  |  |     case SOCK_SHADER: | 
					
						
							| 
									
										
										
										
											2020-04-20 13:41:21 +02:00
										 |  |  |     case SOCK_EMITTERS: | 
					
						
							|  |  |  |     case SOCK_EVENTS: | 
					
						
							|  |  |  |     case SOCK_FORCES: | 
					
						
							|  |  |  |     case SOCK_CONTROL_FLOW: | 
					
						
							| 
									
										
										
										
											2020-03-06 13:29:58 +01:00
										 |  |  |       BLI_assert(false); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  | static void write_node_socket(WriteData *wd, bNodeSocket *sock) | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* actual socket writing */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, bNodeSocket, 1, sock); | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (sock->prop) { | 
					
						
							|  |  |  |     IDP_WriteProperty(sock->prop, wd); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 13:29:58 +01:00
										 |  |  |   write_node_socket_default_value(wd, sock); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  | static void write_node_socket_interface(WriteData *wd, bNodeSocket *sock) | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* actual socket writing */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, bNodeSocket, 1, sock); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (sock->prop) { | 
					
						
							|  |  |  |     IDP_WriteProperty(sock->prop, wd); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 13:29:58 +01:00
										 |  |  |   write_node_socket_default_value(wd, sock); | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | } | 
					
						
							| 
									
										
											  
											
												Orange: more noodle updates!
**** NEW: Group Nodes
Node trees usually become messy and confusing quickly, so we need
not only a way to collapse Nodes into single 'groups', but also a
way to re-use that data to create libraries of effects.
This has been done by making a new Library data type, the NodeTree.
Everything that has been grouped is stored here, and available for
re-use, appending or linking. These NodeTrees are fully generic,
i.e. can store shader trees, composit trees, and so on. The 'type'
value as stored in the NodeTree will keep track of internal type
definitions and execute/drawing callbacks. Needless to say, re-using
shader trees in a composit tree is a bit useless, and will be
prevented in the browsing code. :)
So; any NodeTree can become a "Goup Node" inside in a NodeTree. This
Group Node then works just like any Node.
To prevent the current code to become too complex, I've disabled
the possibility to insert Groups inside of Groups. That might be
enabled later, but is a real nasty piece of code to get OK.
Since Group Nodes are a dynamic Node type, a lot of work has been
done to ensure Node definitions can be dynamic too, but still allow
to be stored in files, and allow to be verified for type-definition
changes on reloading. This system needs a little bit maturing still,
so the Python gurus should better wait a little bit! (Also for me to
write the definite API docs for it).
What works now:
- Press CTRL+G to create a new Group. The grouping code checks for
impossible selections (like an unselected node between selected nodes).
Everthing that's selected then gets removed from the current tree, and
inserted in a new NodeTree library data block. A Group Node then is
added which links to this new NodeTree.
- Press ALT+G to ungroup. This will not delete the NodeTree library
data, but just duplicate the Group into the current tree.
- Press TAB, or click on the NodeTree icon to edit Groups. Note that
NodeTrees are instances, so editing one Group will also change the
other users.
This also means that when removing nodes in a Group (or hiding sockets
or changing internal links) this is immediately corrected for all users
of this Group, also in other Materials.
- While editing Groups, only the internal Nodes can be edited. A single
click outside of the Group boundary will close this 'edit mode'.
What needs to be done:
- SHIFT+A menu in toolbox style, also including a list of Groups
- Enable the single-user button in the Group Node
- Displaying all (visible) internal group UI elements in the Node Panel
- Enable Library linking and prevent editing of Groups then.
**** NEW: Socket Visibility control
Node types will be generated with a lot of possible inputs or outputs,
and drawing all sockets all the time isn't very useful then.
A new option in the Node header ('plus' icon) allows to either hide all
unused sockets (first keypress) or to reveil them (when there are hidden
sockets, the icon displays black, otherwise it's blended).
Hidden sockets in Nodes also are not exported to a Group, so this way
you can control what options (in/outputs) exactly are available.
To be done:
- a way to hide individual sockets, like with a RMB click on it.
**** NEW: Nodes now render!
This is still quite primitive, more on a level to replace the (now
obsolete and disabled) Material Layers.
What needs to be done:
- make the "Geometry" node work properly, also for AA textures
- make the Texture Node work (does very little at the moment)
- give Material Nodes all inputs as needed (like Map-to Panel)
- find a way to export more data from a Material Node, like the
  shadow value, or light intensity only, etc
Very important also to separate from the Material Buttons the
"global" options, like "Ztransp" or "Wire" or "Halo". These can not
be set for each Material-Node individually.
Also note that the Preview Render (Buttons window) now renders a bit
differently. This was a horrid piece of antique code, using a totally
incompatible way of rendering. Target is to fully re-use internal
render code for previews.
OK... that's it mostly. Now test!
											
										 
											2006-01-02 13:06:05 +00:00
										 |  |  | /* this is only direct data, tree itself should have been written */ | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | static void write_nodetree_nolib(WriteData *wd, bNodeTree *ntree) | 
					
						
							| 
									
										
											  
											
												Christmas coding work!
********* Node editor work:
- To enable Nodes for Materials, you have to set the "Use Nodes"
  button, in the new Material buttons "Nodes" Panel or in header
  of the Node editor. Doing this will disable Material-Layers.
- Nodes now execute materials ("shaders"), but still only using the
  previewrender code.
- Nodes have (optional) previews for rendered images.
- Node headers allow to hide buttons and/or preview image
- Nodes can be dragged larger/smaller (right-bottom corner)
- Nodes can be hidden (minimized) with hotkey H
- CTRL+click on an Input Socket gives a popup with default values.
- Changing Material/Texture or Mix node will adjust Node title.
- Click-drag outside of a Node changes cursor to "Knife' and allows to
  draw a rect where to cut Links.
- Added new node types RGBtoBW, Texture, In/Output, ColorRamp
- Material Nodes have options to ouput diffuse or specular, or to use
  a negative normal. The input socket 'Normal' will force the material
  to use that normal, otherwise it uses the normal from the Material
  that has the node tree.
- When drawing a link between two not-matching sockets, Blender inserts
  a converting node (now only for value/rgb combos)
- When drawing a link to an input socket that's already in use, the
  old link will either disappear or flip to another unused socket.
- A click on a Material Node will activate it, and show all its settings
  in the Material Buttons. Active Material Nodes draw the material icon
  in red.
- A click on any node will show its options in the Node Panel in the
  Material buttons.
- Multiple Output Nodes can be used, to sample contents of a tree, but
  only one Output is the real one, which is indicated in a different
  color and red material icon.
- Added ThemeColors for node types
- ALT+C will convert existing Material-Layers to Node... this currently
  only adds the material/mix nodes and connects them. Dunno if this is
  worth a lot of coding work to make perfect?
- Press C to call another "Solve order", which will show all possible
  cyclic conflicts (if there are).
- Technical: nodes now use "Type" structs which define the
  structure of nodes and in/output sockets. The Type structs store all
  fixed info, callbacks, and allow to reconstruct saved Nodes to match
  what is required by Blender.
- Defining (new) nodes now is as simple as filling in a fixed
  Type struct, plus code some callbacks. A doc will be made!
- Node preview images are by default float
********* Icon drawing:
- Cleanup of how old icons were implemented in new system, making
  them 16x16 too, correctly centered *and* scaled.
- Made drawing Icons use float coordinates
- Moved BIF_calcpreview_image() into interface_icons.c, renamed it
  icon_from_image(). Removed a lot of unneeded Imbuf magic here! :)
- Skipped scaling and imbuf copying when icons are OK size
********* Preview render:
- Huge cleanup of code....
- renaming BIF_xxx calls that only were used internally
- BIF_previewrender() now accepts an argument for rendering method,
  so it supports icons, buttonwindow previewrender and node editor
- Only a single BIF_preview_changed() call now exists, supporting all
  signals as needed for buttos and node editor
********* More stuff:
- glutil.c, glaDrawPixelsSafe() and glaDrawPixelsTex() now accept format
  argument for GL_FLOAT rects
- Made the ColorBand become a built-in button for interface.c
  Was a load of cleanup work in buttons_shading.c...
- removed a load of unneeded glBlendFunc() calls
- Fixed bug in calculating text length for buttons (ancient!)
											
										 
											2005-12-28 15:42:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bNode *node; | 
					
						
							|  |  |  |   bNodeSocket *sock; | 
					
						
							|  |  |  |   bNodeLink *link; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* for link_list() speed, we write per list */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ntree->adt) { | 
					
						
							|  |  |  |     write_animdata(wd, ntree->adt); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (node = ntree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, bNode, 1, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (node->prop) { | 
					
						
							|  |  |  |       IDP_WriteProperty(node->prop, wd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (sock = node->inputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  |       write_node_socket(wd, sock); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     for (sock = node->outputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  |       write_node_socket(wd, sock); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (link = node->internal_links.first; link; link = link->next) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeLink, 1, link); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (node->storage) { | 
					
						
							|  |  |  |       /* could be handlerized at some point, now only 1 exception still */ | 
					
						
							|  |  |  |       if ((ntree->type == NTREE_SHADER) && | 
					
						
							|  |  |  |           ELEM(node->type, SH_NODE_CURVE_VEC, SH_NODE_CURVE_RGB)) { | 
					
						
							|  |  |  |         write_curvemapping(wd, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (ntree->type == NTREE_SHADER && (node->type == SH_NODE_SCRIPT)) { | 
					
						
							|  |  |  |         NodeShaderScript *nss = (NodeShaderScript *)node->storage; | 
					
						
							|  |  |  |         if (nss->bytecode) { | 
					
						
							|  |  |  |           writedata(wd, DATA, strlen(nss->bytecode) + 1, nss->bytecode); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         writestruct_id(wd, DATA, node->typeinfo->storagename, 1, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if ((ntree->type == NTREE_COMPOSIT) && ELEM(node->type, | 
					
						
							|  |  |  |                                                        CMP_NODE_TIME, | 
					
						
							|  |  |  |                                                        CMP_NODE_CURVE_VEC, | 
					
						
							|  |  |  |                                                        CMP_NODE_CURVE_RGB, | 
					
						
							|  |  |  |                                                        CMP_NODE_HUECORRECT)) { | 
					
						
							|  |  |  |         write_curvemapping(wd, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if ((ntree->type == NTREE_TEXTURE) && | 
					
						
							|  |  |  |                (node->type == TEX_NODE_CURVE_RGB || node->type == TEX_NODE_CURVE_TIME)) { | 
					
						
							|  |  |  |         write_curvemapping(wd, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if ((ntree->type == NTREE_COMPOSIT) && (node->type == CMP_NODE_MOVIEDISTORTION)) { | 
					
						
							|  |  |  |         /* pass */ | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if ((ntree->type == NTREE_COMPOSIT) && (node->type == CMP_NODE_GLARE)) { | 
					
						
							| 
									
										
										
										
											2019-06-16 13:37:21 +10:00
										 |  |  |         /* Simple forward compatibility for fix for T50736.
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |          * Not ideal (there is no ideal solution here), but should do for now. */ | 
					
						
							|  |  |  |         NodeGlare *ndg = node->storage; | 
					
						
							|  |  |  |         /* Not in undo case. */ | 
					
						
							|  |  |  |         if (wd->use_memfile == false) { | 
					
						
							|  |  |  |           switch (ndg->type) { | 
					
						
							|  |  |  |             case 2: /* Grrrr! magic numbers :( */ | 
					
						
							|  |  |  |               ndg->angle = ndg->streaks; | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case 0: | 
					
						
							|  |  |  |               ndg->angle = ndg->star_45; | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         writestruct_id(wd, DATA, node->typeinfo->storagename, 1, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if ((ntree->type == NTREE_COMPOSIT) && (node->type == CMP_NODE_CRYPTOMATTE)) { | 
					
						
							|  |  |  |         NodeCryptomatte *nc = (NodeCryptomatte *)node->storage; | 
					
						
							|  |  |  |         if (nc->matte_id) { | 
					
						
							|  |  |  |           writedata(wd, DATA, strlen(nc->matte_id) + 1, nc->matte_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         writestruct_id(wd, DATA, node->typeinfo->storagename, 1, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         writestruct_id(wd, DATA, node->typeinfo->storagename, 1, node->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (node->type == CMP_NODE_OUTPUT_FILE) { | 
					
						
							|  |  |  |       /* inputs have own storage data */ | 
					
						
							|  |  |  |       for (sock = node->inputs.first; sock; sock = sock->next) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, NodeImageMultiFileSocket, 1, sock->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ELEM(node->type, CMP_NODE_IMAGE, CMP_NODE_R_LAYERS)) { | 
					
						
							|  |  |  |       /* write extra socket info */ | 
					
						
							|  |  |  |       for (sock = node->outputs.first; sock; sock = sock->next) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, NodeImageLayer, 1, sock->storage); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (link = ntree->links.first; link; link = link->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, bNodeLink, 1, link); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (sock = ntree->inputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  |     write_node_socket_interface(wd, sock); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   for (sock = ntree->outputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2019-04-20 20:25:21 +02:00
										 |  |  |     write_node_socket_interface(wd, sock); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
											  
											
												Christmas coding work!
********* Node editor work:
- To enable Nodes for Materials, you have to set the "Use Nodes"
  button, in the new Material buttons "Nodes" Panel or in header
  of the Node editor. Doing this will disable Material-Layers.
- Nodes now execute materials ("shaders"), but still only using the
  previewrender code.
- Nodes have (optional) previews for rendered images.
- Node headers allow to hide buttons and/or preview image
- Nodes can be dragged larger/smaller (right-bottom corner)
- Nodes can be hidden (minimized) with hotkey H
- CTRL+click on an Input Socket gives a popup with default values.
- Changing Material/Texture or Mix node will adjust Node title.
- Click-drag outside of a Node changes cursor to "Knife' and allows to
  draw a rect where to cut Links.
- Added new node types RGBtoBW, Texture, In/Output, ColorRamp
- Material Nodes have options to ouput diffuse or specular, or to use
  a negative normal. The input socket 'Normal' will force the material
  to use that normal, otherwise it uses the normal from the Material
  that has the node tree.
- When drawing a link between two not-matching sockets, Blender inserts
  a converting node (now only for value/rgb combos)
- When drawing a link to an input socket that's already in use, the
  old link will either disappear or flip to another unused socket.
- A click on a Material Node will activate it, and show all its settings
  in the Material Buttons. Active Material Nodes draw the material icon
  in red.
- A click on any node will show its options in the Node Panel in the
  Material buttons.
- Multiple Output Nodes can be used, to sample contents of a tree, but
  only one Output is the real one, which is indicated in a different
  color and red material icon.
- Added ThemeColors for node types
- ALT+C will convert existing Material-Layers to Node... this currently
  only adds the material/mix nodes and connects them. Dunno if this is
  worth a lot of coding work to make perfect?
- Press C to call another "Solve order", which will show all possible
  cyclic conflicts (if there are).
- Technical: nodes now use "Type" structs which define the
  structure of nodes and in/output sockets. The Type structs store all
  fixed info, callbacks, and allow to reconstruct saved Nodes to match
  what is required by Blender.
- Defining (new) nodes now is as simple as filling in a fixed
  Type struct, plus code some callbacks. A doc will be made!
- Node preview images are by default float
********* Icon drawing:
- Cleanup of how old icons were implemented in new system, making
  them 16x16 too, correctly centered *and* scaled.
- Made drawing Icons use float coordinates
- Moved BIF_calcpreview_image() into interface_icons.c, renamed it
  icon_from_image(). Removed a lot of unneeded Imbuf magic here! :)
- Skipped scaling and imbuf copying when icons are OK size
********* Preview render:
- Huge cleanup of code....
- renaming BIF_xxx calls that only were used internally
- BIF_previewrender() now accepts an argument for rendering method,
  so it supports icons, buttonwindow previewrender and node editor
- Only a single BIF_preview_changed() call now exists, supporting all
  signals as needed for buttos and node editor
********* More stuff:
- glutil.c, glaDrawPixelsSafe() and glaDrawPixelsTex() now accept format
  argument for GL_FLOAT rects
- Made the ColorBand become a built-in button for interface.c
  Was a load of cleanup work in buttons_shading.c...
- removed a load of unneeded glBlendFunc() calls
- Fixed bug in calculating text length for buttons (ancient!)
											
										 
											2005-12-28 15:42:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 05:45:57 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Take care using 'use_active_win', since we wont want the currently active window | 
					
						
							|  |  |  |  * to change which scene renders (currently only used for undo). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void current_screen_compat(Main *mainvar, | 
					
						
							|  |  |  |                                   bool use_active_win, | 
					
						
							|  |  |  |                                   bScreen **r_screen, | 
					
						
							|  |  |  |                                   Scene **r_scene, | 
					
						
							|  |  |  |                                   ViewLayer **r_view_layer) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wmWindowManager *wm; | 
					
						
							|  |  |  |   wmWindow *window = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* find a global current screen in the first open window, to have
 | 
					
						
							|  |  |  |    * a reasonable default for reading in older versions */ | 
					
						
							|  |  |  |   wm = mainvar->wm.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (wm) { | 
					
						
							|  |  |  |     if (use_active_win) { | 
					
						
							|  |  |  |       /* write the active window into the file, needed for multi-window undo T43424 */ | 
					
						
							|  |  |  |       for (window = wm->windows.first; window; window = window->next) { | 
					
						
							|  |  |  |         if (window->active) { | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* fallback */ | 
					
						
							|  |  |  |       if (window == NULL) { | 
					
						
							|  |  |  |         window = wm->windows.first; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       window = wm->windows.first; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *r_screen = (window) ? BKE_workspace_active_screen_get(window->workspace_hook) : NULL; | 
					
						
							|  |  |  |   *r_scene = (window) ? window->scene : NULL; | 
					
						
							|  |  |  |   *r_view_layer = (window && *r_scene) ? BKE_view_layer_find(*r_scene, window->view_layer_name) : | 
					
						
							|  |  |  |                                          NULL; | 
					
						
							| 
									
										
										
										
											2008-12-19 16:36:15 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-26 04:03:25 +00:00
										 |  |  | typedef struct RenderInfo { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int sfra; | 
					
						
							|  |  |  |   int efra; | 
					
						
							|  |  |  |   char scene_name[MAX_ID_NAME - 2]; | 
					
						
							| 
									
										
										
										
											2012-04-26 04:03:25 +00:00
										 |  |  | } RenderInfo; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 00:54:22 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This was originally added for the historic render-daemon feature, | 
					
						
							|  |  |  |  * now write because it can be easily extracted without reading the whole blend file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * See: `release/scripts/modules/blend_render_info.py` | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-10-15 02:15:07 +00:00
										 |  |  | static void write_renderinfo(WriteData *wd, Main *mainvar) | 
					
						
							| 
									
										
										
										
											2008-12-19 16:36:15 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bScreen *curscreen; | 
					
						
							|  |  |  |   Scene *sce, *curscene = NULL; | 
					
						
							|  |  |  |   ViewLayer *view_layer; | 
					
						
							|  |  |  |   RenderInfo data; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* XXX in future, handle multiple windows with multiple screens? */ | 
					
						
							|  |  |  |   current_screen_compat(mainvar, false, &curscreen, &curscene, &view_layer); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (sce = mainvar->scenes.first; sce; sce = sce->id.next) { | 
					
						
							|  |  |  |     if (sce->id.lib == NULL && (sce == curscene || (sce->r.scemode & R_BG_RENDER))) { | 
					
						
							|  |  |  |       data.sfra = sce->r.sfra; | 
					
						
							|  |  |  |       data.efra = sce->r.efra; | 
					
						
							|  |  |  |       memset(data.scene_name, 0, sizeof(data.scene_name)); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       BLI_strncpy(data.scene_name, sce->id.name + 2, sizeof(data.scene_name)); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writedata(wd, REND, sizeof(data), &data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-01 15:16:36 +11:00
										 |  |  | static void write_keymapitem(WriteData *wd, const wmKeyMapItem *kmi) | 
					
						
							| 
									
										
											  
											
												KEYMAP REFACTORING
Diff Keymaps
User edited keymaps now no longer override the builtin keymaps entirely, but
rather save only the difference and reapply those changes. This means they can
stay better in sync when the builtin keymaps change. The diff/patch algorithm
is not perfect, but better for the common case where only a few items are changed
rather than entire keymaps The main weakness is that if a builtin keymap item
changes, user modification of that item may need to be redone in some cases.
Keymap Editor
The most noticeable change here is that there is no longer an "Edit" button for
keymaps, all are editable immediately, but a "Restore" buttons shows for keymaps
and items that have been edited. Shortcuts for addons can also be edited in the
keymap editor. 
Addons
Addons now should only modify the new addon keyconfiguration, the keymap items
there will be added to the builtin ones for handling events, and not get lost
when starting new files. Example code of register/unregister:
km = wm.keyconfigs.addon.keymaps.new("3D View", space_type="VIEW_3D")
km.keymap_items.new('my.operator', 'ESC', 'PRESS')
km = wm.keyconfigs.addon.keymaps["3D View"]
km.keymap_items.remove(km.keymap_items["my.operator"])
Compatibility
The changes made are not forward compatible, i.e. if you save user preferences
with newer versions, older versions will not have key configuration changes that
were made.
											
										 
											2011-08-05 20:45:26 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, DATA, wmKeyMapItem, 1, kmi); | 
					
						
							|  |  |  |   if (kmi->properties) { | 
					
						
							|  |  |  |     IDP_WriteProperty(kmi->properties, wd); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												KEYMAP REFACTORING
Diff Keymaps
User edited keymaps now no longer override the builtin keymaps entirely, but
rather save only the difference and reapply those changes. This means they can
stay better in sync when the builtin keymaps change. The diff/patch algorithm
is not perfect, but better for the common case where only a few items are changed
rather than entire keymaps The main weakness is that if a builtin keymap item
changes, user modification of that item may need to be redone in some cases.
Keymap Editor
The most noticeable change here is that there is no longer an "Edit" button for
keymaps, all are editable immediately, but a "Restore" buttons shows for keymaps
and items that have been edited. Shortcuts for addons can also be edited in the
keymap editor. 
Addons
Addons now should only modify the new addon keyconfiguration, the keymap items
there will be added to the builtin ones for handling events, and not get lost
when starting new files. Example code of register/unregister:
km = wm.keyconfigs.addon.keymaps.new("3D View", space_type="VIEW_3D")
km.keymap_items.new('my.operator', 'ESC', 'PRESS')
km = wm.keyconfigs.addon.keymaps["3D View"]
km.keymap_items.remove(km.keymap_items["my.operator"])
Compatibility
The changes made are not forward compatible, i.e. if you save user preferences
with newer versions, older versions will not have key configuration changes that
were made.
											
										 
											2011-08-05 20:45:26 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-01 15:16:36 +11:00
										 |  |  | static void write_userdef(WriteData *wd, const UserDef *userdef) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, USER, UserDef, 1, userdef); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const bTheme *, btheme, &userdef->themes) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bTheme, 1, btheme); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const wmKeyMap *, keymap, &userdef->user_keymaps) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, wmKeyMap, 1, keymap); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (const wmKeyMapDiffItem *, kmdi, &keymap->diff_items) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, wmKeyMapDiffItem, 1, kmdi); | 
					
						
							|  |  |  |       if (kmdi->remove_item) { | 
					
						
							|  |  |  |         write_keymapitem(wd, kmdi->remove_item); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (kmdi->add_item) { | 
					
						
							|  |  |  |         write_keymapitem(wd, kmdi->add_item); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (const wmKeyMapItem *, kmi, &keymap->items) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_keymapitem(wd, kmi); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const wmKeyConfigPref *, kpt, &userdef->user_keyconfig_prefs) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, wmKeyConfigPref, 1, kpt); | 
					
						
							|  |  |  |     if (kpt->prop) { | 
					
						
							|  |  |  |       IDP_WriteProperty(kpt->prop, wd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const bUserMenu *, um, &userdef->user_menus) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bUserMenu, 1, um); | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (const bUserMenuItem *, umi, &um->items) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (umi->type == USER_MENU_TYPE_OPERATOR) { | 
					
						
							|  |  |  |         const bUserMenuItem_Op *umi_op = (const bUserMenuItem_Op *)umi; | 
					
						
							|  |  |  |         writestruct(wd, DATA, bUserMenuItem_Op, 1, umi_op); | 
					
						
							|  |  |  |         if (umi_op->prop) { | 
					
						
							|  |  |  |           IDP_WriteProperty(umi_op->prop, wd); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (umi->type == USER_MENU_TYPE_MENU) { | 
					
						
							|  |  |  |         const bUserMenuItem_Menu *umi_mt = (const bUserMenuItem_Menu *)umi; | 
					
						
							|  |  |  |         writestruct(wd, DATA, bUserMenuItem_Menu, 1, umi_mt); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (umi->type == USER_MENU_TYPE_PROP) { | 
					
						
							|  |  |  |         const bUserMenuItem_Prop *umi_pr = (const bUserMenuItem_Prop *)umi; | 
					
						
							|  |  |  |         writestruct(wd, DATA, bUserMenuItem_Prop, 1, umi_pr); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         writestruct(wd, DATA, bUserMenuItem, 1, umi); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const bAddon *, bext, &userdef->addons) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bAddon, 1, bext); | 
					
						
							|  |  |  |     if (bext->prop) { | 
					
						
							|  |  |  |       IDP_WriteProperty(bext->prop, wd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const bPathCompare *, path_cmp, &userdef->autoexec_paths) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bPathCompare, 1, path_cmp); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (const uiStyle *, style, &userdef->uistyles) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, uiStyle, 1, style); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | static void write_boid_state(WriteData *wd, BoidState *state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BoidRule *rule = state->rules.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   writestruct(wd, DATA, BoidState, 1, state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (; rule; rule = rule->next) { | 
					
						
							|  |  |  |     switch (rule->type) { | 
					
						
							|  |  |  |       case eBoidRuleType_Goal: | 
					
						
							|  |  |  |       case eBoidRuleType_Avoid: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRuleGoalAvoid, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case eBoidRuleType_AvoidCollision: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRuleAvoidCollision, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case eBoidRuleType_FollowLeader: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRuleFollowLeader, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case eBoidRuleType_AverageSpeed: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRuleAverageSpeed, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case eBoidRuleType_Fight: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRuleFight, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidRule, 1, rule); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BoidCondition *cond = state->conditions.first; | 
					
						
							|  |  |  |   for (; cond; cond = cond->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, BoidCondition, 1, cond); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* update this also to readfile.c */ | 
					
						
							|  |  |  | static const char *ptcache_data_struct[] = { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     "",          // BPHYS_DATA_INDEX
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_LOCATION
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_VELOCITY
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_ROTATION
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_AVELOCITY / BPHYS_DATA_XCONST */
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_SIZE:
 | 
					
						
							|  |  |  |     "",          // BPHYS_DATA_TIMES:
 | 
					
						
							|  |  |  |     "BoidData",  // case BPHYS_DATA_BOIDS:
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | static const char *ptcache_extra_struct[] = { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     "", | 
					
						
							|  |  |  |     "ParticleSpring", | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | static void write_pointcaches(WriteData *wd, ListBase *ptcaches) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PointCache *cache = ptcaches->first; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (; cache; cache = cache->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, PointCache, 1, cache); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((cache->flag & PTCACHE_DISK_CACHE) == 0) { | 
					
						
							|  |  |  |       PTCacheMem *pm = cache->mem_cache.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (; pm; pm = pm->next) { | 
					
						
							|  |  |  |         PTCacheExtra *extra = pm->extradata.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         writestruct(wd, DATA, PTCacheMem, 1, pm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (i = 0; i < BPHYS_TOT_DATA; i++) { | 
					
						
							|  |  |  |           if (pm->data[i] && pm->data_types & (1 << i)) { | 
					
						
							|  |  |  |             if (ptcache_data_struct[i][0] == '\0') { | 
					
						
							|  |  |  |               writedata(wd, DATA, MEM_allocN_len(pm->data[i]), pm->data[i]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |               writestruct_id(wd, DATA, ptcache_data_struct[i], pm->totpoint, pm->data[i]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (; extra; extra = extra->next) { | 
					
						
							|  |  |  |           if (ptcache_extra_struct[extra->type][0] == '\0') { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           writestruct(wd, DATA, PTCacheExtra, 1, extra); | 
					
						
							|  |  |  |           writestruct_id(wd, DATA, ptcache_extra_struct[extra->type], extra->totdata, extra->data); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_particlesettings(WriteData *wd, ParticleSettings *part, const void *id_address) | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (part->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_PA, ParticleSettings, 1, id_address, part); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &part->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (part->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, part->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct(wd, DATA, PartDeflect, 1, part->pd); | 
					
						
							|  |  |  |     writestruct(wd, DATA, PartDeflect, 1, part->pd2); | 
					
						
							|  |  |  |     writestruct(wd, DATA, EffectorWeights, 1, part->effector_weights); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (part->clumpcurve) { | 
					
						
							|  |  |  |       write_curvemapping(wd, part->clumpcurve); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (part->roughcurve) { | 
					
						
							|  |  |  |       write_curvemapping(wd, part->roughcurve); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (part->twistcurve) { | 
					
						
							|  |  |  |       write_curvemapping(wd, part->twistcurve); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (ParticleDupliWeight *, dw, &part->instance_weights) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       /* update indices, but only if dw->ob is set (can be NULL after loading e.g.) */ | 
					
						
							|  |  |  |       if (dw->ob != NULL) { | 
					
						
							|  |  |  |         dw->index = 0; | 
					
						
							|  |  |  |         if (part->instance_collection) { /* can be NULL if lining fails or set to None */ | 
					
						
							|  |  |  |           FOREACH_COLLECTION_OBJECT_RECURSIVE_BEGIN (part->instance_collection, object) { | 
					
						
							|  |  |  |             if (object == dw->ob) { | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             dw->index++; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           FOREACH_COLLECTION_OBJECT_RECURSIVE_END; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       writestruct(wd, DATA, ParticleDupliWeight, 1, dw); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (part->boids && part->phystype == PART_PHYS_BOIDS) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, BoidSettings, 1, part->boids); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |       LISTBASE_FOREACH (BoidState *, state, &part->boids->states) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         write_boid_state(wd, state); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (part->fluid && part->phystype == PART_PHYS_FLUID) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SPHFluidSettings, 1, part->fluid); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int a = 0; a < MAX_MTEX; a++) { | 
					
						
							|  |  |  |       if (part->mtex[a]) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, MTex, 1, part->mtex[a]); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | static void write_particlesystems(WriteData *wd, ListBase *particles) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ParticleSystem *psys = particles->first; | 
					
						
							|  |  |  |   ParticleTarget *pt; | 
					
						
							|  |  |  |   int a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (; psys; psys = psys->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, ParticleSystem, 1, psys); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (psys->particles) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ParticleData, psys->totpart, psys->particles); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (psys->particles->hair) { | 
					
						
							|  |  |  |         ParticleData *pa = psys->particles; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (a = 0; a < psys->totpart; a++, pa++) { | 
					
						
							|  |  |  |           writestruct(wd, DATA, HairKey, pa->totkey, pa->hair); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (psys->particles->boid && (psys->part->phystype == PART_PHYS_BOIDS)) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, BoidParticle, psys->totpart, psys->particles->boid); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (psys->part->fluid && (psys->part->phystype == PART_PHYS_FLUID) && | 
					
						
							|  |  |  |           (psys->part->fluid->flag & SPH_VISCOELASTIC_SPRINGS)) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, ParticleSpring, psys->tot_fluidsprings, psys->fluid_springs); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pt = psys->targets.first; | 
					
						
							|  |  |  |     for (; pt; pt = pt->next) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ParticleTarget, 1, pt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (psys->child) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ChildParticle, psys->totchild, psys->child); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (psys->clmd) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ClothModifierData, 1, psys->clmd); | 
					
						
							|  |  |  |       writestruct(wd, DATA, ClothSimSettings, 1, psys->clmd->sim_parms); | 
					
						
							|  |  |  |       writestruct(wd, DATA, ClothCollSettings, 1, psys->clmd->coll_parms); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_pointcaches(wd, &psys->ptcaches); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:30:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cleanup of MotionPaths+Ghosts (AnimViz) - Part 1
This commit sets up some of the groundwork necessary to extend the animation visualisation capabilities, previously only available for bones in PoseMode, to Objects as well. Also, some of the other goals of this refactor is to make future visualisation goodies (i.e. editable paths) more feasible...
(There's really nothing to see here yet. The following log notes are really just for my own reference to keep track of things.)
Currently, the following things have been done:
* New datastructures + settings have been tidied up, ready for usage
* Added these new types into the Object and PoseBone code as necessary, with freeing/adding/copying accounted for
* File IO code for the new data, including version patching to convert the old system to the new one.
* Set up the drawing system for motionpaths based on the old armature path drawing code. Armatures still draw using the old system, since the two systems use different storage systems.
* Started setting up the motionpath 'baking' code, but the core of this still needs to be coded...
Next Steps (after some semi-urgent Durian Driver changes):
* Port the ghosting/onionskinning code over too
* Finish motionpath baking code
* RNA wrapping for the new types
* Hooking up all the new code into the operators, etc.
											
										 
											2010-01-01 12:24:16 +00:00
										 |  |  | static void write_motionpath(WriteData *wd, bMotionPath *mpath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* sanity checks */ | 
					
						
							|  |  |  |   if (mpath == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* firstly, just write the motionpath struct */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, bMotionPath, 1, mpath); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* now write the array of data */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, bMotionPathVert, mpath->length, mpath->points); | 
					
						
							| 
									
										
											  
											
												Cleanup of MotionPaths+Ghosts (AnimViz) - Part 1
This commit sets up some of the groundwork necessary to extend the animation visualisation capabilities, previously only available for bones in PoseMode, to Objects as well. Also, some of the other goals of this refactor is to make future visualisation goodies (i.e. editable paths) more feasible...
(There's really nothing to see here yet. The following log notes are really just for my own reference to keep track of things.)
Currently, the following things have been done:
* New datastructures + settings have been tidied up, ready for usage
* Added these new types into the Object and PoseBone code as necessary, with freeing/adding/copying accounted for
* File IO code for the new data, including version patching to convert the old system to the new one.
* Set up the drawing system for motionpaths based on the old armature path drawing code. Armatures still draw using the old system, since the two systems use different storage systems.
* Started setting up the motionpath 'baking' code, but the core of this still needs to be coded...
Next Steps (after some semi-urgent Durian Driver changes):
* Port the ghosting/onionskinning code over too
* Finish motionpath baking code
* RNA wrapping for the new types
* Hooking up all the new code into the operators, etc.
											
										 
											2010-01-01 12:24:16 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | static void write_constraints(WriteData *wd, ListBase *conlist) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bConstraint *con; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (con = conlist->first; con; con = con->next) { | 
					
						
							|  |  |  |     const bConstraintTypeInfo *cti = BKE_constraint_typeinfo_get(con); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write the specific data */ | 
					
						
							|  |  |  |     if (cti && con->data) { | 
					
						
							|  |  |  |       /* firstly, just write the plain con->data struct */ | 
					
						
							|  |  |  |       writestruct_id(wd, DATA, cti->structName, 1, con->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* do any constraint specific stuff */ | 
					
						
							|  |  |  |       switch (con->type) { | 
					
						
							|  |  |  |         case CONSTRAINT_TYPE_PYTHON: { | 
					
						
							|  |  |  |           bPythonConstraint *data = con->data; | 
					
						
							|  |  |  |           bConstraintTarget *ct; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* write targets */ | 
					
						
							|  |  |  |           for (ct = data->targets.first; ct; ct = ct->next) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, bConstraintTarget, 1, ct); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* Write ID Properties -- and copy this comment EXACTLY for easy finding
 | 
					
						
							|  |  |  |            * of library blocks that implement this.*/ | 
					
						
							|  |  |  |           IDP_WriteProperty(data->prop, wd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case CONSTRAINT_TYPE_ARMATURE: { | 
					
						
							|  |  |  |           bArmatureConstraint *data = con->data; | 
					
						
							|  |  |  |           bConstraintTarget *ct; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* write targets */ | 
					
						
							|  |  |  |           for (ct = data->targets.first; ct; ct = ct->next) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, bConstraintTarget, 1, ct); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case CONSTRAINT_TYPE_SPLINEIK: { | 
					
						
							|  |  |  |           bSplineIKConstraint *data = con->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* write points array */ | 
					
						
							|  |  |  |           writedata(wd, DATA, sizeof(float) * (data->numpoints), data->points); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write the constraint */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, bConstraint, 1, con); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_pose(WriteData *wd, bPose *pose) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bPoseChannel *chan; | 
					
						
							|  |  |  |   bActionGroup *grp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write each channel */ | 
					
						
							|  |  |  |   if (pose == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write channels */ | 
					
						
							|  |  |  |   for (chan = pose->chanbase.first; chan; chan = chan->next) { | 
					
						
							|  |  |  |     /* Write ID Properties -- and copy this comment EXACTLY for easy finding
 | 
					
						
							|  |  |  |      * of library blocks that implement this.*/ | 
					
						
							|  |  |  |     if (chan->prop) { | 
					
						
							|  |  |  |       IDP_WriteProperty(chan->prop, wd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_constraints(wd, &chan->constraints); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_motionpath(wd, chan->mpath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* prevent crashes with autosave,
 | 
					
						
							|  |  |  |      * when a bone duplicated in editmode has not yet been assigned to its posechannel */ | 
					
						
							|  |  |  |     if (chan->bone) { | 
					
						
							|  |  |  |       /* gets restored on read, for library armatures */ | 
					
						
							|  |  |  |       chan->selectflag = chan->bone->flag & BONE_SELECTED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, bPoseChannel, 1, chan); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write groups */ | 
					
						
							|  |  |  |   for (grp = pose->agroups.first; grp; grp = grp->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, bActionGroup, 1, grp); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* write IK param */ | 
					
						
							|  |  |  |   if (pose->ikparam) { | 
					
						
							|  |  |  |     const char *structname = BKE_pose_ikparam_get_name(pose); | 
					
						
							|  |  |  |     if (structname) { | 
					
						
							|  |  |  |       writestruct_id(wd, DATA, structname, 1, pose->ikparam); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write this pose */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, bPose, 1, pose); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_defgroups(WriteData *wd, ListBase *defbase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (bDeformGroup *, defgroup, defbase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bDeformGroup, 1, defgroup); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-30 17:58:24 +10:00
										 |  |  | static void write_fmaps(WriteData *wd, ListBase *fbase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (bFaceMap *, fmap, fbase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, bFaceMap, 1, fmap); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-05-30 17:58:24 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-19 16:49:21 +00:00
										 |  |  | static void write_modifiers(WriteData *wd, ListBase *modbase) | 
					
						
							| 
									
										
										
										
											2005-07-19 20:14:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ModifierData *md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (modbase == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (md = modbase->first; md; md = md->next) { | 
					
						
							| 
									
										
										
										
											2020-05-08 10:14:02 +02:00
										 |  |  |     const ModifierTypeInfo *mti = BKE_modifier_get_info(md->type); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (mti == NULL) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct_id(wd, DATA, mti->structName, 1, md); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (md->type == eModifierType_Hook) { | 
					
						
							|  |  |  |       HookModifierData *hmd = (HookModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (hmd->curfalloff) { | 
					
						
							|  |  |  |         write_curvemapping(wd, hmd->curfalloff); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(int) * hmd->totindex, hmd->indexar); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_Cloth) { | 
					
						
							|  |  |  |       ClothModifierData *clmd = (ClothModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, ClothSimSettings, 1, clmd->sim_parms); | 
					
						
							|  |  |  |       writestruct(wd, DATA, ClothCollSettings, 1, clmd->coll_parms); | 
					
						
							|  |  |  |       writestruct(wd, DATA, EffectorWeights, 1, clmd->sim_parms->effector_weights); | 
					
						
							|  |  |  |       write_pointcaches(wd, &clmd->ptcaches); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |     else if (md->type == eModifierType_Fluid) { | 
					
						
							|  |  |  |       FluidModifierData *mmd = (FluidModifierData *)md; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |       if (mmd->type & MOD_FLUID_TYPE_DOMAIN) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, FluidDomainSettings, 1, mmd->domain); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |         if (mmd->domain) { | 
					
						
							|  |  |  |           write_pointcaches(wd, &(mmd->domain->ptcaches[0])); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |           /* create fake pointcache so that old blender versions can read it */ | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |           mmd->domain->point_cache[1] = BKE_ptcache_add(&mmd->domain->ptcaches[1]); | 
					
						
							|  |  |  |           mmd->domain->point_cache[1]->flag |= PTCACHE_DISK_CACHE | PTCACHE_FAKE_SMOKE; | 
					
						
							|  |  |  |           mmd->domain->point_cache[1]->step = 1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |           write_pointcaches(wd, &(mmd->domain->ptcaches[1])); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |           if (mmd->domain->coba) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, ColorBand, 1, mmd->domain->coba); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* cleanup the fake pointcache */ | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |           BKE_ptcache_free_list(&mmd->domain->ptcaches[1]); | 
					
						
							|  |  |  |           mmd->domain->point_cache[1] = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |           writestruct(wd, DATA, EffectorWeights, 1, mmd->domain->effector_weights); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |       else if (mmd->type & MOD_FLUID_TYPE_FLOW) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, FluidFlowSettings, 1, mmd->flow); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-12-16 15:50:14 +01:00
										 |  |  |       else if (mmd->type & MOD_FLUID_TYPE_EFFEC) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, FluidEffectorSettings, 1, mmd->effector); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_Fluidsim) { | 
					
						
							|  |  |  |       FluidsimModifierData *fluidmd = (FluidsimModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, FluidsimSettings, 1, fluidmd->fss); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_DynamicPaint) { | 
					
						
							|  |  |  |       DynamicPaintModifierData *pmd = (DynamicPaintModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (pmd->canvas) { | 
					
						
							|  |  |  |         DynamicPaintSurface *surface; | 
					
						
							|  |  |  |         writestruct(wd, DATA, DynamicPaintCanvasSettings, 1, pmd->canvas); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* write surfaces */ | 
					
						
							|  |  |  |         for (surface = pmd->canvas->surfaces.first; surface; surface = surface->next) { | 
					
						
							|  |  |  |           writestruct(wd, DATA, DynamicPaintSurface, 1, surface); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         /* write caches and effector weights */ | 
					
						
							|  |  |  |         for (surface = pmd->canvas->surfaces.first; surface; surface = surface->next) { | 
					
						
							|  |  |  |           write_pointcaches(wd, &(surface->ptcaches)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           writestruct(wd, DATA, EffectorWeights, 1, surface->effector_weights); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (pmd->brush) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, DynamicPaintBrushSettings, 1, pmd->brush); | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, pmd->brush->paint_ramp); | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, pmd->brush->vel_ramp); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_Collision) { | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 22:49:33 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       CollisionModifierData *collmd = (CollisionModifierData *)md; | 
					
						
							|  |  |  |       // TODO: CollisionModifier should use pointcache
 | 
					
						
							|  |  |  |       // + have proper reset events before enabling this
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, MVert, collmd->numverts, collmd->x); | 
					
						
							|  |  |  |       writestruct(wd, DATA, MVert, collmd->numverts, collmd->xnew); | 
					
						
							|  |  |  |       writestruct(wd, DATA, MFace, collmd->numfaces, collmd->mfaces); | 
					
						
							| 
									
										
										
										
											2012-06-30 22:49:33 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_MeshDeform) { | 
					
						
							|  |  |  |       MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  |       int size = mmd->dyngridsize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, MDefInfluence, mmd->totinfluence, mmd->bindinfluences); | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(int) * (mmd->totvert + 1), mmd->bindoffsets); | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(float) * 3 * mmd->totcagevert, mmd->bindcagecos); | 
					
						
							|  |  |  |       writestruct(wd, DATA, MDefCell, size * size * size, mmd->dyngrid); | 
					
						
							|  |  |  |       writestruct(wd, DATA, MDefInfluence, mmd->totinfluence, mmd->dyninfluences); | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(int) * mmd->totvert, mmd->dynverts); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_Warp) { | 
					
						
							|  |  |  |       WarpModifierData *tmd = (WarpModifierData *)md; | 
					
						
							|  |  |  |       if (tmd->curfalloff) { | 
					
						
							|  |  |  |         write_curvemapping(wd, tmd->curfalloff); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_WeightVGEdit) { | 
					
						
							|  |  |  |       WeightVGEditModifierData *wmd = (WeightVGEditModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (wmd->cmap_curve) { | 
					
						
							|  |  |  |         write_curvemapping(wd, wmd->cmap_curve); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_LaplacianDeform) { | 
					
						
							|  |  |  |       LaplacianDeformModifierData *lmd = (LaplacianDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(float) * lmd->total_verts * 3, lmd->vertexco); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_CorrectiveSmooth) { | 
					
						
							|  |  |  |       CorrectiveSmoothModifierData *csmd = (CorrectiveSmoothModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (csmd->bind_coords) { | 
					
						
							|  |  |  |         writedata(wd, DATA, sizeof(float[3]) * csmd->bind_coords_num, csmd->bind_coords); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eModifierType_SurfaceDeform) { | 
					
						
							|  |  |  |       SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, SDefVert, smd->numverts, smd->verts); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (smd->verts) { | 
					
						
							|  |  |  |         for (int i = 0; i < smd->numverts; i++) { | 
					
						
							|  |  |  |           writestruct(wd, DATA, SDefBind, smd->verts[i].numbinds, smd->verts[i].binds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (smd->verts[i].binds) { | 
					
						
							|  |  |  |             for (int j = 0; j < smd->verts[i].numbinds; j++) { | 
					
						
							|  |  |  |               writedata(wd, | 
					
						
							|  |  |  |                         DATA, | 
					
						
							|  |  |  |                         sizeof(int) * smd->verts[i].binds[j].numverts, | 
					
						
							|  |  |  |                         smd->verts[i].binds[j].vert_inds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               if (smd->verts[i].binds[j].mode == MOD_SDEF_MODE_CENTROID || | 
					
						
							|  |  |  |                   smd->verts[i].binds[j].mode == MOD_SDEF_MODE_LOOPTRI) { | 
					
						
							|  |  |  |                 writedata(wd, DATA, sizeof(float) * 3, smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               else { | 
					
						
							|  |  |  |                 writedata(wd, | 
					
						
							|  |  |  |                           DATA, | 
					
						
							|  |  |  |                           sizeof(float) * smd->verts[i].binds[j].numverts, | 
					
						
							|  |  |  |                           smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-11-20 16:12:32 -05:00
										 |  |  |     else if (md->type == eModifierType_Bevel) { | 
					
						
							|  |  |  |       BevelModifierData *bmd = (BevelModifierData *)md; | 
					
						
							|  |  |  |       if (bmd->custom_profile) { | 
					
						
							|  |  |  |         write_CurveProfile(wd, bmd->custom_profile); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-07-19 20:14:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | static void write_gpencil_modifiers(WriteData *wd, ListBase *modbase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GpencilModifierData *md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (modbase == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (md = modbase->first; md; md = md->next) { | 
					
						
							| 
									
										
										
										
											2020-05-08 10:32:30 +02:00
										 |  |  |     const GpencilModifierTypeInfo *mti = BKE_gpencil_modifier_get_info(md->type); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (mti == NULL) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct_id(wd, DATA, mti->struct_name, 1, md); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (md->type == eGpencilModifierType_Thick) { | 
					
						
							|  |  |  |       ThickGpencilModifierData *gpmd = (ThickGpencilModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (gpmd->curve_thickness) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_thickness); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |     else if (md->type == eGpencilModifierType_Noise) { | 
					
						
							|  |  |  |       NoiseGpencilModifierData *gpmd = (NoiseGpencilModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (gpmd->curve_intensity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_intensity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else if (md->type == eGpencilModifierType_Hook) { | 
					
						
							|  |  |  |       HookGpencilModifierData *gpmd = (HookGpencilModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (gpmd->curfalloff) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curfalloff); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-13 10:28:30 +01:00
										 |  |  |     else if (md->type == eGpencilModifierType_Tint) { | 
					
						
							|  |  |  |       TintGpencilModifierData *gpmd = (TintGpencilModifierData *)md; | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |       if (gpmd->colorband) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, gpmd->colorband); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (gpmd->curve_intensity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_intensity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eGpencilModifierType_Smooth) { | 
					
						
							|  |  |  |       SmoothGpencilModifierData *gpmd = (SmoothGpencilModifierData *)md; | 
					
						
							|  |  |  |       if (gpmd->curve_intensity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_intensity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eGpencilModifierType_Color) { | 
					
						
							|  |  |  |       ColorGpencilModifierData *gpmd = (ColorGpencilModifierData *)md; | 
					
						
							|  |  |  |       if (gpmd->curve_intensity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_intensity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (md->type == eGpencilModifierType_Opacity) { | 
					
						
							|  |  |  |       OpacityGpencilModifierData *gpmd = (OpacityGpencilModifierData *)md; | 
					
						
							|  |  |  |       if (gpmd->curve_intensity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, gpmd->curve_intensity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_shaderfxs(WriteData *wd, ListBase *fxbase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ShaderFxData *fx; | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (fxbase == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (fx = fxbase->first; fx; fx = fx->next) { | 
					
						
							| 
									
										
										
										
											2020-05-08 10:32:30 +02:00
										 |  |  |     const ShaderFxTypeInfo *fxi = BKE_shaderfx_get_info(fx->type); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (fxi == NULL) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct_id(wd, DATA, fxi->struct_name, 1, fx); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-31 10:22:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_object(WriteData *wd, Object *ob, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ob->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 10:04:08 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:58:37 +02:00
										 |  |  |     BKE_object_runtime_reset(ob); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_OB, Object, 1, id_address, ob); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &ob->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ob->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, ob->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * ob->totcol, ob->mat); | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(char) * ob->totcol, ob->matbits); | 
					
						
							|  |  |  |     /* write_effects(wd, &ob->effect); */ /* not used anymore */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ob->type == OB_ARMATURE) { | 
					
						
							|  |  |  |       bArmature *arm = ob->data; | 
					
						
							|  |  |  |       if (arm && ob->pose && arm->act_bone) { | 
					
						
							|  |  |  |         BLI_strncpy( | 
					
						
							|  |  |  |             ob->pose->proxy_act_bone, arm->act_bone->name, sizeof(ob->pose->proxy_act_bone)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_pose(wd, ob->pose); | 
					
						
							|  |  |  |     write_defgroups(wd, &ob->defbase); | 
					
						
							|  |  |  |     write_fmaps(wd, &ob->fmaps); | 
					
						
							|  |  |  |     write_constraints(wd, &ob->constraints); | 
					
						
							|  |  |  |     write_motionpath(wd, ob->mpath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, PartDeflect, 1, ob->pd); | 
					
						
							|  |  |  |     if (ob->soft) { | 
					
						
							|  |  |  |       /* Set deprecated pointers to prevent crashes of older Blenders */ | 
					
						
							|  |  |  |       ob->soft->pointcache = ob->soft->shared->pointcache; | 
					
						
							|  |  |  |       ob->soft->ptcaches = ob->soft->shared->ptcaches; | 
					
						
							|  |  |  |       writestruct(wd, DATA, SoftBody, 1, ob->soft); | 
					
						
							|  |  |  |       writestruct(wd, DATA, SoftBody_Shared, 1, ob->soft->shared); | 
					
						
							|  |  |  |       write_pointcaches(wd, &(ob->soft->shared->ptcaches)); | 
					
						
							|  |  |  |       writestruct(wd, DATA, EffectorWeights, 1, ob->soft->effector_weights); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ob->rigidbody_object) { | 
					
						
							|  |  |  |       /* TODO: if any extra data is added to handle duplis, will need separate function then */ | 
					
						
							|  |  |  |       writestruct(wd, DATA, RigidBodyOb, 1, ob->rigidbody_object); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ob->rigidbody_constraint) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, RigidBodyCon, 1, ob->rigidbody_constraint); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ob->type == OB_EMPTY && ob->empty_drawtype == OB_EMPTY_IMAGE) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ImageUser, 1, ob->iuser); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_particlesystems(wd, &ob->particlesystem); | 
					
						
							|  |  |  |     write_modifiers(wd, &ob->modifiers); | 
					
						
							|  |  |  |     write_gpencil_modifiers(wd, &ob->greasepencil_modifiers); | 
					
						
							|  |  |  |     write_shaderfxs(wd, &ob->shader_fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writelist(wd, DATA, LinkData, &ob->pc_ids); | 
					
						
							|  |  |  |     writelist(wd, DATA, LodLevel, &ob->lodlevels); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, ob->preview); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_vfont(WriteData *wd, VFont *vf, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (vf->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 10:18:39 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     vf->data = NULL; | 
					
						
							|  |  |  |     vf->temp_pf = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_VF, VFont, 1, id_address, vf); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &vf->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     if (vf->packedfile) { | 
					
						
							|  |  |  |       PackedFile *pf = vf->packedfile; | 
					
						
							|  |  |  |       writestruct(wd, DATA, PackedFile, 1, pf); | 
					
						
							|  |  |  |       writedata(wd, DATA, pf->size, pf->data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_key(WriteData *wd, Key *key, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (key->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_KE, Key, 1, id_address, key); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &key->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (key->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, key->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (KeyBlock *, kb, &key->block) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, KeyBlock, 1, kb); | 
					
						
							|  |  |  |       if (kb->data) { | 
					
						
							|  |  |  |         writedata(wd, DATA, kb->totelem * key->elemsize, kb->data); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_camera(WriteData *wd, Camera *cam, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (cam->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_CA, Camera, 1, id_address, cam); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &cam->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cam->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, cam->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (CameraBGImage *, bgpic, &cam->bg_images) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, CameraBGImage, 1, bgpic); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_mball(WriteData *wd, MetaBall *mb, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (mb->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 10:12:19 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     BLI_listbase_clear(&mb->disp); | 
					
						
							|  |  |  |     mb->editelems = NULL; | 
					
						
							|  |  |  |     /* Must always be cleared (meta's don't have their own edit-data). */ | 
					
						
							|  |  |  |     mb->needs_flush_to_id = 0; | 
					
						
							|  |  |  |     mb->lastelem = NULL; | 
					
						
							|  |  |  |     mb->batch_cache = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_MB, MetaBall, 1, id_address, mb); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &mb->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * mb->totcol, mb->mat); | 
					
						
							|  |  |  |     if (mb->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, mb->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (MetaElem *, ml, &mb->elems) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, MetaElem, 1, ml); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_curve(WriteData *wd, Curve *cu, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (cu->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 10:04:08 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:56:29 +02:00
										 |  |  |     cu->editnurb = NULL; | 
					
						
							|  |  |  |     cu->editfont = NULL; | 
					
						
							|  |  |  |     cu->batch_cache = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_CU, Curve, 1, id_address, cu); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &cu->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * cu->totcol, cu->mat); | 
					
						
							|  |  |  |     if (cu->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, cu->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cu->vfont) { | 
					
						
							|  |  |  |       writedata(wd, DATA, cu->len + 1, cu->str); | 
					
						
							|  |  |  |       writestruct(wd, DATA, CharInfo, cu->len_wchar + 1, cu->strinfo); | 
					
						
							|  |  |  |       writestruct(wd, DATA, TextBox, cu->totbox, cu->tb); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       /* is also the order of reading */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |       LISTBASE_FOREACH (Nurb *, nu, &cu->nurb) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         writestruct(wd, DATA, Nurb, 1, nu); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |       LISTBASE_FOREACH (Nurb *, nu, &cu->nurb) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         if (nu->type == CU_BEZIER) { | 
					
						
							|  |  |  |           writestruct(wd, DATA, BezTriple, nu->pntsu, nu->bezt); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           writestruct(wd, DATA, BPoint, nu->pntsu * nu->pntsv, nu->bp); | 
					
						
							|  |  |  |           if (nu->knotsu) { | 
					
						
							|  |  |  |             writedata(wd, DATA, KNOTSU(nu) * sizeof(float), nu->knotsu); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (nu->knotsv) { | 
					
						
							|  |  |  |             writedata(wd, DATA, KNOTSV(nu) * sizeof(float), nu->knotsv); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_dverts(WriteData *wd, int count, MDeformVert *dvlist) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (dvlist) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write the dvert list */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, MDeformVert, count, dvlist); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write deformation data for each dvert */ | 
					
						
							|  |  |  |     for (int i = 0; i < count; i++) { | 
					
						
							|  |  |  |       if (dvlist[i].dw) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, MDeformWeight, dvlist[i].totweight, dvlist[i].dw); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 14:27:50 +00:00
										 |  |  | static void write_mdisps(WriteData *wd, int count, MDisps *mdlist, int external) | 
					
						
							| 
									
										
										
										
											2009-01-06 18:59:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (mdlist) { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, MDisps, count, mdlist); | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (i = 0; i < count; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       MDisps *md = &mdlist[i]; | 
					
						
							|  |  |  |       if (md->disps) { | 
					
						
							|  |  |  |         if (!external) { | 
					
						
							|  |  |  |           writedata(wd, DATA, sizeof(float) * 3 * md->totdisp, md->disps); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (md->hidden) { | 
					
						
							|  |  |  |         writedata(wd, DATA, BLI_BITMAP_SIZE(md->totdisp), md->hidden); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-01-06 18:59:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:24 +00:00
										 |  |  | static void write_grid_paint_mask(WriteData *wd, int count, GridPaintMask *grid_paint_mask) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (grid_paint_mask) { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, GridPaintMask, count, grid_paint_mask); | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (i = 0; i < count; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       GridPaintMask *gpm = &grid_paint_mask[i]; | 
					
						
							|  |  |  |       if (gpm->data) { | 
					
						
							|  |  |  |         const int gridsize = BKE_ccg_gridsize(gpm->level); | 
					
						
							|  |  |  |         writedata(wd, DATA, sizeof(*gpm->data) * gridsize * gridsize, gpm->data); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void write_customdata(WriteData *wd, | 
					
						
							|  |  |  |                              ID *id, | 
					
						
							|  |  |  |                              int count, | 
					
						
							|  |  |  |                              CustomData *data, | 
					
						
							|  |  |  |                              CustomDataLayer *layers, | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |                              CustomDataMask cddata_mask) | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* write external customdata (not for undo) */ | 
					
						
							|  |  |  |   if (data->external && (wd->use_memfile == false)) { | 
					
						
							|  |  |  |     CustomData_external_write(data, id, cddata_mask, count, 0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   writestruct_at_address(wd, DATA, CustomDataLayer, data->totlayer, data->layers, layers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < data->totlayer; i++) { | 
					
						
							|  |  |  |     CustomDataLayer *layer = &layers[i]; | 
					
						
							|  |  |  |     const char *structname; | 
					
						
							|  |  |  |     int structnum, datasize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (layer->type == CD_MDEFORMVERT) { | 
					
						
							|  |  |  |       /* layer types that allocate own memory need special handling */ | 
					
						
							|  |  |  |       write_dverts(wd, count, layer->data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (layer->type == CD_MDISPS) { | 
					
						
							|  |  |  |       write_mdisps(wd, count, layer->data, layer->flag & CD_FLAG_EXTERNAL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (layer->type == CD_PAINT_MASK) { | 
					
						
							|  |  |  |       const float *layer_data = layer->data; | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(*layer_data) * count, layer_data); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-05 14:53:23 +01:00
										 |  |  |     else if (layer->type == CD_SCULPT_FACE_SETS) { | 
					
						
							|  |  |  |       const float *layer_data = layer->data; | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(*layer_data) * count, layer_data); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else if (layer->type == CD_GRID_PAINT_MASK) { | 
					
						
							|  |  |  |       write_grid_paint_mask(wd, count, layer->data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (layer->type == CD_FACEMAP) { | 
					
						
							|  |  |  |       const int *layer_data = layer->data; | 
					
						
							|  |  |  |       writedata(wd, DATA, sizeof(*layer_data) * count, layer_data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       CustomData_file_write_info(layer->type, &structname, &structnum); | 
					
						
							|  |  |  |       if (structnum) { | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |         datasize = structnum * count; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         writestruct_id(wd, DATA, structname, datasize, layer->data); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-04-22 16:14:35 +02:00
										 |  |  |       else if (!wd->use_memfile) { /* Do not warn on undo. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         printf("%s error: layer '%s':%d - can't be written to file\n", | 
					
						
							|  |  |  |                __func__, | 
					
						
							|  |  |  |                structname, | 
					
						
							|  |  |  |                layer->type); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (data->external) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, CustomDataExternal, 1, data->external); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_mesh(WriteData *wd, Mesh *mesh, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (mesh->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     /* cache only - don't write */ | 
					
						
							|  |  |  |     mesh->mface = NULL; | 
					
						
							|  |  |  |     mesh->totface = 0; | 
					
						
							|  |  |  |     memset(&mesh->fdata, 0, sizeof(mesh->fdata)); | 
					
						
							| 
									
										
										
										
											2020-03-28 21:50:56 +01:00
										 |  |  |     memset(&mesh->runtime, 0, sizeof(mesh->runtime)); | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Reduce xdata layers, fill xlayers with layers to be written.
 | 
					
						
							|  |  |  |      * This makes xdata invalid for Blender, which is why we made a | 
					
						
							|  |  |  |      * temporary local copy. */ | 
					
						
							|  |  |  |     CustomDataLayer *vlayers = NULL, vlayers_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomDataLayer *elayers = NULL, elayers_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomDataLayer *flayers = NULL, flayers_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomDataLayer *llayers = NULL, llayers_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomDataLayer *players = NULL, players_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&mesh->vdata, &vlayers, vlayers_buff, ARRAY_SIZE(vlayers_buff)); | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&mesh->edata, &elayers, elayers_buff, ARRAY_SIZE(elayers_buff)); | 
					
						
							|  |  |  |     flayers = flayers_buff; | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&mesh->ldata, &llayers, llayers_buff, ARRAY_SIZE(llayers_buff)); | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&mesh->pdata, &players, players_buff, ARRAY_SIZE(players_buff)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_ME, Mesh, 1, id_address, mesh); | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     write_iddata(wd, &mesh->id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     /* direct data */ | 
					
						
							|  |  |  |     if (mesh->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, mesh->adt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     writedata(wd, DATA, sizeof(void *) * mesh->totcol, mesh->mat); | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(MSelect) * mesh->totselect, mesh->mselect); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     write_customdata(wd, &mesh->id, mesh->totvert, &mesh->vdata, vlayers, CD_MASK_MESH.vmask); | 
					
						
							|  |  |  |     write_customdata(wd, &mesh->id, mesh->totedge, &mesh->edata, elayers, CD_MASK_MESH.emask); | 
					
						
							|  |  |  |     /* fdata is really a dummy - written so slots align */ | 
					
						
							|  |  |  |     write_customdata(wd, &mesh->id, mesh->totface, &mesh->fdata, flayers, CD_MASK_MESH.fmask); | 
					
						
							|  |  |  |     write_customdata(wd, &mesh->id, mesh->totloop, &mesh->ldata, llayers, CD_MASK_MESH.lmask); | 
					
						
							|  |  |  |     write_customdata(wd, &mesh->id, mesh->totpoly, &mesh->pdata, players, CD_MASK_MESH.pmask); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-22 18:07:39 +01:00
										 |  |  |     /* free temporary data */ | 
					
						
							|  |  |  |     if (vlayers && vlayers != vlayers_buff) { | 
					
						
							|  |  |  |       MEM_freeN(vlayers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (elayers && elayers != elayers_buff) { | 
					
						
							|  |  |  |       MEM_freeN(elayers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (flayers && flayers != flayers_buff) { | 
					
						
							|  |  |  |       MEM_freeN(flayers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (llayers && llayers != llayers_buff) { | 
					
						
							|  |  |  |       MEM_freeN(llayers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (players && players != players_buff) { | 
					
						
							|  |  |  |       MEM_freeN(players); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_lattice(WriteData *wd, Lattice *lt, const void *id_address) | 
					
						
							| 
									
										
										
										
											2006-09-03 12:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (lt->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:28:41 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     lt->editlatt = NULL; | 
					
						
							|  |  |  |     lt->batch_cache = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_LT, Lattice, 1, id_address, lt); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, <->id); | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write animdata */ | 
					
						
							|  |  |  |     if (lt->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, lt->adt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* direct data */ | 
					
						
							|  |  |  |     writestruct(wd, DATA, BPoint, lt->pntsu * lt->pntsv * lt->pntsw, lt->def); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_dverts(wd, lt->pntsu * lt->pntsv * lt->pntsw, lt->dvert); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2006-09-03 12:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_image(WriteData *wd, Image *ima, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ima->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     ImagePackedFile *imapf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Some trickery to keep forward compatibility of packed images. */ | 
					
						
							|  |  |  |     BLI_assert(ima->packedfile == NULL); | 
					
						
							|  |  |  |     if (ima->packedfiles.first != NULL) { | 
					
						
							|  |  |  |       imapf = ima->packedfiles.first; | 
					
						
							|  |  |  |       ima->packedfile = imapf->packedfile; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_IM, Image, 1, id_address, ima); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &ima->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (imapf = ima->packedfiles.first; imapf; imapf = imapf->next) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ImagePackedFile, 1, imapf); | 
					
						
							|  |  |  |       if (imapf->packedfile) { | 
					
						
							|  |  |  |         PackedFile *pf = imapf->packedfile; | 
					
						
							|  |  |  |         writestruct(wd, DATA, PackedFile, 1, pf); | 
					
						
							|  |  |  |         writedata(wd, DATA, pf->size, pf->data); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, ima->preview); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (ImageView *, iv, &ima->views) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, ImageView, 1, iv); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct(wd, DATA, Stereo3dFormat, 1, ima->stereo3d_format); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Add support for tiled images and the UDIM naming scheme
This patch contains the work that I did during my week at the Code Quest - adding support for tiled images to Blender.
With this patch, images now contain a list of tiles. By default, this just contains one tile, but if the source type is set to Tiled, the user can add additional tiles. When acquiring an ImBuf, the tile to be loaded is specified in the ImageUser.
Therefore, code that is not yet aware of tiles will just access the default tile as usual.
The filenames of the additional tiles are derived from the original filename according to the UDIM naming scheme - the filename contains an index that is calculated as (1001 + 10*<y coordinate of the tile> + <x coordinate of the tile>), where the x coordinate never goes above 9.
Internally, the various tiles are stored in a cache just like sequences. When acquired for the first time, the code will try to load the corresponding file from disk. Alternatively, a new operator can be used to initialize the tile similar to the New Image operator.
The following features are supported so far:
- Automatic detection and loading of all tiles when opening the first tile (1001)
- Saving all tiles
- Adding and removing tiles
- Filling tiles with generated images
- Drawing all tiles in the Image Editor
- Viewing a tiled grid even if no image is selected
- Rendering tiled images in Eevee
- Rendering tiled images in Cycles (in SVM mode)
- Automatically skipping loading of unused tiles in Cycles
- 2D texture painting (also across tiles)
- 3D texture painting (also across tiles, only limitation: individual faces can not cross tile borders)
- Assigning custom labels to individual tiles (drawn in the Image Editor instead of the ID)
- Different resolutions between tiles
There still are some missing features that will be added later (see T72390):
- Workbench engine support
- Packing/Unpacking support
- Baking support
- Cycles OSL support
- many other Blender features that rely on images
Thanks to Brecht for the review and to all who tested the intermediate versions!
Differential Revision: https://developer.blender.org/D3509
											
										 
											2019-12-12 16:06:08 +01:00
										 |  |  |     writelist(wd, DATA, ImageTile, &ima->tiles); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     ima->packedfile = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writelist(wd, DATA, RenderSlot, &ima->renderslots); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_texture(WriteData *wd, Tex *tex, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (tex->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_TE, Tex, 1, id_address, tex); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &tex->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (tex->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, tex->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     if (tex->coba) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ColorBand, 1, tex->coba); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* nodetree is integral part of texture, no libdata */ | 
					
						
							|  |  |  |     if (tex->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, tex->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, tex->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, tex->preview); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_material(WriteData *wd, Material *ma, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ma->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 10:15:13 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     ma->texpaintslot = NULL; | 
					
						
							|  |  |  |     BLI_listbase_clear(&ma->gpumaterial); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_MA, Material, 1, id_address, ma); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &ma->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ma->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, ma->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* nodetree is integral part of material, no libdata */ | 
					
						
							|  |  |  |     if (ma->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, ma->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, ma->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, ma->preview); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* grease pencil settings */ | 
					
						
							|  |  |  |     if (ma->gp_style) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, MaterialGPencilStyle, 1, ma->gp_style); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_world(WriteData *wd, World *wrld, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wrld->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:30:30 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     BLI_listbase_clear(&wrld->gpumaterial); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_WO, World, 1, id_address, wrld); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &wrld->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (wrld->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, wrld->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* nodetree is integral part of world, no libdata */ | 
					
						
							|  |  |  |     if (wrld->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, wrld->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, wrld->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, wrld->preview); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_light(WriteData *wd, Light *la, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (la->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_LA, Light, 1, id_address, la); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &la->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (la->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, la->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (la->curfalloff) { | 
					
						
							|  |  |  |       write_curvemapping(wd, la->curfalloff); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Node-tree is integral part of lights, no libdata. */ | 
					
						
							|  |  |  |     if (la->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, la->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, la->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_previews(wd, la->preview); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | static void write_collection_nolib(WriteData *wd, Collection *collection) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-12 09:04:10 +10:00
										 |  |  |   /* Shared function for collection data-blocks and scene master collection. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_previews(wd, collection->preview); | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (CollectionObject *, cob, &collection->gobject) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, CollectionObject, 1, cob); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (CollectionChild *, child, &collection->children) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, CollectionChild, 1, child); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_collection(WriteData *wd, Collection *collection, const void *id_address) | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (collection->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:50:42 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     collection->flag &= ~COLLECTION_HAS_OBJECT_CACHE; | 
					
						
							|  |  |  |     collection->tag = 0; | 
					
						
							|  |  |  |     BLI_listbase_clear(&collection->object_cache); | 
					
						
							|  |  |  |     BLI_listbase_clear(&collection->parents); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_GR, Collection, 1, id_address, collection); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &collection->id); | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_collection_nolib(wd, collection); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Collections and groups unification
OVERVIEW
* In 2.7 terminology, all layers and groups are now collection datablocks.
* These collections are nestable, linkable, instanceable, overrideable, ..
  which opens up new ways to set up scenes and link + override data.
* Viewport/render visibility and selectability are now a part of the collection
  and shared across all view layers and linkable.
* View layers define which subset of the scene collection hierarchy is excluded
  for each. For many workflows one view layer can be used, these are more of an
  advanced feature now.
OUTLINER
* The outliner now has a "View Layer" display mode instead of "Collections",
  which can display the collections and/or objects in the view layer.
* In this display mode, collections can be excluded with the right click menu.
  These will then be greyed out and their objects will be excluded.
* To view collections not linked to any scene, the "Blender File" display mode
  can be used, with the new filtering option to just see Colleciton datablocks.
* The outliner right click menus for collections and objects were reorganized.
* Drag and drop still needs to be improved. Like before, dragging the icon or
  text gives different results, we'll unify this later.
LINKING AND OVERRIDES
* Collections can now be linked into the scene without creating an instance,
  with the link/append operator or from the collections view in the outliner.
* Collections can get static overrides with the right click menu in the outliner,
  but this is rather unreliable and not clearly communicated at the moment.
* We still need to improve the make override operator to turn collection instances
  into collections with overrides directly in the scene.
PERFORMANCE
* We tried to make performance not worse than before and improve it in some
  cases. The main thing that's still a bit slower is multiple scenes, we have to
  change the layer syncing to only updated affected scenes.
* Collections keep a list of their parent collections for faster incremental
  updates in syncing and caching.
* View layer bases are now in a object -> base hash to avoid quadratic time
  lookups internally and in API functions like visible_get().
VERSIONING
* Compatibility with 2.7 files should be improved due to the new visibility
  controls. Of course users may not want to set up their scenes differently
  now to avoid having separate layers and groups.
* Compatibility with 2.8 is mostly there, and was tested on Eevee demo and Hero
  files. There's a few things which are know to be not quite compatible, like
  nested layer collections inside groups.
* The versioning code for 2.8 files is quite complicated, and isolated behind
  #ifdef so it can be removed at the end of the release cycle.
KNOWN ISSUES
* The G-key group operators in the 3D viewport were left mostly as is, they
  need to be modified still to fit better.
* Same for the groups panel in the object properties. This needs to be updated
  still, or perhaps replaced by something better.
* Collections must all have a unique name. Less restrictive namespacing is to
  be done later, we'll have to see how important this is as all objects within
  the collections must also have a unique name anyway.
* Full scene copy and delete scene are exactly doing the right thing yet.
Differential Revision: https://developer.blender.org/D3383
https://code.blender.org/2018/05/collections-and-groups/
											
										 
											2018-04-30 15:57:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | static void write_sequence_modifiers(WriteData *wd, ListBase *modbase) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   SequenceModifierData *smd; | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (smd = modbase->first; smd; smd = smd->next) { | 
					
						
							|  |  |  |     const SequenceModifierTypeInfo *smti = BKE_sequence_modifier_type_info_get(smd->type); | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (smti) { | 
					
						
							|  |  |  |       writestruct_id(wd, DATA, smti->struct_name, 1, smd); | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (smd->type == seqModifierType_Curves) { | 
					
						
							|  |  |  |         CurvesModifierData *cmd = (CurvesModifierData *)smd; | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         write_curvemapping(wd, &cmd->curve_mapping); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (smd->type == seqModifierType_HueCorrect) { | 
					
						
							|  |  |  |         HueCorrectModifierData *hcmd = (HueCorrectModifierData *)smd; | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         write_curvemapping(wd, &hcmd->curve_mapping); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SequenceModifierData, 1, smd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-08-19 15:41:56 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Color Management, Stage 2: Switch color pipeline to use OpenColorIO
Replace old color pipeline which was supporting linear/sRGB color spaces
only with OpenColorIO-based pipeline.
This introduces two configurable color spaces:
- Input color space for images and movie clips. This space is used to convert
  images/movies from color space in which file is saved to Blender's linear
  space (for float images, byte images are not internally converted, only input
  space is stored for such images and used later).
  This setting could be found in image/clip data block settings.
- Display color space which defines space in which particular display is working.
  This settings could be found in scene's Color Management panel.
When render result is being displayed on the screen, apart from converting image
to display space, some additional conversions could happen.
This conversions are:
- View, which defines tone curve applying before display transformation.
  These are different ways to view the image on the same display device.
  For example it could be used to emulate film view on sRGB display.
- Exposure affects on image exposure before tone map is applied.
- Gamma is post-display gamma correction, could be used to match particular
  display gamma.
- RGB curves are user-defined curves which are applying before display
  transformation, could be used for different purposes.
All this settings by default are only applying on render result and does not
affect on other images. If some particular image needs to be affected by this
transformation, "View as Render" setting of image data block should be set to
truth. Movie clips are always affected by all display transformations.
This commit also introduces configurable color space in which sequencer is
working. This setting could be found in scene's Color Management panel and
it should be used if such stuff as grading needs to be done in color space
different from sRGB (i.e. when Film view on sRGB display is use, using VD16
space as sequencer's internal space would make grading working in space
which is close to the space using for display).
Some technical notes:
- Image buffer's float buffer is now always in linear space, even if it was
  created from 16bit byte images.
- Space of byte buffer is stored in image buffer's rect_colorspace property.
- Profile of image buffer was removed since it's not longer meaningful.
- OpenGL and GLSL is supposed to always work in sRGB space. It is possible
  to support other spaces, but it's quite large project which isn't so
  much important.
- Legacy Color Management option disabled is emulated by using None display.
  It could have some regressions, but there's no clear way to avoid them.
- If OpenColorIO is disabled on build time, it should make blender behaving
  in the same way as previous release with color management enabled.
More details could be found at this page (more details would be added soon):
http://wiki.blender.org/index.php/Dev:Ref/Release_Notes/2.64/Color_Management
--
Thanks to Xavier Thomas, Lukas Toene for initial work on OpenColorIO
integration and to Brecht van Lommel for some further development and code/
usecase review!
											
										 
											2012-09-15 10:05:07 +00:00
										 |  |  | static void write_view_settings(WriteData *wd, ColorManagedViewSettings *view_settings) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (view_settings->curve_mapping) { | 
					
						
							|  |  |  |     write_curvemapping(wd, view_settings->curve_mapping); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Color Management, Stage 2: Switch color pipeline to use OpenColorIO
Replace old color pipeline which was supporting linear/sRGB color spaces
only with OpenColorIO-based pipeline.
This introduces two configurable color spaces:
- Input color space for images and movie clips. This space is used to convert
  images/movies from color space in which file is saved to Blender's linear
  space (for float images, byte images are not internally converted, only input
  space is stored for such images and used later).
  This setting could be found in image/clip data block settings.
- Display color space which defines space in which particular display is working.
  This settings could be found in scene's Color Management panel.
When render result is being displayed on the screen, apart from converting image
to display space, some additional conversions could happen.
This conversions are:
- View, which defines tone curve applying before display transformation.
  These are different ways to view the image on the same display device.
  For example it could be used to emulate film view on sRGB display.
- Exposure affects on image exposure before tone map is applied.
- Gamma is post-display gamma correction, could be used to match particular
  display gamma.
- RGB curves are user-defined curves which are applying before display
  transformation, could be used for different purposes.
All this settings by default are only applying on render result and does not
affect on other images. If some particular image needs to be affected by this
transformation, "View as Render" setting of image data block should be set to
truth. Movie clips are always affected by all display transformations.
This commit also introduces configurable color space in which sequencer is
working. This setting could be found in scene's Color Management panel and
it should be used if such stuff as grading needs to be done in color space
different from sRGB (i.e. when Film view on sRGB display is use, using VD16
space as sequencer's internal space would make grading working in space
which is close to the space using for display).
Some technical notes:
- Image buffer's float buffer is now always in linear space, even if it was
  created from 16bit byte images.
- Space of byte buffer is stored in image buffer's rect_colorspace property.
- Profile of image buffer was removed since it's not longer meaningful.
- OpenGL and GLSL is supposed to always work in sRGB space. It is possible
  to support other spaces, but it's quite large project which isn't so
  much important.
- Legacy Color Management option disabled is emulated by using None display.
  It could have some regressions, but there's no clear way to avoid them.
- If OpenColorIO is disabled on build time, it should make blender behaving
  in the same way as previous release with color management enabled.
More details could be found at this page (more details would be added soon):
http://wiki.blender.org/index.php/Dev:Ref/Release_Notes/2.64/Color_Management
--
Thanks to Xavier Thomas, Lukas Toene for initial work on OpenColorIO
integration and to Brecht van Lommel for some further development and code/
usecase review!
											
										 
											2012-09-15 10:05:07 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-05 12:47:20 +02:00
										 |  |  | static void write_view3dshading(WriteData *wd, View3DShading *shading) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (shading->prop) { | 
					
						
							|  |  |  |     IDP_WriteProperty(shading->prop, wd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-11 15:07:04 +01:00
										 |  |  | static void write_paint(WriteData *wd, Paint *p) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (p->cavity_curve) { | 
					
						
							|  |  |  |     write_curvemapping(wd, p->cavity_curve); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-06 18:02:22 +02:00
										 |  |  |   writestruct(wd, DATA, PaintToolSlot, p->tool_slots_len, p->tool_slots); | 
					
						
							| 
									
										
										
										
											2015-02-11 15:07:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
  Option to add objects based on their names
* Collection Manager operators
  The existing buttons  are placeholders
* Collection Manager drawing
  The editor main region is empty
* Collection Override
* Per-Collection engine settings
  This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
  We are replacing Base by a new extended Base while keeping it backward
  compatible with some legacy settings (i.e., lay, flag_legacy).
  Renamed all Base to BaseLegacy to make it clear the areas of code that
  still need to be converted
  Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
  - read, write, add/copy/remove objects, copy scene, collection
  link/unlinking, context)
* New Editor: Collection Manager
  Based on patch by Julian Eisel
  This is extracted from the layer-manager branch. With the following changes:
    - Renamed references of layer manager to collections manager
    - I doesn't include the editors/space_collections/ draw and util files
    - The drawing code itself will be implemented separately by Julian
* Base / Object:
  A little note about them. Original Blender code would try to keep them
  in sync through the code, juggling flags back and forth. This will now
  be handled by Depsgraph, keeping Object and Bases more separated
  throughout the non-rendering code.
  Scene.base is being cleared in doversion, and the old viewport drawing
  code was poorly converted to use the new bases while the new viewport
  code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
											
										 
											2017-02-07 10:18:38 +01:00
										 |  |  | static void write_layer_collections(WriteData *wd, ListBase *lb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (LayerCollection *, lc, lb) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, LayerCollection, 1, lc); | 
					
						
							| 
									
										
											  
											
												Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
  Option to add objects based on their names
* Collection Manager operators
  The existing buttons  are placeholders
* Collection Manager drawing
  The editor main region is empty
* Collection Override
* Per-Collection engine settings
  This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
  We are replacing Base by a new extended Base while keeping it backward
  compatible with some legacy settings (i.e., lay, flag_legacy).
  Renamed all Base to BaseLegacy to make it clear the areas of code that
  still need to be converted
  Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
  - read, write, add/copy/remove objects, copy scene, collection
  link/unlinking, context)
* New Editor: Collection Manager
  Based on patch by Julian Eisel
  This is extracted from the layer-manager branch. With the following changes:
    - Renamed references of layer manager to collections manager
    - I doesn't include the editors/space_collections/ draw and util files
    - The drawing code itself will be implemented separately by Julian
* Base / Object:
  A little note about them. Original Blender code would try to keep them
  in sync through the code, juggling flags back and forth. This will now
  be handled by Depsgraph, keeping Object and Bases more separated
  throughout the non-rendering code.
  Scene.base is being cleared in doversion, and the old viewport drawing
  code was poorly converted to use the new bases while the new viewport
  code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
											
										 
											2017-02-07 10:18:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_layer_collections(wd, &lc->layer_collections); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
  Option to add objects based on their names
* Collection Manager operators
  The existing buttons  are placeholders
* Collection Manager drawing
  The editor main region is empty
* Collection Override
* Per-Collection engine settings
  This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
  We are replacing Base by a new extended Base while keeping it backward
  compatible with some legacy settings (i.e., lay, flag_legacy).
  Renamed all Base to BaseLegacy to make it clear the areas of code that
  still need to be converted
  Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
  - read, write, add/copy/remove objects, copy scene, collection
  link/unlinking, context)
* New Editor: Collection Manager
  Based on patch by Julian Eisel
  This is extracted from the layer-manager branch. With the following changes:
    - Renamed references of layer manager to collections manager
    - I doesn't include the editors/space_collections/ draw and util files
    - The drawing code itself will be implemented separately by Julian
* Base / Object:
  A little note about them. Original Blender code would try to keep them
  in sync through the code, juggling flags back and forth. This will now
  be handled by Depsgraph, keeping Object and Bases more separated
  throughout the non-rendering code.
  Scene.base is being cleared in doversion, and the old viewport drawing
  code was poorly converted to use the new bases while the new viewport
  code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
											
										 
											2017-02-07 10:18:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-01 11:24:21 -02:00
										 |  |  | static void write_view_layer(WriteData *wd, ViewLayer *view_layer) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, DATA, ViewLayer, 1, view_layer); | 
					
						
							|  |  |  |   writelist(wd, DATA, Base, &view_layer->object_bases); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (view_layer->id_properties) { | 
					
						
							|  |  |  |     IDP_WriteProperty(view_layer->id_properties, wd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (FreestyleModuleConfig *, fmc, &view_layer->freestyle_config.modules) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, FreestyleModuleConfig, 1, fmc); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (FreestyleLineSet *, fls, &view_layer->freestyle_config.linesets) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, FreestyleLineSet, 1, fls); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   write_layer_collections(wd, &view_layer->layer_collections); | 
					
						
							| 
									
										
										
										
											2017-12-01 11:24:21 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 15:02:25 +02:00
										 |  |  | static void write_lightcache_texture(WriteData *wd, LightCacheTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (tex->data) { | 
					
						
							|  |  |  |     size_t data_size = tex->components * tex->tex_size[0] * tex->tex_size[1] * tex->tex_size[2]; | 
					
						
							|  |  |  |     if (tex->data_type == LIGHTCACHETEX_FLOAT) { | 
					
						
							|  |  |  |       data_size *= sizeof(float); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (tex->data_type == LIGHTCACHETEX_UINT) { | 
					
						
							|  |  |  |       data_size *= sizeof(uint); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writedata(wd, DATA, data_size, tex->data); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-10 15:02:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_lightcache(WriteData *wd, LightCache *cache) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_lightcache_texture(wd, &cache->grid_tx); | 
					
						
							|  |  |  |   write_lightcache_texture(wd, &cache->cube_tx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (cache->cube_mips) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, LightCacheTexture, cache->mips_len, cache->cube_mips); | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (int i = 0; i < cache->mips_len; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_lightcache_texture(wd, &cache->cube_mips[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   writestruct(wd, DATA, LightGridCache, cache->grid_len, cache->grid_data); | 
					
						
							|  |  |  |   writestruct(wd, DATA, LightProbeCache, cache->cube_len, cache->cube_data); | 
					
						
							| 
									
										
										
										
											2018-07-10 15:02:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_scene(WriteData *wd, Scene *sce, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-22 11:35:28 +02:00
										 |  |  |   if (wd->use_memfile) { | 
					
						
							|  |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     /* XXX This UI data should not be stored in Scene at all... */ | 
					
						
							|  |  |  |     memset(&sce->cursor, 0, sizeof(sce->cursor)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |   writestruct_at_address(wd, ID_SCE, Scene, 1, id_address, sce); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_iddata(wd, &sce->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sce->adt) { | 
					
						
							|  |  |  |     write_animdata(wd, sce->adt); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   write_keyingsets(wd, &sce->keyingsets); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* direct data */ | 
					
						
							|  |  |  |   ToolSettings *tos = sce->toolsettings; | 
					
						
							|  |  |  |   writestruct(wd, DATA, ToolSettings, 1, tos); | 
					
						
							|  |  |  |   if (tos->vpaint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, VPaint, 1, tos->vpaint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->vpaint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->wpaint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, VPaint, 1, tos->wpaint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->wpaint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->sculpt) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, Sculpt, 1, tos->sculpt); | 
					
						
							|  |  |  |     write_paint(wd, &tos->sculpt->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->uvsculpt) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, UvSculpt, 1, tos->uvsculpt); | 
					
						
							|  |  |  |     write_paint(wd, &tos->uvsculpt->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->gp_paint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, GpPaint, 1, tos->gp_paint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->gp_paint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |   if (tos->gp_vertexpaint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, GpVertexPaint, 1, tos->gp_vertexpaint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->gp_vertexpaint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->gp_sculptpaint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, GpSculptPaint, 1, tos->gp_sculptpaint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->gp_sculptpaint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (tos->gp_weightpaint) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, GpWeightPaint, 1, tos->gp_weightpaint); | 
					
						
							|  |  |  |     write_paint(wd, &tos->gp_weightpaint->paint); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write grease-pencil custom ipo curve to file */ | 
					
						
							|  |  |  |   if (tos->gp_interpolate.custom_ipo) { | 
					
						
							|  |  |  |     write_curvemapping(wd, tos->gp_interpolate.custom_ipo); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* write grease-pencil multiframe falloff curve to file */ | 
					
						
							|  |  |  |   if (tos->gp_sculpt.cur_falloff) { | 
					
						
							|  |  |  |     write_curvemapping(wd, tos->gp_sculpt.cur_falloff); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* write grease-pencil primitive curve to file */ | 
					
						
							|  |  |  |   if (tos->gp_sculpt.cur_primitive) { | 
					
						
							|  |  |  |     write_curvemapping(wd, tos->gp_sculpt.cur_primitive); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-11-20 16:12:32 -05:00
										 |  |  |   /* Write the curve profile to the file. */ | 
					
						
							|  |  |  |   if (tos->custom_bevel_profile_preset) { | 
					
						
							|  |  |  |     write_CurveProfile(wd, tos->custom_bevel_profile_preset); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   write_paint(wd, &tos->imapaint.paint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Editing *ed = sce->ed; | 
					
						
							|  |  |  |   if (ed) { | 
					
						
							|  |  |  |     Sequence *seq; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, Editing, 1, ed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* reset write flags too */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SEQ_BEGIN (ed, seq) { | 
					
						
							|  |  |  |       if (seq->strip) { | 
					
						
							|  |  |  |         seq->strip->done = false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       writestruct(wd, DATA, Sequence, 1, seq); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SEQ_END; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SEQ_BEGIN (ed, seq) { | 
					
						
							|  |  |  |       if (seq->strip && seq->strip->done == 0) { | 
					
						
							|  |  |  |         /* write strip with 'done' at 0 because readfile */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (seq->effectdata) { | 
					
						
							|  |  |  |           switch (seq->type) { | 
					
						
							|  |  |  |             case SEQ_TYPE_COLOR: | 
					
						
							|  |  |  |               writestruct(wd, DATA, SolidColorVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_SPEED: | 
					
						
							|  |  |  |               writestruct(wd, DATA, SpeedControlVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_WIPE: | 
					
						
							|  |  |  |               writestruct(wd, DATA, WipeVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_GLOW: | 
					
						
							|  |  |  |               writestruct(wd, DATA, GlowVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_TRANSFORM: | 
					
						
							|  |  |  |               writestruct(wd, DATA, TransformVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_GAUSSIAN_BLUR: | 
					
						
							|  |  |  |               writestruct(wd, DATA, GaussianBlurVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_TEXT: | 
					
						
							|  |  |  |               writestruct(wd, DATA, TextVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case SEQ_TYPE_COLORMIX: | 
					
						
							|  |  |  |               writestruct(wd, DATA, ColorMixVars, 1, seq->effectdata); | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         writestruct(wd, DATA, Stereo3dFormat, 1, seq->stereo3d_format); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Strip *strip = seq->strip; | 
					
						
							|  |  |  |         writestruct(wd, DATA, Strip, 1, strip); | 
					
						
							| 
									
										
										
										
											2019-05-31 14:40:41 +02:00
										 |  |  |         if (strip->crop) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           writestruct(wd, DATA, StripCrop, 1, strip->crop); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-31 14:40:41 +02:00
										 |  |  |         if (strip->transform) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           writestruct(wd, DATA, StripTransform, 1, strip->transform); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-31 14:40:41 +02:00
										 |  |  |         if (strip->proxy) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           writestruct(wd, DATA, StripProxy, 1, strip->proxy); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (seq->type == SEQ_TYPE_IMAGE) { | 
					
						
							|  |  |  |           writestruct(wd, | 
					
						
							|  |  |  |                       DATA, | 
					
						
							|  |  |  |                       StripElem, | 
					
						
							|  |  |  |                       MEM_allocN_len(strip->stripdata) / sizeof(struct StripElem), | 
					
						
							|  |  |  |                       strip->stripdata); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (ELEM(seq->type, SEQ_TYPE_MOVIE, SEQ_TYPE_SOUND_RAM, SEQ_TYPE_SOUND_HD)) { | 
					
						
							|  |  |  |           writestruct(wd, DATA, StripElem, 1, strip->stripdata); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         strip->done = true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (seq->prop) { | 
					
						
							|  |  |  |         IDP_WriteProperty(seq->prop, wd); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       write_sequence_modifiers(wd, &seq->modifiers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SEQ_END; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* new; meta stack too, even when its nasty restore code */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (MetaStack *, ms, &ed->metastack) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, MetaStack, 1, ms); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sce->r.avicodecdata) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, AviCodecData, 1, sce->r.avicodecdata); | 
					
						
							|  |  |  |     if (sce->r.avicodecdata->lpFormat) { | 
					
						
							|  |  |  |       writedata(wd, DATA, sce->r.avicodecdata->cbFormat, sce->r.avicodecdata->lpFormat); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sce->r.avicodecdata->lpParms) { | 
					
						
							|  |  |  |       writedata(wd, DATA, sce->r.avicodecdata->cbParms, sce->r.avicodecdata->lpParms); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (sce->r.ffcodecdata.properties) { | 
					
						
							|  |  |  |     IDP_WriteProperty(sce->r.ffcodecdata.properties, wd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* writing dynamic list of TimeMarkers to the blend file */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (TimeMarker *, marker, &sce->markers) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, TimeMarker, 1, marker); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* writing dynamic list of TransformOrientations to the blend file */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (TransformOrientation *, ts, &sce->transform_spaces) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, TransformOrientation, 1, ts); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* writing MultiView to the blend file */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (SceneRenderView *, srv, &sce->r.views) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, SceneRenderView, 1, srv); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sce->nodetree) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, bNodeTree, 1, sce->nodetree); | 
					
						
							|  |  |  |     write_nodetree_nolib(wd, sce->nodetree); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   write_view_settings(wd, &sce->view_settings); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* writing RigidBodyWorld data to the blend file */ | 
					
						
							|  |  |  |   if (sce->rigidbody_world) { | 
					
						
							|  |  |  |     /* Set deprecated pointers to prevent crashes of older Blenders */ | 
					
						
							|  |  |  |     sce->rigidbody_world->pointcache = sce->rigidbody_world->shared->pointcache; | 
					
						
							|  |  |  |     sce->rigidbody_world->ptcaches = sce->rigidbody_world->shared->ptcaches; | 
					
						
							|  |  |  |     writestruct(wd, DATA, RigidBodyWorld, 1, sce->rigidbody_world); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writestruct(wd, DATA, RigidBodyWorld_Shared, 1, sce->rigidbody_world->shared); | 
					
						
							|  |  |  |     writestruct(wd, DATA, EffectorWeights, 1, sce->rigidbody_world->effector_weights); | 
					
						
							|  |  |  |     write_pointcaches(wd, &(sce->rigidbody_world->shared->ptcaches)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   write_previews(wd, sce->preview); | 
					
						
							|  |  |  |   write_curvemapping_curves(wd, &sce->r.mblur_shutter_curve); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (ViewLayer *, view_layer, &sce->view_layers) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_view_layer(wd, view_layer); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sce->master_collection) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, Collection, 1, sce->master_collection); | 
					
						
							|  |  |  |     write_collection_nolib(wd, sce->master_collection); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Eevee Lightcache */ | 
					
						
							| 
									
										
										
										
											2020-03-11 17:12:01 +01:00
										 |  |  |   if (sce->eevee.light_cache_data && !wd->use_memfile) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, LightCache, 1, sce->eevee.light_cache_data); | 
					
						
							|  |  |  |     write_lightcache(wd, sce->eevee.light_cache_data); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-05 12:47:20 +02:00
										 |  |  |   write_view3dshading(wd, &sce->display.shading); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Freed on doversion. */ | 
					
						
							|  |  |  |   BLI_assert(sce->layer_properties == NULL); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_gpencil(WriteData *wd, bGPdata *gpd, const void *id_address) | 
					
						
							| 
									
										
										
										
											2008-07-22 09:53:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (gpd->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-03 12:38:04 +11:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed data-blocks. */ | 
					
						
							|  |  |  |     /* XXX not sure why the whole run-time data is not cleared in reading code,
 | 
					
						
							|  |  |  |      * for now mimicking it here. */ | 
					
						
							| 
									
										
										
										
											2020-04-01 12:23:12 +02:00
										 |  |  |     gpd->runtime.sbuffer = NULL; | 
					
						
							|  |  |  |     gpd->runtime.sbuffer_used = 0; | 
					
						
							|  |  |  |     gpd->runtime.sbuffer_size = 0; | 
					
						
							|  |  |  |     gpd->runtime.tot_cp_points = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write gpd data block to file */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_GD, bGPdata, 1, id_address, gpd); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &gpd->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (gpd->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, gpd->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * gpd->totcol, gpd->mat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* write grease-pencil layers to file */ | 
					
						
							|  |  |  |     writelist(wd, DATA, bGPDlayer, &gpd->layers); | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |     LISTBASE_FOREACH (bGPDlayer *, gpl, &gpd->layers) { | 
					
						
							|  |  |  |       /* Write mask list. */ | 
					
						
							|  |  |  |       writelist(wd, DATA, bGPDlayer_Mask, &gpl->mask_layers); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       /* write this layer's frames to file */ | 
					
						
							|  |  |  |       writelist(wd, DATA, bGPDframe, &gpl->frames); | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |       LISTBASE_FOREACH (bGPDframe *, gpf, &gpl->frames) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         /* write strokes */ | 
					
						
							|  |  |  |         writelist(wd, DATA, bGPDstroke, &gpf->strokes); | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |         LISTBASE_FOREACH (bGPDstroke *, gps, &gpf->strokes) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           writestruct(wd, DATA, bGPDspoint, gps->totpoints, gps->points); | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  |           writestruct(wd, DATA, bGPDtriangle, gps->tot_triangles, gps->triangles); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           write_dverts(wd, gps->totpoints, gps->dvert); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-07-22 09:53:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												VR: Initial Virtual Reality support - Milestone 1, Scene Inspection
NOTE: While most of the milestone 1 goals are there, a few smaller features and
improvements are still to be done.
Big picture of this milestone: Initial, OpenXR-based virtual reality support
for users and foundation for advanced use cases.
Maniphest Task: https://developer.blender.org/T71347
The tasks contains more information about this milestone.
To be clear: This is not a feature rich VR implementation, it's focused on the
initial scene inspection use case. We intentionally focused on that, further
features like controller support are part of the next milestone.
- How to use?
Instructions on how to use this are here:
https://wiki.blender.org/wiki/User:Severin/GSoC-2019/How_to_Test
These will be updated and moved to a more official place (likely the manual) soon.
Currently Windows Mixed Reality and Oculus devices are usable. Valve/HTC
headsets don't support the OpenXR standard yet and hence, do not work with this
implementation.
---------------
This is the C-side implementation of the features added for initial VR
support as per milestone 1. A "VR Scene Inspection" Add-on will be
committed separately, to expose the VR functionality in the UI. It also
adds some further features for milestone 1, namely a landmarking system
(stored view locations in the VR space)
Main additions/features:
* Support for rendering viewports to an HMD, with good performance.
* Option to sync the VR view perspective with a fully interactive,
  regular 3D View (VR-Mirror).
* Option to disable positional tracking. Keeps the current position (calculated
  based on the VR eye center pose) when enabled while a VR session is running.
* Some regular viewport settings for the VR view
* RNA/Python-API to query and set VR session state information.
* WM-XR: Layer tying Ghost-XR to the Blender specific APIs/data
* wmSurface API: drawable, non-window container (manages Ghost-OpenGL and GPU
  context)
* DNA/RNA for management of VR session settings
* `--debug-xr` and `--debug-xr-time` commandline options
* Utility batch & config file for using the Oculus runtime on Windows.
* Most VR data is runtime only. The exception is user settings which are saved
  to files (`XrSessionSettings`).
* VR support can be disabled through the `WITH_XR_OPENXR` compiler flag.
For architecture and code documentation, see
https://wiki.blender.org/wiki/Source/Interface/XR.
---------------
A few thank you's:
* A huge shoutout to Ray Molenkamp for his help during the project - it would
  have not been that successful without him!
* Sebastian Koenig and Simeon Conzendorf for testing and feedback!
* The reviewers, especially Brecht Van Lommel!
* Dalai Felinto for pushing and managing me to get this done ;)
* The OpenXR working group for providing an open standard. I think we're the
  first bigger application to adopt OpenXR. Congratulations to them and
  ourselves :)
This project started as a Google Summer of Code 2019 project - "Core Support of
Virtual Reality Headsets through OpenXR" (see
https://wiki.blender.org/wiki/User:Severin/GSoC-2019/).
Some further information, including ideas for further improvements can be found
in the final GSoC report:
https://wiki.blender.org/wiki/User:Severin/GSoC-2019/Final_Report
Differential Revisions: D6193, D7098
Reviewed by: Brecht Van Lommel, Jeroen Bakker
											
										 
											2020-03-17 20:20:55 +01:00
										 |  |  | static void write_wm_xr_data(WriteData *wd, wmXrData *xr_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   write_view3dshading(wd, &xr_data->session_settings.shading); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  | static void write_region(WriteData *wd, ARegion *region, int spacetype) | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |   writestruct(wd, DATA, ARegion, 1, region); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |   if (region->regiondata) { | 
					
						
							|  |  |  |     if (region->flag & RGN_FLAG_TEMP_REGIONDATA) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (spacetype) { | 
					
						
							|  |  |  |       case SPACE_VIEW3D: | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |         if (region->regiontype == RGN_TYPE_WINDOW) { | 
					
						
							|  |  |  |           RegionView3D *rv3d = region->regiondata; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           writestruct(wd, DATA, RegionView3D, 1, rv3d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (rv3d->localvd) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, RegionView3D, 1, rv3d->localvd); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (rv3d->clipbb) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, BoundBox, 1, rv3d->clipbb); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-22 09:13:00 +10:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           printf("regiondata write missing!\n"); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:13:00 +10:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         printf("regiondata write missing!\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												2.5
View3D has been split now in a local part (RegionView3D) and a
per-area part (old View3D). Currently local is:
- view transform
- camera zoom/offset
- gpencil (todo)
- custom clipping planes
Rest is in Area still, like active camera, draw type, layers,
localview, custom centers, around-settings, transform widget,
gridlines, and so on (mostly stuff as available in header).
To see it work; also added new feature for region split, 
press SHIFT+ALT+CTRL+S for four-split. 
The idea is to make a preset 4-split, configured to stick
to top/right/front views for three views.
Another cool idea to explore is to then box-clip all drawing
based on these 3 views.
Note about the code:
- currently view3d still stores some depricated settings, to
  convert from older files. Not all settings are copied over
  though, like custom clip planes or the 'lock view to object'.
- since some view3d ops are now on area level, the operators
  for it should keep track of that.
Bugfix in transform: quat initialize in operator-invoke missed
one zero.
Als brought back GE to compile for missing Ipos and channels.
											
										 
											2009-01-19 16:54:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-29 12:55:31 +00:00
										 |  |  | static void write_uilist(WriteData *wd, uiList *ui_list) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, DATA, uiList, 1, ui_list); | 
					
						
							| 
									
										
										
										
											2013-08-29 12:55:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ui_list->properties) { | 
					
						
							|  |  |  |     IDP_WriteProperty(ui_list->properties, wd); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-08-29 12:55:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-16 09:47:19 +11:00
										 |  |  | static void write_soops(WriteData *wd, SpaceOutliner *so) | 
					
						
							| 
									
										
										
										
											2013-08-03 11:35:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_mempool *ts = so->treestore; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ts) { | 
					
						
							|  |  |  |     SpaceOutliner so_flat = *so; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int elems = BLI_mempool_len(ts); | 
					
						
							|  |  |  |     /* linearize mempool to array */ | 
					
						
							|  |  |  |     TreeStoreElem *data = elems ? BLI_mempool_as_arrayN(ts, "TreeStoreElem") : NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (data) { | 
					
						
							|  |  |  |       /* In this block we use the memory location of the treestore
 | 
					
						
							|  |  |  |        * but _not_ its data, the addresses in this case are UUID's, | 
					
						
							|  |  |  |        * since we can't rely on malloc giving us different values each time. | 
					
						
							|  |  |  |        */ | 
					
						
							|  |  |  |       TreeStore ts_flat = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* we know the treestore is at least as big as a pointer,
 | 
					
						
							|  |  |  |        * so offsetting works to give us a UUID. */ | 
					
						
							|  |  |  |       void *data_addr = (void *)POINTER_OFFSET(ts, sizeof(void *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ts_flat.usedelem = elems; | 
					
						
							|  |  |  |       ts_flat.totelem = elems; | 
					
						
							|  |  |  |       ts_flat.data = data_addr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceOutliner, 1, so); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct_at_address(wd, DATA, TreeStore, 1, ts, &ts_flat); | 
					
						
							|  |  |  |       writestruct_at_address(wd, DATA, TreeStoreElem, elems, data_addr, data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       MEM_freeN(data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       so_flat.treestore = NULL; | 
					
						
							|  |  |  |       writestruct_at_address(wd, DATA, SpaceOutliner, 1, so, &so_flat); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     writestruct(wd, DATA, SpaceOutliner, 1, so); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-08-03 11:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-03 13:32:36 +02:00
										 |  |  | static void write_panel_list(WriteData *wd, ListBase *lb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 22:20:25 -05:00
										 |  |  |   LISTBASE_FOREACH (Panel *, panel, lb) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, Panel, 1, panel); | 
					
						
							|  |  |  |     write_panel_list(wd, &panel->children); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-03 13:32:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-13 20:29:16 +02:00
										 |  |  | static void write_area_regions(WriteData *wd, ScrArea *area) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (ARegion *, region, &area->regionbase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_region(wd, region, area->spacetype); | 
					
						
							|  |  |  |     write_panel_list(wd, ®ion->panels); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (PanelCategoryStack *, pc_act, ®ion->panels_category_active) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, PanelCategoryStack, 1, pc_act); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (uiList *, ui_list, ®ion->ui_lists) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_uilist(wd, ui_list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (uiPreview *, ui_preview, ®ion->ui_previews) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, uiPreview, 1, ui_preview); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) { | 
					
						
							|  |  |  |     LISTBASE_FOREACH (ARegion *, region, &sl->regionbase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_region(wd, region, sl->spacetype); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sl->spacetype == SPACE_VIEW3D) { | 
					
						
							|  |  |  |       View3D *v3d = (View3D *)sl; | 
					
						
							|  |  |  |       writestruct(wd, DATA, View3D, 1, v3d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (v3d->localvd) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, View3D, 1, v3d->localvd); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-05 12:47:20 +02:00
										 |  |  |       write_view3dshading(wd, &v3d->shading); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_GRAPH) { | 
					
						
							|  |  |  |       SpaceGraph *sipo = (SpaceGraph *)sl; | 
					
						
							|  |  |  |       ListBase tmpGhosts = sipo->runtime.ghost_curves; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* temporarily disable ghost curves when saving */ | 
					
						
							|  |  |  |       BLI_listbase_clear(&sipo->runtime.ghost_curves); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceGraph, 1, sl); | 
					
						
							|  |  |  |       if (sipo->ads) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, bDopeSheet, 1, sipo->ads); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* reenable ghost curves */ | 
					
						
							|  |  |  |       sipo->runtime.ghost_curves = tmpGhosts; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_PROPERTIES) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceProperties, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_FILE) { | 
					
						
							|  |  |  |       SpaceFile *sfile = (SpaceFile *)sl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceFile, 1, sl); | 
					
						
							|  |  |  |       if (sfile->params) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, FileSelectParams, 1, sfile->params); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_SEQ) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceSeq, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_OUTLINER) { | 
					
						
							|  |  |  |       SpaceOutliner *so = (SpaceOutliner *)sl; | 
					
						
							|  |  |  |       write_soops(wd, so); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_IMAGE) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceImage, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_TEXT) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceText, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_SCRIPT) { | 
					
						
							|  |  |  |       SpaceScript *scr = (SpaceScript *)sl; | 
					
						
							|  |  |  |       scr->but_refs = NULL; | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceScript, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_ACTION) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceAction, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_NLA) { | 
					
						
							|  |  |  |       SpaceNla *snla = (SpaceNla *)sl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceNla, 1, snla); | 
					
						
							|  |  |  |       if (snla->ads) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, bDopeSheet, 1, snla->ads); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_NODE) { | 
					
						
							|  |  |  |       SpaceNode *snode = (SpaceNode *)sl; | 
					
						
							|  |  |  |       bNodeTreePath *path; | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceNode, 1, snode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (path = snode->treepath.first; path; path = path->next) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, bNodeTreePath, 1, path); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_CONSOLE) { | 
					
						
							|  |  |  |       SpaceConsole *con = (SpaceConsole *)sl; | 
					
						
							|  |  |  |       ConsoleLine *cl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (cl = con->history.first; cl; cl = cl->next) { | 
					
						
							|  |  |  |         /* 'len_alloc' is invalid on write, set from 'len' on read */ | 
					
						
							|  |  |  |         writestruct(wd, DATA, ConsoleLine, 1, cl); | 
					
						
							|  |  |  |         writedata(wd, DATA, cl->len + 1, cl->line); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceConsole, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-23 22:38:25 +02:00
										 |  |  | #ifdef WITH_GLOBAL_AREA_WRITING
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else if (sl->spacetype == SPACE_TOPBAR) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceTopBar, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_STATUSBAR) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceStatusBar, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else if (sl->spacetype == SPACE_USERPREF) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceUserPref, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_CLIP) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceClip, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (sl->spacetype == SPACE_INFO) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, SpaceInfo, 1, sl); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_area_map(WriteData *wd, ScrAreaMap *area_map) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writelist(wd, DATA, ScrVert, &area_map->vertbase); | 
					
						
							|  |  |  |   writelist(wd, DATA, ScrEdge, &area_map->edgebase); | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (ScrArea *, area, &area_map->areabase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     area->butspacetype = area->spacetype; /* Just for compatibility, will be reset below. */ | 
					
						
							| 
									
										
										
										
											2018-04-22 19:58:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, ScrArea, 1, area); | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-23 22:38:25 +02:00
										 |  |  | #ifdef WITH_GLOBAL_AREA_WRITING
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, ScrGlobalAreaData, 1, area->global); | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-07-08 12:22:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_area_regions(wd, area); | 
					
						
							| 
									
										
										
										
											2018-04-22 19:58:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     area->butspacetype = SPACE_EMPTY; /* Unset again, was changed above. */ | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-13 20:29:16 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  | static void write_windowmanager(BlendWriter *writer, wmWindowManager *wm, const void *id_address) | 
					
						
							| 
									
										
										
										
											2018-04-13 20:29:16 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  |   BLO_write_id_struct(writer, wmWindowManager, id_address, &wm->id); | 
					
						
							|  |  |  |   write_iddata(writer->wd, &wm->id); | 
					
						
							|  |  |  |   write_wm_xr_data(writer->wd, &wm->xr); | 
					
						
							| 
									
										
										
										
											2016-06-28 17:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (wmWindow *, win, &wm->windows) { | 
					
						
							| 
									
										
										
										
											2018-05-23 22:38:25 +02:00
										 |  |  | #ifndef WITH_GLOBAL_AREA_WRITING
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* Don't write global areas yet, while we make changes to them. */ | 
					
						
							|  |  |  |     ScrAreaMap global_areas = win->global_areas; | 
					
						
							|  |  |  |     memset(&win->global_areas, 0, sizeof(win->global_areas)); | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-23 09:24:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* update deprecated screen member (for so loading in 2.7x uses the correct screen) */ | 
					
						
							|  |  |  |     win->screen = BKE_workspace_active_screen_get(win->workspace_hook); | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  |     BLO_write_struct(writer, wmWindow, win); | 
					
						
							|  |  |  |     BLO_write_struct(writer, WorkSpaceInstanceHook, win->workspace_hook); | 
					
						
							|  |  |  |     BLO_write_struct(writer, Stereo3dFormat, win->stereo3d_format); | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-23 22:38:25 +02:00
										 |  |  | #ifdef WITH_GLOBAL_AREA_WRITING
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  |     write_area_map(writer->wd, &win->global_areas); | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     win->global_areas = global_areas; | 
					
						
							| 
									
										
											  
											
												UI: New Global Top-Bar (WIP)
== Main Features/Changes for Users
* Add horizontal bar at top of all non-temp windows, consisting out of two horizontal sub-bars.
* Upper sub-bar contains global menus (File, Render, etc.), tabs for workspaces and scene selector.
* Lower sub-bar contains object mode selector, screen-layout and render-layer selector. Later operator and/or tool settings will be placed here.
* Individual sections of the topbar are individually scrollable.
* Workspace tabs can be double- or ctrl-clicked for renaming and contain 'x' icon for deleting.
* Top-bar should scale nicely with DPI.
* The lower half of the top-bar can be hided by dragging the lower top-bar edge up. Better hiding options are planned (e.g. hide in fullscreen modes).
* Info editors at the top of the window and using the full window width with be replaced by the top-bar.
* In fullscreen modes, no more info editor is added on top, the top-bar replaces it.
== Technical Features/Changes
* Adds initial support for global areas
  A global area is part of the window, not part of the regular screen-layout.
  I've added a macro iterator to iterate over both, global and screen-layout level areas. When iterating over areas, from now on developers should always consider if they have to include global areas.
* Adds a TOPBAR editor type
  The editor type is hidden in the UI editor type menu.
* Adds a variation of the ID template to display IDs as tab buttons (template_ID_tabs in BPY)
* Does various changes to RNA button creation code to improve their appearance in the horizontal top-bar.
* Adds support for dynamically sized regions. That is, regions that scale automatically to the layout bounds.
  The code for this is currently a big hack (it's based on drawing the UI multiple times). This should definitely be improved.
* Adds a template for displaying operator properties optimized for the top-bar. This will probably change a lot still and is in fact disabled in code.
Since the final top-bar design depends a lot on other 2.8 designs (mainly tool-system and workspaces), we decided to not show the operator or tool settings in the top-bar for now. That means most of the lower sub-bar is empty for the time being.
NOTE: Top-bar or global area data is not written to files or SDNA. They are simply added to the window when opening Blender or reading a file. This allows us doing changes to the top-bar without having to care for compatibility.
== ToDo's
It's a bit hard to predict all the ToDo's here are the known main ones:
* Add options for the new active-tool system and for operator redo to the topbar.
* Automatically hide the top-bar in fullscreen modes.
* General visual polish.
* Top-bar drag & drop support (WIP in temp-tab_drag_drop).
* Improve dynamic regions (should also fix some layout glitches).
* Make internal terminology consistent.
* Enable topbar file writing once design is more advanced.
* Address TODO's and XXX's in code :)
Thanks @brecht for the review! And @sergey for the complaining ;)
Differential Revision: D2758
											
										 
											2018-04-20 17:14:03 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* data is written, clear deprecated data again */ | 
					
						
							|  |  |  |     win->screen = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-13 20:29:16 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
											
												Refactor writefile handling of data-blocks.
Instead of calling a function looping over whole list of a given ID
type, make whole loop over Main in parent function, and call functions
writing a single datablock at a time.
This design is more in line with all other places in Blender where we
handle whole content of Main (including readfile.c), and much more easy
to extend and add e.g. some generic processing of IDs before/after
writing, etc.
From user point, there should be no change at all, only difference is
that data-block types won't be saved in same order as before (.blend
file specs enforces no order here, so this is not an issue, but it could
bug some third party users using other, simplified .blend file reader maybe).
Reviewers: sergey, campbellbarton
Differential Revision: https://developer.blender.org/D2510
											
										 
											2017-03-17 10:02:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 14:23:21 +02:00
										 |  |  | static void write_screen(WriteData *wd, bScreen *screen, const void *id_address) | 
					
						
							| 
									
										
										
										
											2018-04-13 20:29:16 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Screens are reference counted, only saved if used by a workspace. */ | 
					
						
							| 
									
										
										
										
											2020-04-03 14:23:21 +02:00
										 |  |  |   if (screen->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							|  |  |  |     /* in 2.50+ files, the file identifier for screens is patched, forward compatibility */ | 
					
						
							| 
									
										
										
										
											2020-04-03 14:23:21 +02:00
										 |  |  |     writestruct_at_address(wd, ID_SCRN, bScreen, 1, id_address, screen); | 
					
						
							|  |  |  |     write_iddata(wd, &screen->id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 14:23:21 +02:00
										 |  |  |     write_previews(wd, screen->preview); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							| 
									
										
										
										
											2020-04-03 14:23:21 +02:00
										 |  |  |     write_area_map(wd, AREAMAP_FROM_SCREEN(screen)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-26 11:35:43 +00:00
										 |  |  | static void write_bone(WriteData *wd, Bone *bone) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* PATCH for upward compatibility after 2.37+ armature recode */ | 
					
						
							|  |  |  |   bone->size[0] = bone->size[1] = bone->size[2] = 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write this bone */ | 
					
						
							|  |  |  |   writestruct(wd, DATA, Bone, 1, bone); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write ID Properties -- and copy this comment EXACTLY for easy finding
 | 
					
						
							|  |  |  |    * of library blocks that implement this.*/ | 
					
						
							|  |  |  |   if (bone->prop) { | 
					
						
							|  |  |  |     IDP_WriteProperty(bone->prop, wd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write Children */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (Bone *, cbone, &bone->childbase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_bone(wd, cbone); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_armature(WriteData *wd, bArmature *arm, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (arm->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:57:32 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     arm->bonehash = NULL; | 
					
						
							|  |  |  |     arm->edbo = NULL; | 
					
						
							|  |  |  |     /* Must always be cleared (armatures don't have their own edit-data). */ | 
					
						
							|  |  |  |     arm->needs_flush_to_id = 0; | 
					
						
							|  |  |  |     arm->act_edbone = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_AR, bArmature, 1, id_address, arm); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &arm->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (arm->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, arm->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Direct data */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (Bone *, bone, &arm->bonebase) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_bone(wd, bone); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_text(WriteData *wd, Text *text, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |   /* Note: we are clearing local temp data here, *not* the flag in the actual 'real' ID. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((text->flags & TXT_ISMEM) && (text->flags & TXT_ISEXT)) { | 
					
						
							|  |  |  |     text->flags &= ~TXT_ISEXT; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 10:38:20 +02:00
										 |  |  |   /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |   text->compiled = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |   writestruct_at_address(wd, ID_TXT, Text, 1, id_address, text); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_iddata(wd, &text->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (text->name) { | 
					
						
							|  |  |  |     writedata(wd, DATA, strlen(text->name) + 1, text->name); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!(text->flags & TXT_ISEXT)) { | 
					
						
							|  |  |  |     /* now write the text data, in two steps for optimization in the readfunction */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (TextLine *, tmp, &text->lines) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, TextLine, 1, tmp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (TextLine *, tmp, &text->lines) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writedata(wd, DATA, tmp->len + 1, tmp->line); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_speaker(WriteData *wd, Speaker *spk, const void *id_address) | 
					
						
							| 
									
										
										
										
											2011-08-01 11:44:20 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (spk->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_SPK, Speaker, 1, id_address, spk); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &spk->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (spk->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, spk->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-08-01 11:44:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_sound(WriteData *wd, bSound *sound, const void *id_address) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (sound->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:43:03 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     sound->tags = 0; | 
					
						
							|  |  |  |     sound->handle = NULL; | 
					
						
							|  |  |  |     sound->playback_handle = NULL; | 
					
						
							|  |  |  |     sound->spinlock = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_SO, bSound, 1, id_address, sound); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &sound->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sound->packedfile) { | 
					
						
							|  |  |  |       PackedFile *pf = sound->packedfile; | 
					
						
							|  |  |  |       writestruct(wd, DATA, PackedFile, 1, pf); | 
					
						
							|  |  |  |       writedata(wd, DATA, pf->size, pf->data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_probe(WriteData *wd, LightProbe *prb, const void *id_address) | 
					
						
							| 
									
										
										
										
											2017-06-06 22:47:41 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (prb->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_LP, LightProbe, 1, id_address, prb); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &prb->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (prb->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, prb->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-06 22:47:41 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_nodetree(WriteData *wd, bNodeTree *ntree, const void *id_address) | 
					
						
							| 
									
										
											  
											
												Orange: more noodle updates!
**** NEW: Group Nodes
Node trees usually become messy and confusing quickly, so we need
not only a way to collapse Nodes into single 'groups', but also a
way to re-use that data to create libraries of effects.
This has been done by making a new Library data type, the NodeTree.
Everything that has been grouped is stored here, and available for
re-use, appending or linking. These NodeTrees are fully generic,
i.e. can store shader trees, composit trees, and so on. The 'type'
value as stored in the NodeTree will keep track of internal type
definitions and execute/drawing callbacks. Needless to say, re-using
shader trees in a composit tree is a bit useless, and will be
prevented in the browsing code. :)
So; any NodeTree can become a "Goup Node" inside in a NodeTree. This
Group Node then works just like any Node.
To prevent the current code to become too complex, I've disabled
the possibility to insert Groups inside of Groups. That might be
enabled later, but is a real nasty piece of code to get OK.
Since Group Nodes are a dynamic Node type, a lot of work has been
done to ensure Node definitions can be dynamic too, but still allow
to be stored in files, and allow to be verified for type-definition
changes on reloading. This system needs a little bit maturing still,
so the Python gurus should better wait a little bit! (Also for me to
write the definite API docs for it).
What works now:
- Press CTRL+G to create a new Group. The grouping code checks for
impossible selections (like an unselected node between selected nodes).
Everthing that's selected then gets removed from the current tree, and
inserted in a new NodeTree library data block. A Group Node then is
added which links to this new NodeTree.
- Press ALT+G to ungroup. This will not delete the NodeTree library
data, but just duplicate the Group into the current tree.
- Press TAB, or click on the NodeTree icon to edit Groups. Note that
NodeTrees are instances, so editing one Group will also change the
other users.
This also means that when removing nodes in a Group (or hiding sockets
or changing internal links) this is immediately corrected for all users
of this Group, also in other Materials.
- While editing Groups, only the internal Nodes can be edited. A single
click outside of the Group boundary will close this 'edit mode'.
What needs to be done:
- SHIFT+A menu in toolbox style, also including a list of Groups
- Enable the single-user button in the Group Node
- Displaying all (visible) internal group UI elements in the Node Panel
- Enable Library linking and prevent editing of Groups then.
**** NEW: Socket Visibility control
Node types will be generated with a lot of possible inputs or outputs,
and drawing all sockets all the time isn't very useful then.
A new option in the Node header ('plus' icon) allows to either hide all
unused sockets (first keypress) or to reveil them (when there are hidden
sockets, the icon displays black, otherwise it's blended).
Hidden sockets in Nodes also are not exported to a Group, so this way
you can control what options (in/outputs) exactly are available.
To be done:
- a way to hide individual sockets, like with a RMB click on it.
**** NEW: Nodes now render!
This is still quite primitive, more on a level to replace the (now
obsolete and disabled) Material Layers.
What needs to be done:
- make the "Geometry" node work properly, also for AA textures
- make the Texture Node work (does very little at the moment)
- give Material Nodes all inputs as needed (like Map-to Panel)
- find a way to export more data from a Material Node, like the
  shadow value, or light intensity only, etc
Very important also to separate from the Material Buttons the
"global" options, like "Ztransp" or "Wire" or "Halo". These can not
be set for each Material-Node individually.
Also note that the Preview Render (Buttons window) now renders a bit
differently. This was a horrid piece of antique code, using a totally
incompatible way of rendering. Target is to fully re-use internal
render code for previews.
OK... that's it mostly. Now test!
											
										 
											2006-01-02 13:06:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ntree->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 12:01:27 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     ntree->init = 0; /* to set callbacks and force setting types */ | 
					
						
							|  |  |  |     ntree->is_updating = false; | 
					
						
							|  |  |  |     ntree->typeinfo = NULL; | 
					
						
							|  |  |  |     ntree->interface_type = NULL; | 
					
						
							|  |  |  |     ntree->progress = NULL; | 
					
						
							|  |  |  |     ntree->execdata = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_NT, bNodeTree, 1, id_address, ntree); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* Note that trees directly used by other IDs (materials etc.) are not 'real' ID, they cannot
 | 
					
						
							|  |  |  |      * be linked, etc., so we write actual id data here only, for 'real' ID trees. */ | 
					
						
							|  |  |  |     write_iddata(wd, &ntree->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_nodetree_nolib(wd, ntree); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_brush(WriteData *wd, Brush *brush, const void *id_address) | 
					
						
							| 
									
										
										
											
												
Brush Datablock:
- Added a new Brush datablock, only used by image paint, but intended
  to be used in texture paint, vertex paint, weight paint and sculpt
  mode also.
- Being a datablock, these brushes can be saved, appended and linked.
  They have a fake user by default, to make sure they are saved even if
  not selected.
Image Painting:
- Replaced the img module with C code in imagepaint.c
- Airbrush is no longer a separate tool, but rather an option that can
  be used for soften, smear and clone also.
- Blend modes mix, add, subtract, multiply, darken and lighten have been
  added, code taken directly from vertex paint.
Note to project files maintainers:
- The img module was removed from SCons and Makefiles, and this should
  be done in other build systems also. I'll wait to remove the module
  from cvs, to not break compilation.
											
										 
											2006-07-26 22:29:23 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (brush->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_BR, Brush, 1, id_address, brush); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &brush->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (brush->curve) { | 
					
						
							|  |  |  |       write_curvemapping(wd, brush->curve); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (brush->gpencil_settings) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, BrushGpencilSettings, 1, brush->gpencil_settings); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_sensitivity) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_sensitivity); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_strength) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_strength); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_jitter) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_jitter); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-05-07 15:02:21 +02:00
										 |  |  |       if (brush->gpencil_settings->curve_rand_pressure) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_pressure); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_rand_strength) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_strength); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_rand_uv) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_uv); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_rand_hue) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_hue); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_rand_saturation) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_saturation); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (brush->gpencil_settings->curve_rand_value) { | 
					
						
							|  |  |  |         write_curvemapping(wd, brush->gpencil_settings->curve_rand_value); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (brush->gradient) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, ColorBand, 1, brush->gradient); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-07-21 12:02:05 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_palette(WriteData *wd, Palette *palette, const void *id_address) | 
					
						
							| 
									
										
										
										
											2014-07-21 12:02:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (palette->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     PaletteColor *color; | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_PAL, Palette, 1, id_address, palette); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &palette->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (color = palette->colors.first; color; color = color->next) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, PaletteColor, 1, color); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-07-21 12:02:05 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_paintcurve(WriteData *wd, PaintCurve *pc, const void *id_address) | 
					
						
							| 
									
										
										
										
											2014-07-21 12:02:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (pc->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_PC, PaintCurve, 1, id_address, pc); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &pc->id); | 
					
						
							| 
									
										
										
										
											2014-07-21 12:02:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writestruct(wd, DATA, PaintCurvePoint, pc->tot_points, pc->points); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
											
												
Brush Datablock:
- Added a new Brush datablock, only used by image paint, but intended
  to be used in texture paint, vertex paint, weight paint and sculpt
  mode also.
- Being a datablock, these brushes can be saved, appended and linked.
  They have a fake user by default, to make sure they are saved even if
  not selected.
Image Painting:
- Replaced the img module with C code in imagepaint.c
- Airbrush is no longer a separate tool, but rather an option that can
  be used for soften, smear and clone also.
- Blend modes mix, add, subtract, multiply, darken and lighten have been
  added, code taken directly from vertex paint.
Note to project files maintainers:
- The img module was removed from SCons and Makefiles, and this should
  be done in other build systems also. I'll wait to remove the module
  from cvs, to not break compilation.
											
										 
											2006-07-26 22:29:23 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | static void write_movieTracks(WriteData *wd, ListBase *tracks) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MovieTrackingTrack *track; | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   track = tracks->first; | 
					
						
							|  |  |  |   while (track) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, MovieTrackingTrack, 1, track); | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (track->markers) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, MovieTrackingMarker, track->markersnr, track->markers); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     track = track->next; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merge plane track feature from tomato branch
This commit includes all the changes made for plane tracker
in tomato branch.
Movie clip editor changes:
- Artist might create a plane track out of multiple point
  tracks which belongs to the same track (minimum amount of
  point tracks is 4, maximum is not actually limited).
  When new plane track is added, it's getting "tracked"
  across all point tracks, which makes it stick to the same
  plane point tracks belong to.
- After plane track was added, it need to be manually adjusted
  in a way it covers feature one might to mask/replace.
  General transform tools (G, R, S) or sliding corners with
  a mouse could be sued for this. Plane corner which
  corresponds to left bottom image corner has got X/Y axis
  on it (red is for X axis, green for Y).
- Re-adjusting plane corners makes plane to be "re-tracked"
  for the frames sequence between current frame and next
  and previous keyframes.
- Kayframes might be removed from the plane, using Shit-X
  (Marker Delete) operator. However, currently manual
  re-adjustment or "re-track" trigger is needed.
Compositor changes:
- Added new node called Plane Track Deform.
- User selects which plane track to use (for this he need
  to select movie clip datablock, object and track names).
- Node gets an image input, which need to be warped into
  the plane.
- Node outputs:
  * Input image warped into the plane.
  * Plane, rasterized to a mask.
Masking changes:
- Mask points might be parented to a plane track, which
  makes this point deforming in a way as if it belongs
  to the tracked plane.
Some video tutorials are available:
- Coder video: http://www.youtube.com/watch?v=vISEwqNHqe4
- Artist video: https://vimeo.com/71727578
This is mine and Keir's holiday code project :)
											
										 
											2013-08-16 09:46:30 +00:00
										 |  |  | static void write_moviePlaneTracks(WriteData *wd, ListBase *plane_tracks_base) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MovieTrackingPlaneTrack *plane_track; | 
					
						
							| 
									
										
											  
											
												Merge plane track feature from tomato branch
This commit includes all the changes made for plane tracker
in tomato branch.
Movie clip editor changes:
- Artist might create a plane track out of multiple point
  tracks which belongs to the same track (minimum amount of
  point tracks is 4, maximum is not actually limited).
  When new plane track is added, it's getting "tracked"
  across all point tracks, which makes it stick to the same
  plane point tracks belong to.
- After plane track was added, it need to be manually adjusted
  in a way it covers feature one might to mask/replace.
  General transform tools (G, R, S) or sliding corners with
  a mouse could be sued for this. Plane corner which
  corresponds to left bottom image corner has got X/Y axis
  on it (red is for X axis, green for Y).
- Re-adjusting plane corners makes plane to be "re-tracked"
  for the frames sequence between current frame and next
  and previous keyframes.
- Kayframes might be removed from the plane, using Shit-X
  (Marker Delete) operator. However, currently manual
  re-adjustment or "re-track" trigger is needed.
Compositor changes:
- Added new node called Plane Track Deform.
- User selects which plane track to use (for this he need
  to select movie clip datablock, object and track names).
- Node gets an image input, which need to be warped into
  the plane.
- Node outputs:
  * Input image warped into the plane.
  * Plane, rasterized to a mask.
Masking changes:
- Mask points might be parented to a plane track, which
  makes this point deforming in a way as if it belongs
  to the tracked plane.
Some video tutorials are available:
- Coder video: http://www.youtube.com/watch?v=vISEwqNHqe4
- Artist video: https://vimeo.com/71727578
This is mine and Keir's holiday code project :)
											
										 
											2013-08-16 09:46:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (plane_track = plane_tracks_base->first; plane_track; plane_track = plane_track->next) { | 
					
						
							|  |  |  |     writestruct(wd, DATA, MovieTrackingPlaneTrack, 1, plane_track); | 
					
						
							| 
									
										
											  
											
												Merge plane track feature from tomato branch
This commit includes all the changes made for plane tracker
in tomato branch.
Movie clip editor changes:
- Artist might create a plane track out of multiple point
  tracks which belongs to the same track (minimum amount of
  point tracks is 4, maximum is not actually limited).
  When new plane track is added, it's getting "tracked"
  across all point tracks, which makes it stick to the same
  plane point tracks belong to.
- After plane track was added, it need to be manually adjusted
  in a way it covers feature one might to mask/replace.
  General transform tools (G, R, S) or sliding corners with
  a mouse could be sued for this. Plane corner which
  corresponds to left bottom image corner has got X/Y axis
  on it (red is for X axis, green for Y).
- Re-adjusting plane corners makes plane to be "re-tracked"
  for the frames sequence between current frame and next
  and previous keyframes.
- Kayframes might be removed from the plane, using Shit-X
  (Marker Delete) operator. However, currently manual
  re-adjustment or "re-track" trigger is needed.
Compositor changes:
- Added new node called Plane Track Deform.
- User selects which plane track to use (for this he need
  to select movie clip datablock, object and track names).
- Node gets an image input, which need to be warped into
  the plane.
- Node outputs:
  * Input image warped into the plane.
  * Plane, rasterized to a mask.
Masking changes:
- Mask points might be parented to a plane track, which
  makes this point deforming in a way as if it belongs
  to the tracked plane.
Some video tutorials are available:
- Coder video: http://www.youtube.com/watch?v=vISEwqNHqe4
- Artist video: https://vimeo.com/71727578
This is mine and Keir's holiday code project :)
											
										 
											2013-08-16 09:46:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     writedata(wd, | 
					
						
							|  |  |  |               DATA, | 
					
						
							|  |  |  |               sizeof(MovieTrackingTrack *) * plane_track->point_tracksnr, | 
					
						
							|  |  |  |               plane_track->point_tracks); | 
					
						
							|  |  |  |     writestruct(wd, DATA, MovieTrackingPlaneMarker, plane_track->markersnr, plane_track->markers); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Merge plane track feature from tomato branch
This commit includes all the changes made for plane tracker
in tomato branch.
Movie clip editor changes:
- Artist might create a plane track out of multiple point
  tracks which belongs to the same track (minimum amount of
  point tracks is 4, maximum is not actually limited).
  When new plane track is added, it's getting "tracked"
  across all point tracks, which makes it stick to the same
  plane point tracks belong to.
- After plane track was added, it need to be manually adjusted
  in a way it covers feature one might to mask/replace.
  General transform tools (G, R, S) or sliding corners with
  a mouse could be sued for this. Plane corner which
  corresponds to left bottom image corner has got X/Y axis
  on it (red is for X axis, green for Y).
- Re-adjusting plane corners makes plane to be "re-tracked"
  for the frames sequence between current frame and next
  and previous keyframes.
- Kayframes might be removed from the plane, using Shit-X
  (Marker Delete) operator. However, currently manual
  re-adjustment or "re-track" trigger is needed.
Compositor changes:
- Added new node called Plane Track Deform.
- User selects which plane track to use (for this he need
  to select movie clip datablock, object and track names).
- Node gets an image input, which need to be warped into
  the plane.
- Node outputs:
  * Input image warped into the plane.
  * Plane, rasterized to a mask.
Masking changes:
- Mask points might be parented to a plane track, which
  makes this point deforming in a way as if it belongs
  to the tracked plane.
Some video tutorials are available:
- Coder video: http://www.youtube.com/watch?v=vISEwqNHqe4
- Artist video: https://vimeo.com/71727578
This is mine and Keir's holiday code project :)
											
										 
											2013-08-16 09:46:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | static void write_movieReconstruction(WriteData *wd, MovieTrackingReconstruction *reconstruction) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (reconstruction->camnr) { | 
					
						
							|  |  |  |     writestruct( | 
					
						
							|  |  |  |         wd, DATA, MovieReconstructedCamera, reconstruction->camnr, reconstruction->cameras); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-12-05 18:57:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_movieclip(WriteData *wd, MovieClip *clip, const void *id_address) | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (clip->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 12:27:24 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     clip->anim = NULL; | 
					
						
							|  |  |  |     clip->tracking_context = NULL; | 
					
						
							|  |  |  |     clip->tracking.stats = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MovieTracking *tracking = &clip->tracking; | 
					
						
							|  |  |  |     MovieTrackingObject *object; | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_MC, MovieClip, 1, id_address, clip); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &clip->id); | 
					
						
							| 
									
										
										
										
											2012-05-08 07:31:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (clip->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, clip->adt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-17 08:13:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_movieTracks(wd, &tracking->tracks); | 
					
						
							|  |  |  |     write_moviePlaneTracks(wd, &tracking->plane_tracks); | 
					
						
							|  |  |  |     write_movieReconstruction(wd, &tracking->reconstruction); | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     object = tracking->objects.first; | 
					
						
							|  |  |  |     while (object) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, MovieTrackingObject, 1, object); | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_movieTracks(wd, &object->tracks); | 
					
						
							|  |  |  |       write_moviePlaneTracks(wd, &object->plane_tracks); | 
					
						
							|  |  |  |       write_movieReconstruction(wd, &object->reconstruction); | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       object = object->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-11-07 12:55:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_mask(WriteData *wd, Mask *mask, const void *id_address) | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (mask->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     MaskLayer *masklay; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_MSK, Mask, 1, id_address, mask); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &mask->id); | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (mask->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, mask->adt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     for (masklay = mask->masklayers.first; masklay; masklay = masklay->next) { | 
					
						
							|  |  |  |       MaskSpline *spline; | 
					
						
							|  |  |  |       MaskLayerShape *masklay_shape; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       writestruct(wd, DATA, MaskLayer, 1, masklay); | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       for (spline = masklay->splines.first; spline; spline = spline->next) { | 
					
						
							|  |  |  |         int i; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         void *points_deform = spline->points_deform; | 
					
						
							|  |  |  |         spline->points_deform = NULL; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         writestruct(wd, DATA, MaskSpline, 1, spline); | 
					
						
							|  |  |  |         writestruct(wd, DATA, MaskSplinePoint, spline->tot_point, spline->points); | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         spline->points_deform = points_deform; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         for (i = 0; i < spline->tot_point; i++) { | 
					
						
							|  |  |  |           MaskSplinePoint *point = &spline->points[i]; | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           if (point->tot_uw) { | 
					
						
							|  |  |  |             writestruct(wd, DATA, MaskSplinePointUW, point->tot_uw, point->uw); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       for (masklay_shape = masklay->splines_shapes.first; masklay_shape; | 
					
						
							|  |  |  |            masklay_shape = masklay_shape->next) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, MaskLayerShape, 1, masklay_shape); | 
					
						
							|  |  |  |         writedata(wd, | 
					
						
							|  |  |  |                   DATA, | 
					
						
							|  |  |  |                   masklay_shape->tot_vert * sizeof(float) * MASK_OBJECT_SHAPE_ELEM_SIZE, | 
					
						
							|  |  |  |                   masklay_shape->data); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-06-04 16:42:58 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-24 20:37:54 +00:00
										 |  |  | static void write_linestyle_color_modifiers(WriteData *wd, ListBase *modifiers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   LineStyleModifier *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     int struct_nr; | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_AlongStroke); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_DistanceFromCamera); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_DistanceFromObject); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_Material); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_Tangent); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_NOISE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_Noise); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_CreaseAngle); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleColorModifier_Curvature_3D); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleModifier); /* this should not happen */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct_nr(wd, DATA, struct_nr, 1, m); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, ((LineStyleColorModifier_AlongStroke *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         writestruct( | 
					
						
							|  |  |  |             wd, DATA, ColorBand, 1, ((LineStyleColorModifier_DistanceFromCamera *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         writestruct( | 
					
						
							|  |  |  |             wd, DATA, ColorBand, 1, ((LineStyleColorModifier_DistanceFromObject *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, ((LineStyleColorModifier_Material *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, ((LineStyleColorModifier_Tangent *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_NOISE: | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, ((LineStyleColorModifier_Noise *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         writestruct(wd, DATA, ColorBand, 1, ((LineStyleColorModifier_CreaseAngle *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         writestruct( | 
					
						
							|  |  |  |             wd, DATA, ColorBand, 1, ((LineStyleColorModifier_Curvature_3D *)m)->color_ramp); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-07-24 20:37:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_linestyle_alpha_modifiers(WriteData *wd, ListBase *modifiers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   LineStyleModifier *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     int struct_nr; | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_AlongStroke); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_DistanceFromCamera); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_DistanceFromObject); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_Material); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_Tangent); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_NOISE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_Noise); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_CreaseAngle); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleAlphaModifier_Curvature_3D); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleModifier); /* this should not happen */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct_nr(wd, DATA, struct_nr, 1, m); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_AlongStroke *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_DistanceFromCamera *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_DistanceFromObject *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_Material *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_Tangent *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_NOISE: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_Noise *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_CreaseAngle *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleAlphaModifier_Curvature_3D *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-07-24 20:37:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_linestyle_thickness_modifiers(WriteData *wd, ListBase *modifiers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   LineStyleModifier *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     int struct_nr; | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_AlongStroke); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_DistanceFromCamera); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_DistanceFromObject); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_Material); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CALLIGRAPHY: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_Calligraphy); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_Tangent); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_NOISE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_Noise); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_CreaseAngle); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleThicknessModifier_Curvature_3D); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleModifier); /* this should not happen */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct_nr(wd, DATA, struct_nr, 1, m); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_ALONG_STROKE: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_AlongStroke *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_CAMERA: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_DistanceFromCamera *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_DISTANCE_FROM_OBJECT: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_DistanceFromObject *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_MATERIAL: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_Material *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TANGENT: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_Tangent *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CREASE_ANGLE: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_CreaseAngle *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_CURVATURE_3D: | 
					
						
							|  |  |  |         write_curvemapping(wd, ((LineStyleThicknessModifier_Curvature_3D *)m)->curve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-07-24 20:37:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 14:05:11 +00:00
										 |  |  | static void write_linestyle_geometry_modifiers(WriteData *wd, ListBase *modifiers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   LineStyleModifier *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (m = modifiers->first; m; m = m->next) { | 
					
						
							|  |  |  |     int struct_nr; | 
					
						
							|  |  |  |     switch (m->type) { | 
					
						
							|  |  |  |       case LS_MODIFIER_SAMPLING: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_Sampling); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_BEZIER_CURVE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_BezierCurve); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_SINUS_DISPLACEMENT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_SinusDisplacement); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_SPATIAL_NOISE: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_SpatialNoise); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_PERLIN_NOISE_1D: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_PerlinNoise1D); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_PERLIN_NOISE_2D: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_PerlinNoise2D); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_BACKBONE_STRETCHER: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_BackboneStretcher); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_TIP_REMOVER: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_TipRemover); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_POLYGONIZATION: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_Polygonalization); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_GUIDING_LINES: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_GuidingLines); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_BLUEPRINT: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_Blueprint); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_2D_OFFSET: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_2DOffset); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_2D_TRANSFORM: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_2DTransform); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case LS_MODIFIER_SIMPLIFICATION: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleGeometryModifier_Simplification); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         struct_nr = SDNA_TYPE_FROM_STRUCT(LineStyleModifier); /* this should not happen */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     writestruct_nr(wd, DATA, struct_nr, 1, m); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-08-19 14:05:11 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_linestyle(WriteData *wd, FreestyleLineStyle *linestyle, const void *id_address) | 
					
						
							| 
									
										
										
										
											2010-06-25 22:45:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (linestyle->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_LS, FreestyleLineStyle, 1, id_address, linestyle); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     write_iddata(wd, &linestyle->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (linestyle->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, linestyle->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write_linestyle_color_modifiers(wd, &linestyle->color_modifiers); | 
					
						
							|  |  |  |     write_linestyle_alpha_modifiers(wd, &linestyle->alpha_modifiers); | 
					
						
							|  |  |  |     write_linestyle_thickness_modifiers(wd, &linestyle->thickness_modifiers); | 
					
						
							|  |  |  |     write_linestyle_geometry_modifiers(wd, &linestyle->geometry_modifiers); | 
					
						
							|  |  |  |     for (int a = 0; a < MAX_MTEX; a++) { | 
					
						
							|  |  |  |       if (linestyle->mtex[a]) { | 
					
						
							|  |  |  |         writestruct(wd, DATA, MTex, 1, linestyle->mtex[a]); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (linestyle->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, linestyle->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, linestyle->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-06-25 22:45:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_cachefile(WriteData *wd, CacheFile *cache_file, const void *id_address) | 
					
						
							| 
									
										
										
											
												Basic Alembic support
All in all, this patch adds an Alembic importer, an Alembic exporter,
and a new CacheFile data block which, for now, wraps around an Alembic
archive. This data block is made available through a new modifier ("Mesh
Sequence Cache") as well as a new constraint ("Transform Cache") to
somewhat properly support respectively geometric and transformation data
streaming from alembic caches.
A more in-depth documentation is to be found on the wiki, as well as a
 guide to compile alembic: https://wiki.blender.org/index.php/
User:Kevindietrich/AlembicBasicIo.
Many thanks to everyone involved in this little project, and huge shout
out to "cgstrive" for the thorough testings with Maya, 3ds Max, Houdini
and Realflow as well as @fjuhec, @jensverwiebe and @jasperge for the
custom builds and compile fixes.
Reviewers: sergey, campbellbarton, mont29
Reviewed By: sergey, campbellbarton, mont29
Differential Revision: https://developer.blender.org/D2060
											
										 
											2016-08-06 06:20:37 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (cache_file->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 12:33:29 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     BLI_listbase_clear(&cache_file->object_paths); | 
					
						
							|  |  |  |     cache_file->handle = NULL; | 
					
						
							|  |  |  |     memset(cache_file->handle_filepath, 0, sizeof(cache_file->handle_filepath)); | 
					
						
							|  |  |  |     cache_file->handle_readers = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_CF, CacheFile, 1, id_address, cache_file); | 
					
						
							| 
									
										
										
											
												Basic Alembic support
All in all, this patch adds an Alembic importer, an Alembic exporter,
and a new CacheFile data block which, for now, wraps around an Alembic
archive. This data block is made available through a new modifier ("Mesh
Sequence Cache") as well as a new constraint ("Transform Cache") to
somewhat properly support respectively geometric and transformation data
streaming from alembic caches.
A more in-depth documentation is to be found on the wiki, as well as a
 guide to compile alembic: https://wiki.blender.org/index.php/
User:Kevindietrich/AlembicBasicIo.
Many thanks to everyone involved in this little project, and huge shout
out to "cgstrive" for the thorough testings with Maya, 3ds Max, Houdini
and Realflow as well as @fjuhec, @jensverwiebe and @jasperge for the
custom builds and compile fixes.
Reviewers: sergey, campbellbarton, mont29
Reviewed By: sergey, campbellbarton, mont29
Differential Revision: https://developer.blender.org/D2060
											
										 
											2016-08-06 06:20:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (cache_file->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, cache_file->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
											
												Basic Alembic support
All in all, this patch adds an Alembic importer, an Alembic exporter,
and a new CacheFile data block which, for now, wraps around an Alembic
archive. This data block is made available through a new modifier ("Mesh
Sequence Cache") as well as a new constraint ("Transform Cache") to
somewhat properly support respectively geometric and transformation data
streaming from alembic caches.
A more in-depth documentation is to be found on the wiki, as well as a
 guide to compile alembic: https://wiki.blender.org/index.php/
User:Kevindietrich/AlembicBasicIo.
Many thanks to everyone involved in this little project, and huge shout
out to "cgstrive" for the thorough testings with Maya, 3ds Max, Houdini
and Realflow as well as @fjuhec, @jensverwiebe and @jasperge for the
custom builds and compile fixes.
Reviewers: sergey, campbellbarton, mont29
Reviewed By: sergey, campbellbarton, mont29
Differential Revision: https://developer.blender.org/D2060
											
										 
											2016-08-06 06:20:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:35:03 +02:00
										 |  |  | static void write_workspace(BlendWriter *writer, WorkSpace *workspace, const void *id_address) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_id_struct(writer, WorkSpace, id_address, &workspace->id); | 
					
						
							|  |  |  |   write_iddata(writer->wd, &workspace->id); | 
					
						
							|  |  |  |   BLO_write_struct_list(writer, WorkSpaceLayout, &workspace->layouts); | 
					
						
							|  |  |  |   BLO_write_struct_list(writer, WorkSpaceDataRelation, &workspace->hook_layout_relations); | 
					
						
							|  |  |  |   BLO_write_struct_list(writer, wmOwnerID, &workspace->owner_ids); | 
					
						
							|  |  |  |   BLO_write_struct_list(writer, bToolRef, &workspace->tools); | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (bToolRef *, tref, &workspace->tools) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (tref->properties) { | 
					
						
							| 
									
										
										
										
											2020-06-05 12:35:03 +02:00
										 |  |  |       IDP_WriteProperty_new_api(tref->properties, writer); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												Main Workspace Integration
This commit does the main integration of workspaces, which is a design we agreed on during the 2.8 UI workshop (see https://wiki.blender.org/index.php/Dev:2.8/UI/Workshop_Writeup)
Workspaces should generally be stable, I'm not aware of any remaining bugs (or I've forgotten them :) ). If you find any, let me know!
(Exception: mode switching button might get out of sync with actual mode in some cases, would consider that a limitation/ToDo. Needs to be resolved at some point.)
== Main Changes/Features
* Introduces the new Workspaces as data-blocks.
* Allow storing a number of custom workspaces as part of the user configuration. Needs further work to allow adding and deleting individual workspaces.
* Bundle a default workspace configuration with Blender (current screen-layouts converted to workspaces).
* Pressing button to add a workspace spawns a menu to select between "Duplicate Current" and the workspaces from the user configuration. If no workspaces are stored in the user configuration, the default workspaces are listed instead.
* Store screen-layouts (`bScreen`) per workspace.
* Store an active screen-layout per workspace. Changing the workspace will enable this layout.
* Store active mode in workspace. Changing the workspace will also enter the mode of the new workspace. (Note that we still store the active mode in the object, moving this completely to workspaces is a separate project.)
* Store an active render layer per workspace.
* Moved mode switch from 3D View header to Info Editor header.
* Store active scene in window (not directly workspace related, but overlaps quite a bit).
* Removed 'Use Global Scene' User Preference option.
* Compatibility with old files - a new workspace is created for every screen-layout of old files. Old Blender versions should be able to read files saved with workspace support as well.
* Default .blend only contains one workspace ("General").
* Support appending workspaces.
Opening files without UI and commandline rendering should work fine.
Note that the UI is temporary! We plan to introduce a new global topbar
that contains the workspace options and tabs for switching workspaces.
== Technical Notes
* Workspaces are data-blocks.
* Adding and removing `bScreen`s should be done through `ED_workspace_layout` API now.
* A workspace can be active in multiple windows at the same time.
* The mode menu (which is now in the Info Editor header) doesn't display "Grease Pencil Edit" mode anymore since its availability depends on the active editor. Will be fixed by making Grease Pencil an own object type (as planned).
* The button to change the active workspace object mode may get out of sync with the mode of the active object. Will either be resolved by moving mode out of object data, or we'll disable workspace modes again (there's a `#define USE_WORKSPACE_MODE` for that).
* Screen-layouts (`bScreen`) are IDs and thus stored in a main list-base. Had to add a wrapper `WorkSpaceLayout` so we can store them in a list-base within workspaces, too. On the long run we could completely replace `bScreen` by workspace structs.
* `WorkSpace` types use some special compiler trickery to allow marking structs and struct members as private. BKE_workspace API should be used for accessing those.
* Added scene operators `SCENE_OT_`. Was previously done through screen operators.
== BPY API Changes
* Removed `Screen.scene`, added `Window.scene`
* Removed `UserPreferencesView.use_global_scene`
* Added `Context.workspace`, `Window.workspace` and `BlendData.workspaces`
* Added `bpy.types.WorkSpace` containing `screens`, `object_mode` and `render_layer`
* Added Screen.layout_name for the layout name that'll be displayed in the UI (may differ from internal name)
== What's left?
* There are a few open design questions (T50521). We should find the needed answers and implement them.
* Allow adding and removing individual workspaces from workspace configuration (needs UI design).
* Get the override system ready and support overrides per workspace.
* Support custom UI setups as part of workspaces (hidden panels, hidden buttons, customizable toolbars, etc).
* Allow enabling add-ons per workspace.
* Support custom workspace keymaps.
* Remove special exception for workspaces in linking code (so they're always appended, never linked). Depends on a few things, so best to solve later.
* Get the topbar done.
* Workspaces need a proper icon, current one is just a placeholder :)
Reviewed By: campbellbarton, mont29
Tags: #user_interface, #bf_blender_2.8
Maniphest Tasks: T50521
Differential Revision: https://developer.blender.org/D2451
											
										 
											2017-06-01 19:56:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_hair(WriteData *wd, Hair *hair, const void *id_address) | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (hair->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     CustomDataLayer *players = NULL, players_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomDataLayer *clayers = NULL, clayers_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&hair->pdata, &players, players_buff, ARRAY_SIZE(players_buff)); | 
					
						
							|  |  |  |     CustomData_file_write_prepare(&hair->cdata, &clayers, clayers_buff, ARRAY_SIZE(clayers_buff)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_HA, Hair, 1, id_address, hair); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |     write_iddata(wd, &hair->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Direct data */ | 
					
						
							|  |  |  |     write_customdata(wd, &hair->id, hair->totpoint, &hair->pdata, players, CD_MASK_ALL); | 
					
						
							|  |  |  |     write_customdata(wd, &hair->id, hair->totcurve, &hair->cdata, clayers, CD_MASK_ALL); | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * hair->totcol, hair->mat); | 
					
						
							|  |  |  |     if (hair->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, hair->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Remove temporary data. */ | 
					
						
							|  |  |  |     if (players && players != players_buff) { | 
					
						
							|  |  |  |       MEM_freeN(players); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (clayers && clayers != clayers_buff) { | 
					
						
							|  |  |  |       MEM_freeN(clayers); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_pointcloud(WriteData *wd, PointCloud *pointcloud, const void *id_address) | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (pointcloud->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     CustomDataLayer *players = NULL, players_buff[CD_TEMP_CHUNK_SIZE]; | 
					
						
							|  |  |  |     CustomData_file_write_prepare( | 
					
						
							|  |  |  |         &pointcloud->pdata, &players, players_buff, ARRAY_SIZE(players_buff)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_PT, PointCloud, 1, id_address, pointcloud); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |     write_iddata(wd, &pointcloud->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Direct data */ | 
					
						
							|  |  |  |     write_customdata( | 
					
						
							|  |  |  |         wd, &pointcloud->id, pointcloud->totpoint, &pointcloud->pdata, players, CD_MASK_ALL); | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * pointcloud->totcol, pointcloud->mat); | 
					
						
							|  |  |  |     if (pointcloud->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, pointcloud->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Remove temporary data. */ | 
					
						
							|  |  |  |     if (players && players != players_buff) { | 
					
						
							|  |  |  |       MEM_freeN(players); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | static void write_volume(WriteData *wd, Volume *volume, const void *id_address) | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (volume->id.us > 0 || wd->use_memfile) { | 
					
						
							| 
									
										
										
										
											2020-04-01 12:38:23 +02:00
										 |  |  |     /* Clean up, important in undo case to reduce false detection of changed datablocks. */ | 
					
						
							|  |  |  |     volume->runtime.grids = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |     /* write LibData */ | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |     writestruct_at_address(wd, ID_VO, Volume, 1, id_address, volume); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |     write_iddata(wd, &volume->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* direct data */ | 
					
						
							|  |  |  |     writedata(wd, DATA, sizeof(void *) * volume->totcol, volume->mat); | 
					
						
							|  |  |  |     if (volume->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, volume->adt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (volume->packedfile) { | 
					
						
							|  |  |  |       PackedFile *pf = volume->packedfile; | 
					
						
							|  |  |  |       writestruct(wd, DATA, PackedFile, 1, pf); | 
					
						
							|  |  |  |       writedata(wd, DATA, pf->size, pf->data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-20 10:37:38 +02:00
										 |  |  | static void write_simulation(WriteData *wd, Simulation *simulation) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (simulation->id.us > 0 || wd->use_memfile) { | 
					
						
							|  |  |  |     writestruct(wd, ID_SIM, Simulation, 1, simulation); | 
					
						
							|  |  |  |     write_iddata(wd, &simulation->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (simulation->adt) { | 
					
						
							|  |  |  |       write_animdata(wd, simulation->adt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-20 12:56:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* nodetree is integral part of simulation, no libdata */ | 
					
						
							|  |  |  |     if (simulation->nodetree) { | 
					
						
							|  |  |  |       writestruct(wd, DATA, bNodeTree, 1, simulation->nodetree); | 
					
						
							|  |  |  |       write_nodetree_nolib(wd, simulation->nodetree); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-20 10:37:38 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | /* Keep it last of write_foodata functions. */ | 
					
						
							|  |  |  | static void write_libraries(WriteData *wd, Main *main) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ListBase *lbarray[MAX_LIBARRAY]; | 
					
						
							|  |  |  |   ID *id; | 
					
						
							|  |  |  |   int a, tot; | 
					
						
							|  |  |  |   bool found_one; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (; main; main = main->next) { | 
					
						
							|  |  |  |     a = tot = set_listbasepointers(main, lbarray); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* test: is lib being used */ | 
					
						
							|  |  |  |     if (main->curlib && main->curlib->packedfile) { | 
					
						
							|  |  |  |       found_one = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-25 17:39:16 +02:00
										 |  |  |     else if (wd->use_memfile) { | 
					
						
							|  |  |  |       /* When writing undo step we always write all existing libraries, makes reading undo step
 | 
					
						
							|  |  |  |        * much easier when dealing with purely indirectly used libraries. */ | 
					
						
							|  |  |  |       found_one = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else { | 
					
						
							|  |  |  |       found_one = false; | 
					
						
							|  |  |  |       while (!found_one && tot--) { | 
					
						
							|  |  |  |         for (id = lbarray[tot]->first; id; id = id->next) { | 
					
						
							| 
									
										
										
										
											2019-09-16 14:52:06 +02:00
										 |  |  |           if (id->us > 0 && | 
					
						
							|  |  |  |               ((id->tag & LIB_TAG_EXTERN) || | 
					
						
							|  |  |  |                ((id->tag & LIB_TAG_INDIRECT) && (id->flag & LIB_INDIRECT_WEAK_LINK)))) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             found_one = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 01:10:29 +10:00
										 |  |  |     /* To be able to restore 'quit.blend' and temp saves,
 | 
					
						
							|  |  |  |      * the packed blend has to be in undo buffers... */ | 
					
						
							|  |  |  |     /* XXX needs rethink, just like save UI in undo files now -
 | 
					
						
							|  |  |  |      * would be nice to append things only for the 'quit.blend' and temp saves. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (found_one) { | 
					
						
							|  |  |  |       /* Not overridable. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       writestruct(wd, ID_LI, Library, 1, main->curlib); | 
					
						
							|  |  |  |       write_iddata(wd, &main->curlib->id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (main->curlib->packedfile) { | 
					
						
							|  |  |  |         PackedFile *pf = main->curlib->packedfile; | 
					
						
							|  |  |  |         writestruct(wd, DATA, PackedFile, 1, pf); | 
					
						
							|  |  |  |         writedata(wd, DATA, pf->size, pf->data); | 
					
						
							|  |  |  |         if (wd->use_memfile == false) { | 
					
						
							|  |  |  |           printf("write packed .blend: %s\n", main->curlib->name); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Write link placeholders for all direct linked IDs. */ | 
					
						
							|  |  |  |       while (a--) { | 
					
						
							|  |  |  |         for (id = lbarray[a]->first; id; id = id->next) { | 
					
						
							| 
									
										
										
										
											2019-09-16 14:52:06 +02:00
										 |  |  |           if (id->us > 0 && | 
					
						
							|  |  |  |               ((id->tag & LIB_TAG_EXTERN) || | 
					
						
							|  |  |  |                ((id->tag & LIB_TAG_INDIRECT) && (id->flag & LIB_INDIRECT_WEAK_LINK)))) { | 
					
						
							| 
									
										
										
										
											2020-03-19 19:37:00 +01:00
										 |  |  |             if (!BKE_idtype_idcode_is_linkable(GS(id->name))) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |               printf( | 
					
						
							|  |  |  |                   "ERROR: write file: data-block '%s' from lib '%s' is not linkable " | 
					
						
							|  |  |  |                   "but is flagged as directly linked", | 
					
						
							|  |  |  |                   id->name, | 
					
						
							|  |  |  |                   main->curlib->filepath); | 
					
						
							|  |  |  |               BLI_assert(0); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             writestruct(wd, ID_LINK_PLACEHOLDER, ID, 1, id); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mywrite_flush(wd); | 
					
						
							| 
									
										
										
										
											2016-06-14 14:53:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:53:37 +00:00
										 |  |  | /* context is usually defined by WM, two cases where no WM is available:
 | 
					
						
							| 
									
										
											  
											
												
Merge of trunk into blender 2.5:
svn merge https://svn.blender.org/svnroot/bf-blender/trunk/blender -r12987:17416
Issues:
* GHOST/X11 had conflicting changes. Some code was added in 2.5, which was
  later added in trunk also, but reverted partially, specifically revision
  16683. I have left out this reversion in the 2.5 branch since I think it is
  needed there.
  http://projects.blender.org/plugins/scmsvn/viewcvs.php?view=rev&root=bf-blender&revision=16683
* Scons had various conflicting changes, I decided to go with trunk version
  for everything except priorities and some library renaming.
* In creator.c, there were various fixes and fixes for fixes related to the -w
  -W and -p options. In 2.5 -w and -W is not coded yet, and -p is done
  differently. Since this is changed so much, and I don't think those fixes
  would be needed in 2.5, I've left them out.
* Also in creator.c: there was code for a python bugfix where the screen was not
  initialized when running with -P. The code that initializes the screen there
  I had to disable, that can't work in 2.5 anymore but left it commented as a
  reminder.
Further I had to disable some new function calls. using src/ and python/, as
was done already in this branch, disabled function calls:
* bpath.c: error reporting
* BME_conversions.c: editmesh conversion functions.
* SHD_dynamic: disabled almost completely, there is no python/.
* KX_PythonInit.cpp and Ketsji/ build files: Mathutils is not there, disabled.
* text.c: clipboard copy call.
* object.c: OB_SUPPORT_MATERIAL.
* DerivedMesh.c and subsurf_ccg, stipple_quarttone.
Still to be done:
* Go over files and functions that were moved to a different location but could
  still use changes that were done in trunk.
											
										 
											2008-11-12 21:16:53 +00:00
										 |  |  |  * - for forward compatibility, curscreen has to be saved | 
					
						
							|  |  |  |  * - for undofile, curscene needs to be saved */ | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | static void write_global(WriteData *wd, int fileflags, Main *mainvar) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const bool is_undo = wd->use_memfile; | 
					
						
							|  |  |  |   FileGlobal fg; | 
					
						
							|  |  |  |   bScreen *screen; | 
					
						
							|  |  |  |   Scene *scene; | 
					
						
							|  |  |  |   ViewLayer *view_layer; | 
					
						
							|  |  |  |   char subvstr[8]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* prevent mem checkers from complaining */ | 
					
						
							|  |  |  |   memset(fg._pad, 0, sizeof(fg._pad)); | 
					
						
							|  |  |  |   memset(fg.filename, 0, sizeof(fg.filename)); | 
					
						
							|  |  |  |   memset(fg.build_hash, 0, sizeof(fg.build_hash)); | 
					
						
							|  |  |  |   fg._pad1 = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   current_screen_compat(mainvar, is_undo, &screen, &scene, &view_layer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* XXX still remap G */ | 
					
						
							|  |  |  |   fg.curscreen = screen; | 
					
						
							|  |  |  |   fg.curscene = scene; | 
					
						
							|  |  |  |   fg.cur_view_layer = view_layer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* prevent to save this, is not good convention, and feature with concerns... */ | 
					
						
							|  |  |  |   fg.fileflags = (fileflags & ~G_FILE_FLAG_ALL_RUNTIME); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fg.globalf = G.f; | 
					
						
							|  |  |  |   BLI_strncpy(fg.filename, mainvar->name, sizeof(fg.filename)); | 
					
						
							| 
									
										
										
											
												Blender: change bugfix release versioning from a/b/c to .1/.2/.3
The file subversion is no longer used in the Python API or user interface,
and is now internal to Blender.
User interface, Python API and file I/O metadata now use more consistent
formatting for version numbers. Official releases use "2.83.0", "2.83.1",
and releases under development use "2.90.0 Alpha", "2.90.0 Beta".
Some Python add-ons may need to lower the Blender version in bl_info to
(2, 83, 0) or (2, 90, 0) if they used a subversion number higher than 0.
https://wiki.blender.org/wiki/Reference/Release_Notes/2.83/Python_API#Compatibility
This change is in preparation of LTS releases, and also brings us more
in line with semantic versioning.
Fixes T76058.
Differential Revision: https://developer.blender.org/D7748
											
										 
											2020-05-25 10:49:04 +02:00
										 |  |  |   sprintf(subvstr, "%4d", BLENDER_FILE_SUBVERSION); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   memcpy(fg.subvstr, subvstr, 4); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												Blender: change bugfix release versioning from a/b/c to .1/.2/.3
The file subversion is no longer used in the Python API or user interface,
and is now internal to Blender.
User interface, Python API and file I/O metadata now use more consistent
formatting for version numbers. Official releases use "2.83.0", "2.83.1",
and releases under development use "2.90.0 Alpha", "2.90.0 Beta".
Some Python add-ons may need to lower the Blender version in bl_info to
(2, 83, 0) or (2, 90, 0) if they used a subversion number higher than 0.
https://wiki.blender.org/wiki/Reference/Release_Notes/2.83/Python_API#Compatibility
This change is in preparation of LTS releases, and also brings us more
in line with semantic versioning.
Fixes T76058.
Differential Revision: https://developer.blender.org/D7748
											
										 
											2020-05-25 10:49:04 +02:00
										 |  |  |   fg.subversion = BLENDER_FILE_SUBVERSION; | 
					
						
							|  |  |  |   fg.minversion = BLENDER_FILE_MIN_VERSION; | 
					
						
							|  |  |  |   fg.minsubversion = BLENDER_FILE_MIN_SUBVERSION; | 
					
						
							| 
									
										
										
										
											2011-08-22 16:54:26 +00:00
										 |  |  | #ifdef WITH_BUILDINFO
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     extern unsigned long build_commit_timestamp; | 
					
						
							|  |  |  |     extern char build_hash[]; | 
					
						
							|  |  |  |     /* TODO(sergey): Add branch name to file as well? */ | 
					
						
							|  |  |  |     fg.build_commit_timestamp = build_commit_timestamp; | 
					
						
							|  |  |  |     BLI_strncpy(fg.build_hash, build_hash, sizeof(fg.build_hash)); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-01-02 13:33:32 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   fg.build_commit_timestamp = 0; | 
					
						
							|  |  |  |   BLI_strncpy(fg.build_hash, "unknown", sizeof(fg.build_hash)); | 
					
						
							| 
									
										
										
										
											2011-01-02 13:33:32 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   writestruct(wd, GLOB, FileGlobal, 1, &fg); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | /* preview image, first 2 values are width and height
 | 
					
						
							| 
									
										
										
										
											2018-11-30 14:51:16 +11:00
										 |  |  |  * second are an RGBA image (uchar) | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  |  * note, this uses 'TEST' since new types will segfault on file load for older blender versions. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
											  
											
												Make .blend file thumbnail reading simpler and more coherent, read/store them when reading in background mode.
Primary goal of this commit is to fix an annoying issue - when processing and saving .blend
files in background mode you lose their thumbnails, since it can only be generated with
an OpenGL context.
Solution to that is to read .blend thumbnail while reading .blend file (only done in background
mode currently), and store it in Main struct.
Also, this lead to removing .blend file reading code from thumb_blend (no need to have doublons).
We now have a small interface in regular reading code area, which keeps it reasonbaly light
by only reading/parsing header info, and first few BHead blocks.
This makes code reading .blend thumbnail about 3 to 4 times slower than previous highly specialized
one in blend_thumb.c, but overall thumbnail generation of a big .blend files folder only grows
of about 1%, think we can bare with it.
Finally, since thumbnail is now optionally stored in Main struct, it makes it easy to allow user
to define their own custom one (instead of auto-generated one). RNA API for this was not added though,
accessing that kind of .blend meta-data has to be rethought a bit on a bigger level first.
Reviewers: sergey, campbellbarton
Subscribers: Severin, psy-fi
Differential Revision: https://developer.blender.org/D1469
											
										 
											2015-08-27 15:53:23 +02:00
										 |  |  | static void write_thumb(WriteData *wd, const BlendThumbnail *thumb) | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (thumb) { | 
					
						
							|  |  |  |     writedata(wd, TEST, BLEN_THUMB_MEMSIZE_FILE(thumb->width, thumb->height), thumb); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name File Writing (Private)
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-25 13:07:28 +00:00
										 |  |  | /* if MemFile * there's filesave to memory */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool write_file_handle(Main *mainvar, | 
					
						
							|  |  |  |                               WriteWrap *ww, | 
					
						
							|  |  |  |                               MemFile *compare, | 
					
						
							|  |  |  |                               MemFile *current, | 
					
						
							|  |  |  |                               int write_flags, | 
					
						
							|  |  |  |                               const BlendThumbnail *thumb) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BHead bhead; | 
					
						
							|  |  |  |   ListBase mainlist; | 
					
						
							|  |  |  |   char buf[16]; | 
					
						
							|  |  |  |   WriteData *wd; | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   blo_split_main(&mainlist, mainvar); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wd = mywrite_begin(ww, compare, current); | 
					
						
							| 
									
										
										
										
											2011-12-27 13:17:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   sprintf(buf, | 
					
						
							|  |  |  |           "BLENDER%c%c%.3d", | 
					
						
							|  |  |  |           (sizeof(void *) == 8) ? '-' : '_', | 
					
						
							|  |  |  |           (ENDIAN_ORDER == B_ENDIAN) ? 'V' : 'v', | 
					
						
							| 
									
										
										
											
												Blender: change bugfix release versioning from a/b/c to .1/.2/.3
The file subversion is no longer used in the Python API or user interface,
and is now internal to Blender.
User interface, Python API and file I/O metadata now use more consistent
formatting for version numbers. Official releases use "2.83.0", "2.83.1",
and releases under development use "2.90.0 Alpha", "2.90.0 Beta".
Some Python add-ons may need to lower the Blender version in bl_info to
(2, 83, 0) or (2, 90, 0) if they used a subversion number higher than 0.
https://wiki.blender.org/wiki/Reference/Release_Notes/2.83/Python_API#Compatibility
This change is in preparation of LTS releases, and also brings us more
in line with semantic versioning.
Fixes T76058.
Differential Revision: https://developer.blender.org/D7748
											
										 
											2020-05-25 10:49:04 +02:00
										 |  |  |           BLENDER_FILE_VERSION); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   mywrite(wd, buf, 12); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   write_renderinfo(wd, mainvar); | 
					
						
							|  |  |  |   write_thumb(wd, thumb); | 
					
						
							|  |  |  |   write_global(wd, write_flags, mainvar); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* The windowmanager and screen often change,
 | 
					
						
							|  |  |  |    * avoid thumbnail detecting changes because of this. */ | 
					
						
							|  |  |  |   mywrite_flush(wd); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:16:04 +02:00
										 |  |  |   OverrideLibraryStorage *override_storage = | 
					
						
							| 
									
										
										
										
											2020-02-10 18:05:19 +01:00
										 |  |  |       wd->use_memfile ? NULL : BKE_lib_override_library_operations_store_initialize(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | #define ID_BUFFER_STATIC_SIZE 8192
 | 
					
						
							| 
									
										
										
										
											2019-06-12 09:04:10 +10:00
										 |  |  |   /* This outer loop allows to save first data-blocks from real mainvar,
 | 
					
						
							| 
									
										
										
										
											2019-04-22 01:10:29 +10:00
										 |  |  |    * then the temp ones from override process, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |    * if needed, without duplicating whole code. */ | 
					
						
							|  |  |  |   Main *bmain = mainvar; | 
					
						
							|  |  |  |   do { | 
					
						
							|  |  |  |     ListBase *lbarray[MAX_LIBARRAY]; | 
					
						
							|  |  |  |     int a = set_listbasepointers(bmain, lbarray); | 
					
						
							|  |  |  |     while (a--) { | 
					
						
							|  |  |  |       ID *id = lbarray[a]->first; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |       if (id == NULL || GS(id->name) == ID_LI) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         continue; /* Libraries are handled separately below. */ | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |       char id_buffer_static[ID_BUFFER_STATIC_SIZE]; | 
					
						
							|  |  |  |       void *id_buffer = id_buffer_static; | 
					
						
							|  |  |  |       const size_t idtype_struct_size = BKE_idtype_get_info_from_id(id)->struct_size; | 
					
						
							|  |  |  |       if (idtype_struct_size > ID_BUFFER_STATIC_SIZE) { | 
					
						
							|  |  |  |         BLI_assert(0); | 
					
						
							|  |  |  |         id_buffer = MEM_mallocN(idtype_struct_size, __func__); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       for (; id; id = id->next) { | 
					
						
							| 
									
										
										
										
											2019-04-22 01:10:29 +10:00
										 |  |  |         /* We should never attempt to write non-regular IDs
 | 
					
						
							|  |  |  |          * (i.e. all kind of temp/runtime ones). */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         BLI_assert( | 
					
						
							|  |  |  |             (id->tag & (LIB_TAG_NO_MAIN | LIB_TAG_NO_USER_REFCOUNT | LIB_TAG_NOT_ALLOCATED)) == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:16:04 +02:00
										 |  |  |         const bool do_override = !ELEM(override_storage, NULL, bmain) && id->override_library; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (do_override) { | 
					
						
							| 
									
										
										
										
											2020-02-10 18:05:19 +01:00
										 |  |  |           BKE_lib_override_library_operations_store_start(bmain, override_storage, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 17:38:34 +02:00
										 |  |  |         if (wd->use_memfile) { | 
					
						
							|  |  |  |           /* Record the changes that happened up to this undo push in
 | 
					
						
							|  |  |  |            * recalc_up_to_undo_push, and clear recalc_after_undo_push again | 
					
						
							|  |  |  |            * to start accumulating for the next undo push. */ | 
					
						
							|  |  |  |           id->recalc_up_to_undo_push = id->recalc_after_undo_push; | 
					
						
							|  |  |  |           id->recalc_after_undo_push = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           bNodeTree *nodetree = ntreeFromID(id); | 
					
						
							|  |  |  |           if (nodetree != NULL) { | 
					
						
							|  |  |  |             nodetree->id.recalc_up_to_undo_push = nodetree->id.recalc_after_undo_push; | 
					
						
							|  |  |  |             nodetree->id.recalc_after_undo_push = 0; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (GS(id->name) == ID_SCE) { | 
					
						
							|  |  |  |             Scene *scene = (Scene *)id; | 
					
						
							|  |  |  |             if (scene->master_collection != NULL) { | 
					
						
							|  |  |  |               scene->master_collection->id.recalc_up_to_undo_push = | 
					
						
							|  |  |  |                   scene->master_collection->id.recalc_after_undo_push; | 
					
						
							|  |  |  |               scene->master_collection->id.recalc_after_undo_push = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |         mywrite_id_begin(wd, id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |         memcpy(id_buffer, id, idtype_struct_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ((ID *)id_buffer)->tag = 0; | 
					
						
							| 
									
										
										
										
											2020-06-05 14:34:00 +10:00
										 |  |  |         /* Those listbase data change every time we add/remove an ID, and also often when renaming
 | 
					
						
							| 
									
										
										
										
											2020-05-29 14:30:02 +02:00
										 |  |  |          * one (due to re-sorting). This avoids generating a lot of false 'is changed' detections | 
					
						
							|  |  |  |          * between undo steps. */ | 
					
						
							|  |  |  |         ((ID *)id_buffer)->prev = NULL; | 
					
						
							|  |  |  |         ((ID *)id_buffer)->next = NULL; | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  |         BlendWriter writer = {wd}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         switch ((ID_Type)GS(id->name)) { | 
					
						
							|  |  |  |           case ID_WM: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:29:40 +02:00
										 |  |  |             write_windowmanager(&writer, (wmWindowManager *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_WS: | 
					
						
							| 
									
										
										
										
											2020-06-05 12:35:03 +02:00
										 |  |  |             write_workspace(&writer, (WorkSpace *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_SCR: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_screen(wd, (bScreen *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_MC: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_movieclip(wd, (MovieClip *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_MSK: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_mask(wd, (Mask *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_SCE: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_scene(wd, (Scene *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_CU: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_curve(wd, (Curve *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_MB: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_mball(wd, (MetaBall *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_IM: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_image(wd, (Image *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_CA: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_camera(wd, (Camera *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_LA: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_light(wd, (Light *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_LT: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_lattice(wd, (Lattice *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_VF: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_vfont(wd, (VFont *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_KE: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_key(wd, (Key *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_WO: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_world(wd, (World *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_TXT: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_text(wd, (Text *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_SPK: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_speaker(wd, (Speaker *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_LP: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_probe(wd, (LightProbe *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_SO: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_sound(wd, (bSound *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_GR: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_collection(wd, (Collection *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_AR: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_armature(wd, (bArmature *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_AC: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_action(wd, (bAction *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_OB: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_object(wd, (Object *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_MA: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_material(wd, (Material *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_TE: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_texture(wd, (Tex *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_ME: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_mesh(wd, (Mesh *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_PA: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_particlesettings(wd, (ParticleSettings *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_NT: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_nodetree(wd, (bNodeTree *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_BR: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_brush(wd, (Brush *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_PAL: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_palette(wd, (Palette *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_PC: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_paintcurve(wd, (PaintCurve *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_GD: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_gpencil(wd, (bGPdata *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_LS: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_linestyle(wd, (FreestyleLineStyle *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_CF: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_cachefile(wd, (CacheFile *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |           case ID_HA: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_hair(wd, (Hair *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_PT: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_pointcloud(wd, (PointCloud *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |           case ID_VO: | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |             write_volume(wd, (Volume *)id_buffer, id); | 
					
						
							| 
									
										
										
										
											2020-03-17 14:41:48 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2020-04-20 10:37:38 +02:00
										 |  |  |           case ID_SIM: | 
					
						
							|  |  |  |             write_simulation(wd, (Simulation *)id); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           case ID_LI: | 
					
						
							|  |  |  |             /* Do nothing, handled below - and should never be reached. */ | 
					
						
							|  |  |  |             BLI_assert(0); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case ID_IP: | 
					
						
							|  |  |  |             /* Do nothing, deprecated. */ | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           default: | 
					
						
							|  |  |  |             /* Should never be reached. */ | 
					
						
							|  |  |  |             BLI_assert(0); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (do_override) { | 
					
						
							| 
									
										
										
										
											2020-02-10 18:05:19 +01:00
										 |  |  |           BKE_lib_override_library_operations_store_end(override_storage, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-17 12:29:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:07:45 +02:00
										 |  |  |         mywrite_id_end(wd, id); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:40:11 +02:00
										 |  |  |       if (id_buffer != id_buffer_static) { | 
					
						
							|  |  |  |         MEM_SAFE_FREE(id_buffer); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       mywrite_flush(wd); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } while ((bmain != override_storage) && (bmain = override_storage)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (override_storage) { | 
					
						
							| 
									
										
										
										
											2020-02-10 18:05:19 +01:00
										 |  |  |     BKE_lib_override_library_operations_store_finalize(override_storage); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     override_storage = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Special handling, operating over split Mains... */ | 
					
						
							|  |  |  |   write_libraries(wd, mainvar->next); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* So changes above don't cause a 'DNA1' to be detected as changed on undo. */ | 
					
						
							|  |  |  |   mywrite_flush(wd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (write_flags & G_FILE_USERPREFS) { | 
					
						
							|  |  |  |     write_userdef(wd, &U); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Write DNA last, because (to be implemented) test for which structs are written.
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Note that we *borrow* the pointer to 'DNAstr', | 
					
						
							|  |  |  |    * so writing each time uses the same address and doesn't cause unnecessary undo overhead. */ | 
					
						
							|  |  |  |   writedata(wd, DNA1, wd->sdna->data_len, wd->sdna->data); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* end of file */ | 
					
						
							|  |  |  |   memset(&bhead, 0, sizeof(BHead)); | 
					
						
							|  |  |  |   bhead.code = ENDB; | 
					
						
							|  |  |  |   mywrite(wd, &bhead, sizeof(BHead)); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   blo_join_main(&mainlist); | 
					
						
							| 
									
										
										
										
											2004-06-23 18:22:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return mywrite_end(wd); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-02 12:44:59 +00:00
										 |  |  | /* do reverse file history: .blend1 -> .blend2, .blend -> .blend1 */ | 
					
						
							|  |  |  | /* return: success(0), failure(1) */ | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | static bool do_history(const char *name, ReportList *reports) | 
					
						
							| 
									
										
										
										
											2011-06-02 12:44:59 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char tempname1[FILE_MAX], tempname2[FILE_MAX]; | 
					
						
							|  |  |  |   int hisnr = U.versions; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (U.versions == 0) { | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (strlen(name) < 2) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_ERROR, "Unable to make version backup: filename too short"); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (hisnr > 1) { | 
					
						
							|  |  |  |     BLI_snprintf(tempname1, sizeof(tempname1), "%s%d", name, hisnr - 1); | 
					
						
							|  |  |  |     if (BLI_exists(tempname1)) { | 
					
						
							|  |  |  |       BLI_snprintf(tempname2, sizeof(tempname2), "%s%d", name, hisnr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (BLI_rename(tempname1, tempname2)) { | 
					
						
							|  |  |  |         BKE_report(reports, RPT_ERROR, "Unable to make version backup"); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     hisnr--; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* is needed when hisnr==1 */ | 
					
						
							|  |  |  |   if (BLI_exists(name)) { | 
					
						
							|  |  |  |     BLI_snprintf(tempname1, sizeof(tempname1), "%s%d", name, hisnr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (BLI_rename(name, tempname1)) { | 
					
						
							|  |  |  |       BKE_report(reports, RPT_ERROR, "Unable to make version backup"); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2011-06-02 12:44:59 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name File Writing (Public)
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-03 13:35:21 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \return Success. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | bool BLO_write_file(Main *mainvar, | 
					
						
							|  |  |  |                     const char *filepath, | 
					
						
							|  |  |  |                     int write_flags, | 
					
						
							|  |  |  |                     ReportList *reports, | 
					
						
							|  |  |  |                     const BlendThumbnail *thumb) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char tempname[FILE_MAX + 1]; | 
					
						
							|  |  |  |   eWriteWrapType ww_type; | 
					
						
							|  |  |  |   WriteWrap ww; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* path backup/restore */ | 
					
						
							|  |  |  |   void *path_list_backup = NULL; | 
					
						
							|  |  |  |   const int path_list_flag = (BKE_BPATH_TRAVERSE_SKIP_LIBRARY | BKE_BPATH_TRAVERSE_SKIP_MULTIFILE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (G.debug & G_DEBUG_IO && mainvar->lock != NULL) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_INFO, "Checking sanity of current .blend file *BEFORE* save to disk"); | 
					
						
							|  |  |  |     BLO_main_validate_libraries(mainvar, reports); | 
					
						
							|  |  |  |     BLO_main_validate_shapekeys(mainvar, reports); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* open temporary file, so we preserve the original in case we crash */ | 
					
						
							|  |  |  |   BLI_snprintf(tempname, sizeof(tempname), "%s@", filepath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (write_flags & G_FILE_COMPRESS) { | 
					
						
							|  |  |  |     ww_type = WW_WRAP_ZLIB; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     ww_type = WW_WRAP_NONE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ww_handle_init(ww_type, &ww); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ww.open(&ww, tempname) == false) { | 
					
						
							|  |  |  |     BKE_reportf( | 
					
						
							|  |  |  |         reports, RPT_ERROR, "Cannot open file %s for writing: %s", tempname, strerror(errno)); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |   /* Remapping of relative paths to new file location. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (write_flags & G_FILE_RELATIVE_REMAP) { | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |     char dir_src[FILE_MAX]; | 
					
						
							|  |  |  |     char dir_dst[FILE_MAX]; | 
					
						
							|  |  |  |     BLI_split_dir_part(mainvar->name, dir_src, sizeof(dir_src)); | 
					
						
							|  |  |  |     BLI_split_dir_part(filepath, dir_dst, sizeof(dir_dst)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 00:35:55 +11:00
										 |  |  |     /* Just in case there is some subtle difference. */ | 
					
						
							| 
									
										
										
										
											2020-04-07 12:02:21 +10:00
										 |  |  |     BLI_path_normalize(mainvar->name, dir_dst); | 
					
						
							|  |  |  |     BLI_path_normalize(mainvar->name, dir_src); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |     if (G.relbase_valid && (BLI_path_cmp(dir_dst, dir_src) == 0)) { | 
					
						
							|  |  |  |       /* Saved to same path. Nothing to do. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       write_flags &= ~G_FILE_RELATIVE_REMAP; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |       /* Check if we need to backup and restore paths. */ | 
					
						
							|  |  |  |       if (UNLIKELY(G_FILE_SAVE_COPY & write_flags)) { | 
					
						
							|  |  |  |         path_list_backup = BKE_bpath_list_backup(mainvar, path_list_flag); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (G.relbase_valid) { | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |         /* Saved, make relative paths relative to new location (if possible). */ | 
					
						
							|  |  |  |         BKE_bpath_relative_rebase(mainvar, dir_src, dir_dst, NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* Unsaved, make all relative. */ | 
					
						
							|  |  |  |         BKE_bpath_relative_convert(mainvar, dir_dst, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* actual file writing */ | 
					
						
							|  |  |  |   const bool err = write_file_handle(mainvar, &ww, NULL, NULL, write_flags, thumb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ww.close(&ww); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (UNLIKELY(path_list_backup)) { | 
					
						
							|  |  |  |     BKE_bpath_list_restore(mainvar, path_list_flag, path_list_backup); | 
					
						
							|  |  |  |     BKE_bpath_list_free(path_list_backup); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (err) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_ERROR, strerror(errno)); | 
					
						
							|  |  |  |     remove(tempname); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* file save to temporary file was successful */ | 
					
						
							|  |  |  |   /* now do reverse file history (move .blend1 -> .blend2, .blend -> .blend1) */ | 
					
						
							|  |  |  |   if (write_flags & G_FILE_HISTORY) { | 
					
						
							|  |  |  |     const bool err_hist = do_history(filepath, reports); | 
					
						
							|  |  |  |     if (err_hist) { | 
					
						
							|  |  |  |       BKE_report(reports, RPT_ERROR, "Version backup failed (file saved with @)"); | 
					
						
							|  |  |  |       return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (BLI_rename(tempname, filepath) != 0) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_ERROR, "Cannot change old file (file saved with @)"); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (G.debug & G_DEBUG_IO && mainvar->lock != NULL) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_INFO, "Checking sanity of current .blend file *AFTER* save to disk"); | 
					
						
							|  |  |  |     BLO_main_validate_libraries(mainvar, reports); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 1; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-03 13:35:21 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \return Success. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool BLO_write_file_mem(Main *mainvar, MemFile *compare, MemFile *current, int write_flags) | 
					
						
							| 
									
										
										
										
											2004-09-05 13:43:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   write_flags &= ~G_FILE_USERPREFS; | 
					
						
							| 
									
										
										
										
											2004-09-05 13:43:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const bool err = write_file_handle(mainvar, NULL, compare, current, write_flags, NULL); | 
					
						
							| 
									
										
											  
											
												Holiday coding log :)
Nice formatted version (pictures soon):
http://wiki.blender.org/index.php/Dev:Ref/Release_Notes/2.66/Usability
Short list of main changes:
- Transparent region option (over main region), added code to blend in/out such panels.
- Min size window now 640 x 480
- Fixed DPI for ui - lots of cleanup and changes everywhere. Icon image need correct size still, layer-in-use icon needs remake.
- Macbook retina support, use command line --no-native-pixels to disable it
- Timeline Marker label was drawing wrong
- Trackpad and magic mouse: supports zoom (hold ctrl)
- Fix for splash position: removed ghost function and made window size update after creation immediate
- Fast undo buffer save now adds UI as well. Could be checked for regular file save even...
  Quit.blend and temp file saving use this now.
- Dixed filename in window on reading quit.blend or temp saves, and they now add a warning in window title: "(Recovered)"
- New Userpref option "Keep Session" - this always saves quit.blend, and loads on start.
  This allows keeping UI and data without actual saves, until you actually save.
  When you load startup.blend and quit, it recognises the quit.blend as a startup (no file name in header)
- Added 3D view copy/paste buffers (selected objects). Shortcuts ctrl-c, ctrl-v (OSX, cmd-c, cmd-v). 
  Coded partial file saving for it. Could be used for other purposes. Todo: use OS clipboards. 
- User preferences (themes, keymaps, user settings) now can be saved as a separate file.
  Old option is called "Save Startup File" the new one "Save User Settings".
  To visualise this difference, the 'save startup file' button has been removed from user preferences window. That option is available as CTRL+U and in File menu still.
- OSX: fixed bug that stopped giving mouse events outside window.
  This also fixes "Continuous Grab" for OSX. (error since 2009)
											
										 
											2012-12-12 18:58:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (err == 0); | 
					
						
							| 
									
										
										
										
											2016-03-03 13:35:21 +11:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 11:44:36 +02:00
										 |  |  | void BLO_write_raw(BlendWriter *writer, int size_in_bytes, const void *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writedata(writer->wd, DATA, size_in_bytes, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_by_name(BlendWriter *writer, const char *struct_name, const void *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int struct_id = BLO_get_struct_id_by_name(writer, struct_name); | 
					
						
							|  |  |  |   BLO_write_struct_by_id(writer, struct_id, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_array_by_name(BlendWriter *writer, | 
					
						
							|  |  |  |                                     const char *struct_name, | 
					
						
							|  |  |  |                                     int array_size, | 
					
						
							|  |  |  |                                     const void *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int struct_id = BLO_get_struct_id_by_name(writer, struct_name); | 
					
						
							|  |  |  |   BLO_write_struct_array_by_id(writer, struct_id, array_size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_by_id(BlendWriter *writer, int struct_id, const void *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writestruct_nr(writer->wd, DATA, struct_id, 1, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_array_by_id(BlendWriter *writer, | 
					
						
							|  |  |  |                                   int struct_id, | 
					
						
							|  |  |  |                                   int array_size, | 
					
						
							|  |  |  |                                   const void *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writestruct_nr(writer->wd, DATA, struct_id, array_size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_list_by_id(BlendWriter *writer, int struct_id, ListBase *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writelist_nr(writer->wd, DATA, struct_id, list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_struct_list_by_name(BlendWriter *writer, const char *struct_name, ListBase *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_struct_list_by_id(writer, BLO_get_struct_id_by_name(writer, struct_name), list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void blo_write_id_struct(BlendWriter *writer, int struct_id, const void *id_address, const ID *id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   writestruct_at_address_nr(writer->wd, GS(id->name), struct_id, 1, id_address, id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int BLO_get_struct_id_by_name(BlendWriter *writer, const char *struct_name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int struct_id = DNA_struct_find_nr(writer->wd->sdna, struct_name); | 
					
						
							|  |  |  |   BLI_assert(struct_id >= 0); | 
					
						
							|  |  |  |   return struct_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_int32_array(BlendWriter *writer, int size, const int32_t *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_raw(writer, sizeof(int32_t) * size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_uint32_array(BlendWriter *writer, int size, const uint32_t *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_raw(writer, sizeof(uint32_t) * size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_float_array(BlendWriter *writer, int size, const float *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_raw(writer, sizeof(float) * size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLO_write_float3_array(BlendWriter *writer, int size, const float *data_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLO_write_raw(writer, sizeof(float) * 3 * size, data_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Write a null terminated string. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void BLO_write_string(BlendWriter *writer, const char *str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (str != NULL) { | 
					
						
							|  |  |  |     BLO_write_raw(writer, strlen(str) + 1, str); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Sometimes different data is written depending on whether the file is saved to disk or used for | 
					
						
							|  |  |  |  * undo. This function returns true when the current file-writing is done for undo. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool BLO_write_is_undo(BlendWriter *writer) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return writer->wd->use_memfile; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:17:11 +02:00
										 |  |  | /** \} */ |