The Asset Browser will be a sub-editor of the File Browser. This prepares the File Browser code for that. **File-Lists** * Support loading assets with metadata read from external files into the file-list. * New main based file-list type, for the "Current File" asset library. * Refresh file-list when switching between browse modes or asset libraries. * Support empty file-lists (asset library with no assets). * Store file previews as icons, so scripts can reference them via icon-id. See previous commit. **Space Data** * Introduce "browse mode" to differeniate between file and asset browsing. * Add `FileAssetSelectParams` to `SpaceFile`, with `FileSelectParams` as base. Makes sure data is separated between asset and file browsing when switching between them. The active params can be obtained through `ED_fileselect_get_active_params()`. * `FileAssetSelectParams` stores the currently visible asset library ID. * Introduce file history abstraction so file and asset browsing can keep a separate history (previous and next directories). **General** * Option to only show asset data-blocks while file browsing (not exposed here). * Add "active_file" context member, so scripts can get and display info about the active file. * Add "active_id" context member, so `ED_OT_lib_id_load_custom_preview` can set a custom ID preview. (Only for "Current File" asset library) * Expose some of `FileDirEntry` in RNA as (non-editable). That way scripts can obtain name, preview icon and asset-data. Part of the first Asset Browser milestone. Check the #asset_browser_milestone_1 project milestone on developer.blender.org. Differential Revision: https://developer.blender.org/D9724 Reviewed by: Bastien Montagne
1903 lines
54 KiB
C
1903 lines
54 KiB
C
/*
|
|
* 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
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* 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,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/** \file
|
|
* \ingroup bke
|
|
*/
|
|
|
|
/* Allow using deprecated functionality for .blend file I/O. */
|
|
#define DNA_DEPRECATED_ALLOW
|
|
|
|
#ifdef WIN32
|
|
# include "BLI_winstuff.h"
|
|
#endif
|
|
|
|
#include <math.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_collection_types.h"
|
|
#include "DNA_defaults.h"
|
|
#include "DNA_gpencil_types.h"
|
|
#include "DNA_mask_types.h"
|
|
#include "DNA_scene_types.h"
|
|
#include "DNA_screen_types.h"
|
|
#include "DNA_space_types.h"
|
|
#include "DNA_text_types.h"
|
|
#include "DNA_view3d_types.h"
|
|
#include "DNA_workspace_types.h"
|
|
|
|
#include "BLI_listbase.h"
|
|
#include "BLI_math_vector.h"
|
|
#include "BLI_mempool.h"
|
|
#include "BLI_rect.h"
|
|
#include "BLI_utildefines.h"
|
|
|
|
#include "BLT_translation.h"
|
|
|
|
#include "BKE_gpencil.h"
|
|
#include "BKE_icons.h"
|
|
#include "BKE_idprop.h"
|
|
#include "BKE_idtype.h"
|
|
#include "BKE_lib_id.h"
|
|
#include "BKE_lib_query.h"
|
|
#include "BKE_node.h"
|
|
#include "BKE_screen.h"
|
|
#include "BKE_workspace.h"
|
|
|
|
#include "BLO_read_write.h"
|
|
|
|
static void screen_free_data(ID *id)
|
|
{
|
|
bScreen *screen = (bScreen *)id;
|
|
|
|
/* No animdata here. */
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, &screen->regionbase) {
|
|
BKE_area_region_free(NULL, region);
|
|
}
|
|
|
|
BLI_freelistN(&screen->regionbase);
|
|
|
|
BKE_screen_area_map_free(AREAMAP_FROM_SCREEN(screen));
|
|
|
|
BKE_previewimg_free(&screen->preview);
|
|
|
|
/* Region and timer are freed by the window manager. */
|
|
MEM_SAFE_FREE(screen->tool_tip);
|
|
}
|
|
|
|
static void screen_foreach_id_dopesheet(LibraryForeachIDData *data, bDopeSheet *ads)
|
|
{
|
|
if (ads != NULL) {
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, ads->source, IDWALK_CB_NOP);
|
|
BKE_LIB_FOREACHID_PROCESS(data, ads->filter_grp, IDWALK_CB_NOP);
|
|
}
|
|
}
|
|
|
|
void BKE_screen_foreach_id_screen_area(LibraryForeachIDData *data, ScrArea *area)
|
|
{
|
|
BKE_LIB_FOREACHID_PROCESS(data, area->full, IDWALK_CB_NOP);
|
|
|
|
/* TODO this should be moved to a callback in `SpaceType`, defined in each editor's own code.
|
|
* Will be for a later round of cleanup though... */
|
|
LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) {
|
|
switch (sl->spacetype) {
|
|
case SPACE_VIEW3D: {
|
|
View3D *v3d = (View3D *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, v3d->camera, IDWALK_CB_NOP);
|
|
BKE_LIB_FOREACHID_PROCESS(data, v3d->ob_center, IDWALK_CB_NOP);
|
|
|
|
if (v3d->localvd) {
|
|
BKE_LIB_FOREACHID_PROCESS(data, v3d->localvd->camera, IDWALK_CB_NOP);
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_GRAPH: {
|
|
SpaceGraph *sipo = (SpaceGraph *)sl;
|
|
|
|
screen_foreach_id_dopesheet(data, sipo->ads);
|
|
break;
|
|
}
|
|
case SPACE_PROPERTIES: {
|
|
SpaceProperties *sbuts = (SpaceProperties *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, sbuts->pinid, IDWALK_CB_NOP);
|
|
break;
|
|
}
|
|
case SPACE_FILE:
|
|
break;
|
|
case SPACE_ACTION: {
|
|
SpaceAction *saction = (SpaceAction *)sl;
|
|
|
|
screen_foreach_id_dopesheet(data, &saction->ads);
|
|
BKE_LIB_FOREACHID_PROCESS(data, saction->action, IDWALK_CB_NOP);
|
|
break;
|
|
}
|
|
case SPACE_IMAGE: {
|
|
SpaceImage *sima = (SpaceImage *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, sima->image, IDWALK_CB_USER_ONE);
|
|
BKE_LIB_FOREACHID_PROCESS(data, sima->mask_info.mask, IDWALK_CB_USER_ONE);
|
|
BKE_LIB_FOREACHID_PROCESS(data, sima->gpd, IDWALK_CB_USER);
|
|
break;
|
|
}
|
|
case SPACE_SEQ: {
|
|
SpaceSeq *sseq = (SpaceSeq *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, sseq->gpd, IDWALK_CB_USER);
|
|
break;
|
|
}
|
|
case SPACE_NLA: {
|
|
SpaceNla *snla = (SpaceNla *)sl;
|
|
|
|
screen_foreach_id_dopesheet(data, snla->ads);
|
|
break;
|
|
}
|
|
case SPACE_TEXT: {
|
|
SpaceText *st = (SpaceText *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, st->text, IDWALK_CB_NOP);
|
|
break;
|
|
}
|
|
case SPACE_SCRIPT: {
|
|
SpaceScript *scpt = (SpaceScript *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, scpt->script, IDWALK_CB_NOP);
|
|
break;
|
|
}
|
|
case SPACE_OUTLINER: {
|
|
SpaceOutliner *space_outliner = (SpaceOutliner *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, space_outliner->search_tse.id, IDWALK_CB_NOP);
|
|
|
|
if (space_outliner->treestore != NULL) {
|
|
TreeStoreElem *tselem;
|
|
BLI_mempool_iter iter;
|
|
|
|
BLI_mempool_iternew(space_outliner->treestore, &iter);
|
|
while ((tselem = BLI_mempool_iterstep(&iter))) {
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, tselem->id, IDWALK_CB_NOP);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_NODE: {
|
|
SpaceNode *snode = (SpaceNode *)sl;
|
|
|
|
const bool is_private_nodetree = snode->id != NULL &&
|
|
ntreeFromID(snode->id) == snode->nodetree;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, snode->id, IDWALK_CB_NOP);
|
|
BKE_LIB_FOREACHID_PROCESS_ID(data, snode->from, IDWALK_CB_NOP);
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(
|
|
data, snode->nodetree, is_private_nodetree ? IDWALK_CB_EMBEDDED : IDWALK_CB_USER_ONE);
|
|
|
|
LISTBASE_FOREACH (bNodeTreePath *, path, &snode->treepath) {
|
|
if (path == snode->treepath.first) {
|
|
/* first nodetree in path is same as snode->nodetree */
|
|
BKE_LIB_FOREACHID_PROCESS(data,
|
|
path->nodetree,
|
|
is_private_nodetree ? IDWALK_CB_EMBEDDED :
|
|
IDWALK_CB_USER_ONE);
|
|
}
|
|
else {
|
|
BKE_LIB_FOREACHID_PROCESS(data, path->nodetree, IDWALK_CB_USER_ONE);
|
|
}
|
|
|
|
if (path->nodetree == NULL) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, snode->edittree, IDWALK_CB_NOP);
|
|
break;
|
|
}
|
|
case SPACE_CLIP: {
|
|
SpaceClip *sclip = (SpaceClip *)sl;
|
|
|
|
BKE_LIB_FOREACHID_PROCESS(data, sclip->clip, IDWALK_CB_USER_ONE);
|
|
BKE_LIB_FOREACHID_PROCESS(data, sclip->mask_info.mask, IDWALK_CB_USER_ONE);
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void screen_foreach_id(ID *id, LibraryForeachIDData *data)
|
|
{
|
|
if (BKE_lib_query_foreachid_process_flags_get(data) & IDWALK_INCLUDE_UI) {
|
|
bScreen *screen = (bScreen *)id;
|
|
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
BKE_screen_foreach_id_screen_area(data, area);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void screen_blend_write(BlendWriter *writer, ID *id, const void *id_address)
|
|
{
|
|
bScreen *screen = (bScreen *)id;
|
|
/* Screens are reference counted, only saved if used by a workspace. */
|
|
if (screen->id.us > 0 || BLO_write_is_undo(writer)) {
|
|
/* write LibData */
|
|
/* in 2.50+ files, the file identifier for screens is patched, forward compatibility */
|
|
BLO_write_struct_at_address_with_filecode(writer, ID_SCRN, bScreen, id_address, screen);
|
|
BKE_id_blend_write(writer, &screen->id);
|
|
|
|
BKE_previewimg_blend_write(writer, screen->preview);
|
|
|
|
/* direct data */
|
|
BKE_screen_area_map_blend_write(writer, AREAMAP_FROM_SCREEN(screen));
|
|
}
|
|
}
|
|
|
|
/* Cannot use IDTypeInfo callback yet, because of the return value. */
|
|
bool BKE_screen_blend_read_data(BlendDataReader *reader, bScreen *screen)
|
|
{
|
|
bool success = true;
|
|
|
|
screen->regionbase.first = screen->regionbase.last = NULL;
|
|
screen->context = NULL;
|
|
screen->active_region = NULL;
|
|
|
|
BLO_read_data_address(reader, &screen->preview);
|
|
BKE_previewimg_blend_read(reader, screen->preview);
|
|
|
|
if (!BKE_screen_area_map_blend_read_data(reader, AREAMAP_FROM_SCREEN(screen))) {
|
|
printf("Error reading Screen %s... removing it.\n", screen->id.name + 2);
|
|
success = false;
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
/* note: file read without screens option G_FILE_NO_UI;
|
|
* check lib pointers in call below */
|
|
static void screen_blend_read_lib(BlendLibReader *reader, ID *id)
|
|
{
|
|
bScreen *screen = (bScreen *)id;
|
|
/* deprecated, but needed for versioning (will be NULL'ed then) */
|
|
BLO_read_id_address(reader, screen->id.lib, &screen->scene);
|
|
|
|
screen->animtimer = NULL; /* saved in rare cases */
|
|
screen->tool_tip = NULL;
|
|
screen->scrubbing = false;
|
|
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
BKE_screen_area_blend_read_lib(reader, &screen->id, area);
|
|
}
|
|
}
|
|
|
|
IDTypeInfo IDType_ID_SCR = {
|
|
.id_code = ID_SCR,
|
|
.id_filter = 0,
|
|
.main_listbase_index = INDEX_ID_SCR,
|
|
.struct_size = sizeof(bScreen),
|
|
.name = "Screen",
|
|
.name_plural = "screens",
|
|
.translation_context = BLT_I18NCONTEXT_ID_SCREEN,
|
|
.flags = IDTYPE_FLAGS_NO_COPY | IDTYPE_FLAGS_NO_MAKELOCAL | IDTYPE_FLAGS_NO_ANIMDATA,
|
|
|
|
.init_data = NULL,
|
|
.copy_data = NULL,
|
|
.free_data = screen_free_data,
|
|
.make_local = NULL,
|
|
.foreach_id = screen_foreach_id,
|
|
.foreach_cache = NULL,
|
|
|
|
.blend_write = screen_blend_write,
|
|
/* Cannot be used yet, because #direct_link_screen has a return value. */
|
|
.blend_read_data = NULL,
|
|
.blend_read_lib = screen_blend_read_lib,
|
|
.blend_read_expand = NULL,
|
|
|
|
.blend_read_undo_preserve = NULL,
|
|
};
|
|
|
|
/* ************ Spacetype/regiontype handling ************** */
|
|
|
|
/* keep global; this has to be accessible outside of windowmanager */
|
|
static ListBase spacetypes = {NULL, NULL};
|
|
|
|
/* not SpaceType itself */
|
|
static void spacetype_free(SpaceType *st)
|
|
{
|
|
LISTBASE_FOREACH (ARegionType *, art, &st->regiontypes) {
|
|
BLI_freelistN(&art->drawcalls);
|
|
|
|
LISTBASE_FOREACH (PanelType *, pt, &art->paneltypes) {
|
|
if (pt->rna_ext.free) {
|
|
pt->rna_ext.free(pt->rna_ext.data);
|
|
}
|
|
|
|
BLI_freelistN(&pt->children);
|
|
}
|
|
|
|
LISTBASE_FOREACH (HeaderType *, ht, &art->headertypes) {
|
|
if (ht->rna_ext.free) {
|
|
ht->rna_ext.free(ht->rna_ext.data);
|
|
}
|
|
}
|
|
|
|
BLI_freelistN(&art->paneltypes);
|
|
BLI_freelistN(&art->headertypes);
|
|
}
|
|
|
|
BLI_freelistN(&st->regiontypes);
|
|
}
|
|
|
|
void BKE_spacetypes_free(void)
|
|
{
|
|
LISTBASE_FOREACH (SpaceType *, st, &spacetypes) {
|
|
spacetype_free(st);
|
|
}
|
|
|
|
BLI_freelistN(&spacetypes);
|
|
}
|
|
|
|
SpaceType *BKE_spacetype_from_id(int spaceid)
|
|
{
|
|
LISTBASE_FOREACH (SpaceType *, st, &spacetypes) {
|
|
if (st->spaceid == spaceid) {
|
|
return st;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
ARegionType *BKE_regiontype_from_id_or_first(const SpaceType *st, int regionid)
|
|
{
|
|
LISTBASE_FOREACH (ARegionType *, art, &st->regiontypes) {
|
|
if (art->regionid == regionid) {
|
|
return art;
|
|
}
|
|
}
|
|
|
|
printf(
|
|
"Error, region type %d missing in - name:\"%s\", id:%d\n", regionid, st->name, st->spaceid);
|
|
return st->regiontypes.first;
|
|
}
|
|
|
|
ARegionType *BKE_regiontype_from_id(const SpaceType *st, int regionid)
|
|
{
|
|
LISTBASE_FOREACH (ARegionType *, art, &st->regiontypes) {
|
|
if (art->regionid == regionid) {
|
|
return art;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
const ListBase *BKE_spacetypes_list(void)
|
|
{
|
|
return &spacetypes;
|
|
}
|
|
|
|
void BKE_spacetype_register(SpaceType *st)
|
|
{
|
|
/* sanity check */
|
|
SpaceType *stype = BKE_spacetype_from_id(st->spaceid);
|
|
if (stype) {
|
|
printf("error: redefinition of spacetype %s\n", stype->name);
|
|
spacetype_free(stype);
|
|
MEM_freeN(stype);
|
|
}
|
|
|
|
BLI_addtail(&spacetypes, st);
|
|
}
|
|
|
|
bool BKE_spacetype_exists(int spaceid)
|
|
{
|
|
return BKE_spacetype_from_id(spaceid) != NULL;
|
|
}
|
|
|
|
/* ***************** Space handling ********************** */
|
|
|
|
void BKE_spacedata_freelist(ListBase *lb)
|
|
{
|
|
LISTBASE_FOREACH (SpaceLink *, sl, lb) {
|
|
SpaceType *st = BKE_spacetype_from_id(sl->spacetype);
|
|
|
|
/* free regions for pushed spaces */
|
|
LISTBASE_FOREACH (ARegion *, region, &sl->regionbase) {
|
|
BKE_area_region_free(st, region);
|
|
}
|
|
|
|
BLI_freelistN(&sl->regionbase);
|
|
|
|
if (st && st->free) {
|
|
st->free(sl);
|
|
}
|
|
}
|
|
|
|
BLI_freelistN(lb);
|
|
}
|
|
|
|
static void panel_list_copy(ListBase *newlb, const ListBase *lb)
|
|
{
|
|
BLI_listbase_clear(newlb);
|
|
BLI_duplicatelist(newlb, lb);
|
|
|
|
/* copy panel pointers */
|
|
Panel *new_panel = newlb->first;
|
|
Panel *panel = lb->first;
|
|
for (; new_panel; new_panel = new_panel->next, panel = panel->next) {
|
|
new_panel->activedata = NULL;
|
|
new_panel->runtime.custom_data_ptr = NULL;
|
|
panel_list_copy(&new_panel->children, &panel->children);
|
|
}
|
|
}
|
|
|
|
ARegion *BKE_area_region_copy(const SpaceType *st, const ARegion *region)
|
|
{
|
|
ARegion *newar = MEM_dupallocN(region);
|
|
|
|
newar->prev = newar->next = NULL;
|
|
BLI_listbase_clear(&newar->handlers);
|
|
BLI_listbase_clear(&newar->uiblocks);
|
|
BLI_listbase_clear(&newar->panels_category);
|
|
BLI_listbase_clear(&newar->panels_category_active);
|
|
BLI_listbase_clear(&newar->ui_lists);
|
|
newar->visible = 0;
|
|
newar->gizmo_map = NULL;
|
|
newar->regiontimer = NULL;
|
|
newar->headerstr = NULL;
|
|
newar->draw_buffer = NULL;
|
|
|
|
/* use optional regiondata callback */
|
|
if (region->regiondata) {
|
|
ARegionType *art = BKE_regiontype_from_id(st, region->regiontype);
|
|
|
|
if (art && art->duplicate) {
|
|
newar->regiondata = art->duplicate(region->regiondata);
|
|
}
|
|
else if (region->flag & RGN_FLAG_TEMP_REGIONDATA) {
|
|
newar->regiondata = NULL;
|
|
}
|
|
else {
|
|
newar->regiondata = MEM_dupallocN(region->regiondata);
|
|
}
|
|
}
|
|
|
|
panel_list_copy(&newar->panels, ®ion->panels);
|
|
|
|
BLI_listbase_clear(&newar->ui_previews);
|
|
BLI_duplicatelist(&newar->ui_previews, ®ion->ui_previews);
|
|
|
|
return newar;
|
|
}
|
|
|
|
/* from lb2 to lb1, lb1 is supposed to be freed */
|
|
static void region_copylist(SpaceType *st, ListBase *lb1, ListBase *lb2)
|
|
{
|
|
/* to be sure */
|
|
BLI_listbase_clear(lb1);
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, lb2) {
|
|
ARegion *region_new = BKE_area_region_copy(st, region);
|
|
BLI_addtail(lb1, region_new);
|
|
}
|
|
}
|
|
|
|
/* lb1 should be empty */
|
|
void BKE_spacedata_copylist(ListBase *lb1, ListBase *lb2)
|
|
{
|
|
BLI_listbase_clear(lb1); /* to be sure */
|
|
|
|
LISTBASE_FOREACH (SpaceLink *, sl, lb2) {
|
|
SpaceType *st = BKE_spacetype_from_id(sl->spacetype);
|
|
|
|
if (st && st->duplicate) {
|
|
SpaceLink *slnew = st->duplicate(sl);
|
|
|
|
BLI_addtail(lb1, slnew);
|
|
|
|
region_copylist(st, &slnew->regionbase, &sl->regionbase);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* facility to set locks for drawing to survive (render) threads accessing drawing data */
|
|
/* lock can become bitflag too */
|
|
/* should be replaced in future by better local data handling for threads */
|
|
void BKE_spacedata_draw_locks(int set)
|
|
{
|
|
LISTBASE_FOREACH (SpaceType *, st, &spacetypes) {
|
|
LISTBASE_FOREACH (ARegionType *, art, &st->regiontypes) {
|
|
if (set) {
|
|
art->do_lock = art->lock;
|
|
}
|
|
else {
|
|
art->do_lock = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Version of #BKE_area_find_region_type that also works if \a slink
|
|
* is not the active space of \a area.
|
|
*/
|
|
ARegion *BKE_spacedata_find_region_type(const SpaceLink *slink,
|
|
const ScrArea *area,
|
|
int region_type)
|
|
{
|
|
const bool is_slink_active = slink == area->spacedata.first;
|
|
const ListBase *regionbase = (is_slink_active) ? &area->regionbase : &slink->regionbase;
|
|
ARegion *region = NULL;
|
|
|
|
BLI_assert(BLI_findindex(&area->spacedata, slink) != -1);
|
|
|
|
LISTBASE_FOREACH (ARegion *, region_iter, regionbase) {
|
|
if (region_iter->regiontype == region_type) {
|
|
region = region_iter;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Should really unit test this instead. */
|
|
BLI_assert(!is_slink_active || region == BKE_area_find_region_type(area, region_type));
|
|
|
|
return region;
|
|
}
|
|
|
|
static void (*spacedata_id_remap_cb)(struct ScrArea *area,
|
|
struct SpaceLink *sl,
|
|
ID *old_id,
|
|
ID *new_id) = NULL;
|
|
|
|
void BKE_spacedata_callback_id_remap_set(void (*func)(ScrArea *area, SpaceLink *sl, ID *, ID *))
|
|
{
|
|
spacedata_id_remap_cb = func;
|
|
}
|
|
|
|
/* UNUSED!!! */
|
|
void BKE_spacedata_id_unref(struct ScrArea *area, struct SpaceLink *sl, struct ID *id)
|
|
{
|
|
if (spacedata_id_remap_cb) {
|
|
spacedata_id_remap_cb(area, sl, id, NULL);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Avoid bad-level calls to #WM_gizmomap_tag_refresh.
|
|
*/
|
|
static void (*region_refresh_tag_gizmomap_callback)(struct wmGizmoMap *) = NULL;
|
|
|
|
void BKE_region_callback_refresh_tag_gizmomap_set(void (*callback)(struct wmGizmoMap *))
|
|
{
|
|
region_refresh_tag_gizmomap_callback = callback;
|
|
}
|
|
|
|
void BKE_screen_gizmo_tag_refresh(struct bScreen *screen)
|
|
{
|
|
if (region_refresh_tag_gizmomap_callback == NULL) {
|
|
return;
|
|
}
|
|
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
if (region->gizmo_map != NULL) {
|
|
region_refresh_tag_gizmomap_callback(region->gizmo_map);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Avoid bad-level calls to #WM_gizmomap_delete.
|
|
*/
|
|
static void (*region_free_gizmomap_callback)(struct wmGizmoMap *) = NULL;
|
|
|
|
void BKE_region_callback_free_gizmomap_set(void (*callback)(struct wmGizmoMap *))
|
|
{
|
|
region_free_gizmomap_callback = callback;
|
|
}
|
|
|
|
static void area_region_panels_free_recursive(Panel *panel)
|
|
{
|
|
MEM_SAFE_FREE(panel->activedata);
|
|
|
|
LISTBASE_FOREACH_MUTABLE (Panel *, child_panel, &panel->children) {
|
|
area_region_panels_free_recursive(child_panel);
|
|
}
|
|
|
|
MEM_freeN(panel);
|
|
}
|
|
|
|
void BKE_area_region_panels_free(ListBase *panels)
|
|
{
|
|
LISTBASE_FOREACH_MUTABLE (Panel *, panel, panels) {
|
|
/* Free custom data just for parent panels to avoid a double free. */
|
|
MEM_SAFE_FREE(panel->runtime.custom_data_ptr);
|
|
area_region_panels_free_recursive(panel);
|
|
}
|
|
BLI_listbase_clear(panels);
|
|
}
|
|
|
|
/* not region itself */
|
|
void BKE_area_region_free(SpaceType *st, ARegion *region)
|
|
{
|
|
if (st) {
|
|
ARegionType *art = BKE_regiontype_from_id(st, region->regiontype);
|
|
|
|
if (art && art->free) {
|
|
art->free(region);
|
|
}
|
|
|
|
if (region->regiondata) {
|
|
printf("regiondata free error\n");
|
|
}
|
|
}
|
|
else if (region->type && region->type->free) {
|
|
region->type->free(region);
|
|
}
|
|
|
|
BKE_area_region_panels_free(®ion->panels);
|
|
|
|
LISTBASE_FOREACH (uiList *, uilst, ®ion->ui_lists) {
|
|
if (uilst->dyn_data) {
|
|
uiListDyn *dyn_data = uilst->dyn_data;
|
|
if (dyn_data->items_filter_flags) {
|
|
MEM_freeN(dyn_data->items_filter_flags);
|
|
}
|
|
if (dyn_data->items_filter_neworder) {
|
|
MEM_freeN(dyn_data->items_filter_neworder);
|
|
}
|
|
MEM_freeN(dyn_data);
|
|
}
|
|
if (uilst->properties) {
|
|
IDP_FreeProperty(uilst->properties);
|
|
}
|
|
}
|
|
|
|
if (region->gizmo_map != NULL) {
|
|
region_free_gizmomap_callback(region->gizmo_map);
|
|
}
|
|
|
|
BLI_freelistN(®ion->ui_lists);
|
|
BLI_freelistN(®ion->ui_previews);
|
|
BLI_freelistN(®ion->panels_category);
|
|
BLI_freelistN(®ion->panels_category_active);
|
|
}
|
|
|
|
/* not area itself */
|
|
void BKE_screen_area_free(ScrArea *area)
|
|
{
|
|
SpaceType *st = BKE_spacetype_from_id(area->spacetype);
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
BKE_area_region_free(st, region);
|
|
}
|
|
|
|
MEM_SAFE_FREE(area->global);
|
|
BLI_freelistN(&area->regionbase);
|
|
|
|
BKE_spacedata_freelist(&area->spacedata);
|
|
|
|
BLI_freelistN(&area->actionzones);
|
|
}
|
|
|
|
void BKE_screen_area_map_free(ScrAreaMap *area_map)
|
|
{
|
|
LISTBASE_FOREACH_MUTABLE (ScrArea *, area, &area_map->areabase) {
|
|
BKE_screen_area_free(area);
|
|
}
|
|
|
|
BLI_freelistN(&area_map->vertbase);
|
|
BLI_freelistN(&area_map->edgebase);
|
|
BLI_freelistN(&area_map->areabase);
|
|
}
|
|
|
|
/** Free (or release) any data used by this screen (does not free the screen itself). */
|
|
void BKE_screen_free(bScreen *screen)
|
|
{
|
|
screen_free_data(&screen->id);
|
|
}
|
|
|
|
/* ***************** Screen edges & verts ***************** */
|
|
|
|
ScrEdge *BKE_screen_find_edge(const bScreen *screen, ScrVert *v1, ScrVert *v2)
|
|
{
|
|
BKE_screen_sort_scrvert(&v1, &v2);
|
|
LISTBASE_FOREACH (ScrEdge *, se, &screen->edgebase) {
|
|
if (se->v1 == v1 && se->v2 == v2) {
|
|
return se;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void BKE_screen_sort_scrvert(ScrVert **v1, ScrVert **v2)
|
|
{
|
|
if (*v1 > *v2) {
|
|
ScrVert *tmp = *v1;
|
|
*v1 = *v2;
|
|
*v2 = tmp;
|
|
}
|
|
}
|
|
|
|
void BKE_screen_remove_double_scrverts(bScreen *screen)
|
|
{
|
|
LISTBASE_FOREACH (ScrVert *, verg, &screen->vertbase) {
|
|
if (verg->newv == NULL) { /* !!! */
|
|
ScrVert *v1 = verg->next;
|
|
while (v1) {
|
|
if (v1->newv == NULL) { /* !?! */
|
|
if (v1->vec.x == verg->vec.x && v1->vec.y == verg->vec.y) {
|
|
/* printf("doublevert\n"); */
|
|
v1->newv = verg;
|
|
}
|
|
}
|
|
v1 = v1->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* replace pointers in edges and faces */
|
|
LISTBASE_FOREACH (ScrEdge *, se, &screen->edgebase) {
|
|
if (se->v1->newv) {
|
|
se->v1 = se->v1->newv;
|
|
}
|
|
if (se->v2->newv) {
|
|
se->v2 = se->v2->newv;
|
|
}
|
|
/* edges changed: so.... */
|
|
BKE_screen_sort_scrvert(&(se->v1), &(se->v2));
|
|
}
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
if (area->v1->newv) {
|
|
area->v1 = area->v1->newv;
|
|
}
|
|
if (area->v2->newv) {
|
|
area->v2 = area->v2->newv;
|
|
}
|
|
if (area->v3->newv) {
|
|
area->v3 = area->v3->newv;
|
|
}
|
|
if (area->v4->newv) {
|
|
area->v4 = area->v4->newv;
|
|
}
|
|
}
|
|
|
|
/* remove */
|
|
LISTBASE_FOREACH_MUTABLE (ScrVert *, verg, &screen->vertbase) {
|
|
if (verg->newv) {
|
|
BLI_remlink(&screen->vertbase, verg);
|
|
MEM_freeN(verg);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_remove_double_scredges(bScreen *screen)
|
|
{
|
|
/* compare */
|
|
LISTBASE_FOREACH (ScrEdge *, verg, &screen->edgebase) {
|
|
ScrEdge *se = verg->next;
|
|
while (se) {
|
|
ScrEdge *sn = se->next;
|
|
if (verg->v1 == se->v1 && verg->v2 == se->v2) {
|
|
BLI_remlink(&screen->edgebase, se);
|
|
MEM_freeN(se);
|
|
}
|
|
se = sn;
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_remove_unused_scredges(bScreen *screen)
|
|
{
|
|
/* sets flags when edge is used in area */
|
|
int a = 0;
|
|
LISTBASE_FOREACH_INDEX (ScrArea *, area, &screen->areabase, a) {
|
|
ScrEdge *se = BKE_screen_find_edge(screen, area->v1, area->v2);
|
|
if (se == NULL) {
|
|
printf("error: area %d edge 1 doesn't exist\n", a);
|
|
}
|
|
else {
|
|
se->flag = 1;
|
|
}
|
|
se = BKE_screen_find_edge(screen, area->v2, area->v3);
|
|
if (se == NULL) {
|
|
printf("error: area %d edge 2 doesn't exist\n", a);
|
|
}
|
|
else {
|
|
se->flag = 1;
|
|
}
|
|
se = BKE_screen_find_edge(screen, area->v3, area->v4);
|
|
if (se == NULL) {
|
|
printf("error: area %d edge 3 doesn't exist\n", a);
|
|
}
|
|
else {
|
|
se->flag = 1;
|
|
}
|
|
se = BKE_screen_find_edge(screen, area->v4, area->v1);
|
|
if (se == NULL) {
|
|
printf("error: area %d edge 4 doesn't exist\n", a);
|
|
}
|
|
else {
|
|
se->flag = 1;
|
|
}
|
|
}
|
|
LISTBASE_FOREACH_MUTABLE (ScrEdge *, se, &screen->edgebase) {
|
|
if (se->flag == 0) {
|
|
BLI_remlink(&screen->edgebase, se);
|
|
MEM_freeN(se);
|
|
}
|
|
else {
|
|
se->flag = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_remove_unused_scrverts(bScreen *screen)
|
|
{
|
|
/* we assume edges are ok */
|
|
LISTBASE_FOREACH (ScrEdge *, se, &screen->edgebase) {
|
|
se->v1->flag = 1;
|
|
se->v2->flag = 1;
|
|
}
|
|
|
|
LISTBASE_FOREACH_MUTABLE (ScrVert *, sv, &screen->vertbase) {
|
|
if (sv->flag == 0) {
|
|
BLI_remlink(&screen->vertbase, sv);
|
|
MEM_freeN(sv);
|
|
}
|
|
else {
|
|
sv->flag = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ***************** Utilities ********************** */
|
|
|
|
/**
|
|
* Find a region of type \a region_type in the currently active space of \a area.
|
|
*
|
|
* \note This does _not_ work if the region to look up is not in the active
|
|
* space. Use #BKE_spacedata_find_region_type if that may be the case.
|
|
*/
|
|
ARegion *BKE_area_find_region_type(const ScrArea *area, int region_type)
|
|
{
|
|
if (area) {
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
if (region->regiontype == region_type) {
|
|
return region;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
ARegion *BKE_area_find_region_active_win(ScrArea *area)
|
|
{
|
|
if (area == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
ARegion *region = BLI_findlink(&area->regionbase, area->region_active_win);
|
|
if (region && (region->regiontype == RGN_TYPE_WINDOW)) {
|
|
return region;
|
|
}
|
|
|
|
/* fallback to any */
|
|
return BKE_area_find_region_type(area, RGN_TYPE_WINDOW);
|
|
}
|
|
|
|
ARegion *BKE_area_find_region_xy(ScrArea *area, const int regiontype, int x, int y)
|
|
{
|
|
if (area == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
if (ELEM(regiontype, RGN_TYPE_ANY, region->regiontype)) {
|
|
if (BLI_rcti_isect_pt(®ion->winrct, x, y)) {
|
|
return region;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* \note This is only for screen level regions (typically menus/popups).
|
|
*/
|
|
ARegion *BKE_screen_find_region_xy(bScreen *screen, const int regiontype, int x, int y)
|
|
{
|
|
LISTBASE_FOREACH (ARegion *, region, &screen->regionbase) {
|
|
if (ELEM(regiontype, RGN_TYPE_ANY, region->regiontype)) {
|
|
if (BLI_rcti_isect_pt(®ion->winrct, x, y)) {
|
|
return region;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* \note Ideally we can get the area from the context,
|
|
* there are a few places however where this isn't practical.
|
|
*/
|
|
ScrArea *BKE_screen_find_area_from_space(struct bScreen *screen, SpaceLink *sl)
|
|
{
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
if (BLI_findindex(&area->spacedata, sl) != -1) {
|
|
return area;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* \note Using this function is generally a last resort, you really want to be
|
|
* using the context when you can - campbell
|
|
*/
|
|
ScrArea *BKE_screen_find_big_area(bScreen *screen, const int spacetype, const short min)
|
|
{
|
|
ScrArea *big = NULL;
|
|
int maxsize = 0;
|
|
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
if (ELEM(spacetype, SPACE_TYPE_ANY, area->spacetype)) {
|
|
if (min <= area->winx && min <= area->winy) {
|
|
int size = area->winx * area->winy;
|
|
if (size > maxsize) {
|
|
maxsize = size;
|
|
big = area;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return big;
|
|
}
|
|
|
|
ScrArea *BKE_screen_area_map_find_area_xy(const ScrAreaMap *areamap,
|
|
const int spacetype,
|
|
int x,
|
|
int y)
|
|
{
|
|
LISTBASE_FOREACH (ScrArea *, area, &areamap->areabase) {
|
|
if (BLI_rcti_isect_pt(&area->totrct, x, y)) {
|
|
if (ELEM(spacetype, SPACE_TYPE_ANY, area->spacetype)) {
|
|
return area;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
ScrArea *BKE_screen_find_area_xy(bScreen *screen, const int spacetype, int x, int y)
|
|
{
|
|
return BKE_screen_area_map_find_area_xy(AREAMAP_FROM_SCREEN(screen), spacetype, x, y);
|
|
}
|
|
|
|
void BKE_screen_view3d_sync(View3D *v3d, struct Scene *scene)
|
|
{
|
|
if (v3d->scenelock && v3d->localvd == NULL) {
|
|
v3d->camera = scene->camera;
|
|
|
|
if (v3d->camera == NULL) {
|
|
LISTBASE_FOREACH (ARegion *, region, &v3d->regionbase) {
|
|
if (region->regiontype == RGN_TYPE_WINDOW) {
|
|
RegionView3D *rv3d = region->regiondata;
|
|
if (rv3d->persp == RV3D_CAMOB) {
|
|
rv3d->persp = RV3D_PERSP;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_view3d_scene_sync(bScreen *screen, Scene *scene)
|
|
{
|
|
/* are there cameras in the views that are not in the scene? */
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) {
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
View3D *v3d = (View3D *)sl;
|
|
BKE_screen_view3d_sync(v3d, scene);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_view3d_shading_init(View3DShading *shading)
|
|
{
|
|
const View3DShading *shading_default = DNA_struct_default_get(View3DShading);
|
|
memcpy(shading, shading_default, sizeof(*shading));
|
|
}
|
|
|
|
ARegion *BKE_screen_find_main_region_at_xy(bScreen *screen,
|
|
const int space_type,
|
|
const int x,
|
|
const int y)
|
|
{
|
|
ScrArea *area = BKE_screen_find_area_xy(screen, space_type, x, y);
|
|
if (!area) {
|
|
return NULL;
|
|
}
|
|
return BKE_area_find_region_xy(area, RGN_TYPE_WINDOW, x, y);
|
|
}
|
|
|
|
/* magic zoom calculation, no idea what
|
|
* it signifies, if you find out, tell me! -zr
|
|
*/
|
|
|
|
/* simple, its magic dude!
|
|
* well, to be honest, this gives a natural feeling zooming
|
|
* with multiple keypad presses (ton)
|
|
*/
|
|
float BKE_screen_view3d_zoom_to_fac(float camzoom)
|
|
{
|
|
return powf(((float)M_SQRT2 + camzoom / 50.0f), 2.0f) / 4.0f;
|
|
}
|
|
|
|
float BKE_screen_view3d_zoom_from_fac(float zoomfac)
|
|
{
|
|
return ((sqrtf(4.0f * zoomfac) - (float)M_SQRT2) * 50.0f);
|
|
}
|
|
|
|
bool BKE_screen_is_fullscreen_area(const bScreen *screen)
|
|
{
|
|
return ELEM(screen->state, SCREENMAXIMIZED, SCREENFULL);
|
|
}
|
|
|
|
bool BKE_screen_is_used(const bScreen *screen)
|
|
{
|
|
return (screen->winid != 0);
|
|
}
|
|
|
|
void BKE_screen_header_alignment_reset(bScreen *screen)
|
|
{
|
|
int alignment = (U.uiflag & USER_HEADER_BOTTOM) ? RGN_ALIGN_BOTTOM : RGN_ALIGN_TOP;
|
|
LISTBASE_FOREACH (ScrArea *, area, &screen->areabase) {
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
if (ELEM(region->regiontype, RGN_TYPE_HEADER, RGN_TYPE_TOOL_HEADER)) {
|
|
if (ELEM(area->spacetype, SPACE_FILE, SPACE_USERPREF, SPACE_OUTLINER, SPACE_PROPERTIES)) {
|
|
region->alignment = RGN_ALIGN_TOP;
|
|
continue;
|
|
}
|
|
region->alignment = alignment;
|
|
}
|
|
if (region->regiontype == RGN_TYPE_FOOTER) {
|
|
if (ELEM(area->spacetype, SPACE_FILE, SPACE_USERPREF, SPACE_OUTLINER, SPACE_PROPERTIES)) {
|
|
region->alignment = RGN_ALIGN_BOTTOM;
|
|
continue;
|
|
}
|
|
region->alignment = (U.uiflag & USER_HEADER_BOTTOM) ? RGN_ALIGN_TOP : RGN_ALIGN_BOTTOM;
|
|
}
|
|
}
|
|
}
|
|
screen->do_refresh = true;
|
|
}
|
|
|
|
void BKE_screen_view3d_shading_blend_write(BlendWriter *writer, View3DShading *shading)
|
|
{
|
|
if (shading->prop) {
|
|
IDP_BlendWrite(writer, shading->prop);
|
|
}
|
|
}
|
|
|
|
void BKE_screen_view3d_shading_blend_read_data(BlendDataReader *reader, View3DShading *shading)
|
|
{
|
|
if (shading->prop) {
|
|
BLO_read_data_address(reader, &shading->prop);
|
|
IDP_BlendDataRead(reader, &shading->prop);
|
|
}
|
|
}
|
|
|
|
static void write_region(BlendWriter *writer, ARegion *region, int spacetype)
|
|
{
|
|
BLO_write_struct(writer, ARegion, region);
|
|
|
|
if (region->regiondata) {
|
|
if (region->flag & RGN_FLAG_TEMP_REGIONDATA) {
|
|
return;
|
|
}
|
|
|
|
switch (spacetype) {
|
|
case SPACE_VIEW3D:
|
|
if (region->regiontype == RGN_TYPE_WINDOW) {
|
|
RegionView3D *rv3d = region->regiondata;
|
|
BLO_write_struct(writer, RegionView3D, rv3d);
|
|
|
|
if (rv3d->localvd) {
|
|
BLO_write_struct(writer, RegionView3D, rv3d->localvd);
|
|
}
|
|
if (rv3d->clipbb) {
|
|
BLO_write_struct(writer, BoundBox, rv3d->clipbb);
|
|
}
|
|
}
|
|
else {
|
|
printf("regiondata write missing!\n");
|
|
}
|
|
break;
|
|
default:
|
|
printf("regiondata write missing!\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
static void write_uilist(BlendWriter *writer, uiList *ui_list)
|
|
{
|
|
BLO_write_struct(writer, uiList, ui_list);
|
|
|
|
if (ui_list->properties) {
|
|
IDP_BlendWrite(writer, ui_list->properties);
|
|
}
|
|
}
|
|
|
|
static void write_space_outliner(BlendWriter *writer, SpaceOutliner *space_outliner)
|
|
{
|
|
BLI_mempool *ts = space_outliner->treestore;
|
|
|
|
if (ts) {
|
|
SpaceOutliner space_outliner_flat = *space_outliner;
|
|
|
|
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;
|
|
|
|
BLO_write_struct(writer, SpaceOutliner, space_outliner);
|
|
|
|
BLO_write_struct_at_address(writer, TreeStore, ts, &ts_flat);
|
|
BLO_write_struct_array_at_address(writer, TreeStoreElem, elems, data_addr, data);
|
|
|
|
MEM_freeN(data);
|
|
}
|
|
else {
|
|
space_outliner_flat.treestore = NULL;
|
|
BLO_write_struct_at_address(writer, SpaceOutliner, space_outliner, &space_outliner_flat);
|
|
}
|
|
}
|
|
else {
|
|
BLO_write_struct(writer, SpaceOutliner, space_outliner);
|
|
}
|
|
}
|
|
|
|
static void write_panel_list(BlendWriter *writer, ListBase *lb)
|
|
{
|
|
LISTBASE_FOREACH (Panel *, panel, lb) {
|
|
BLO_write_struct(writer, Panel, panel);
|
|
write_panel_list(writer, &panel->children);
|
|
}
|
|
}
|
|
|
|
static void write_area_regions(BlendWriter *writer, ScrArea *area)
|
|
{
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
write_region(writer, region, area->spacetype);
|
|
write_panel_list(writer, ®ion->panels);
|
|
|
|
LISTBASE_FOREACH (PanelCategoryStack *, pc_act, ®ion->panels_category_active) {
|
|
BLO_write_struct(writer, PanelCategoryStack, pc_act);
|
|
}
|
|
|
|
LISTBASE_FOREACH (uiList *, ui_list, ®ion->ui_lists) {
|
|
write_uilist(writer, ui_list);
|
|
}
|
|
|
|
LISTBASE_FOREACH (uiPreview *, ui_preview, ®ion->ui_previews) {
|
|
BLO_write_struct(writer, uiPreview, ui_preview);
|
|
}
|
|
}
|
|
|
|
LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) {
|
|
LISTBASE_FOREACH (ARegion *, region, &sl->regionbase) {
|
|
write_region(writer, region, sl->spacetype);
|
|
}
|
|
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
View3D *v3d = (View3D *)sl;
|
|
BLO_write_struct(writer, View3D, v3d);
|
|
|
|
if (v3d->localvd) {
|
|
BLO_write_struct(writer, View3D, v3d->localvd);
|
|
}
|
|
|
|
BKE_screen_view3d_shading_blend_write(writer, &v3d->shading);
|
|
}
|
|
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);
|
|
|
|
BLO_write_struct(writer, SpaceGraph, sl);
|
|
if (sipo->ads) {
|
|
BLO_write_struct(writer, bDopeSheet, sipo->ads);
|
|
}
|
|
|
|
/* reenable ghost curves */
|
|
sipo->runtime.ghost_curves = tmpGhosts;
|
|
}
|
|
else if (sl->spacetype == SPACE_PROPERTIES) {
|
|
BLO_write_struct(writer, SpaceProperties, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_FILE) {
|
|
SpaceFile *sfile = (SpaceFile *)sl;
|
|
|
|
BLO_write_struct(writer, SpaceFile, sl);
|
|
if (sfile->params) {
|
|
BLO_write_struct(writer, FileSelectParams, sfile->params);
|
|
}
|
|
if (sfile->asset_params) {
|
|
BLO_write_struct(writer, FileAssetSelectParams, sfile->asset_params);
|
|
}
|
|
}
|
|
else if (sl->spacetype == SPACE_SEQ) {
|
|
BLO_write_struct(writer, SpaceSeq, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_OUTLINER) {
|
|
SpaceOutliner *space_outliner = (SpaceOutliner *)sl;
|
|
write_space_outliner(writer, space_outliner);
|
|
}
|
|
else if (sl->spacetype == SPACE_IMAGE) {
|
|
BLO_write_struct(writer, SpaceImage, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_TEXT) {
|
|
BLO_write_struct(writer, SpaceText, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_SCRIPT) {
|
|
SpaceScript *scr = (SpaceScript *)sl;
|
|
scr->but_refs = NULL;
|
|
BLO_write_struct(writer, SpaceScript, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_ACTION) {
|
|
BLO_write_struct(writer, SpaceAction, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_NLA) {
|
|
SpaceNla *snla = (SpaceNla *)sl;
|
|
|
|
BLO_write_struct(writer, SpaceNla, snla);
|
|
if (snla->ads) {
|
|
BLO_write_struct(writer, bDopeSheet, snla->ads);
|
|
}
|
|
}
|
|
else if (sl->spacetype == SPACE_NODE) {
|
|
SpaceNode *snode = (SpaceNode *)sl;
|
|
BLO_write_struct(writer, SpaceNode, snode);
|
|
|
|
LISTBASE_FOREACH (bNodeTreePath *, path, &snode->treepath) {
|
|
BLO_write_struct(writer, bNodeTreePath, path);
|
|
}
|
|
}
|
|
else if (sl->spacetype == SPACE_CONSOLE) {
|
|
SpaceConsole *con = (SpaceConsole *)sl;
|
|
|
|
LISTBASE_FOREACH (ConsoleLine *, cl, &con->history) {
|
|
/* 'len_alloc' is invalid on write, set from 'len' on read */
|
|
BLO_write_struct(writer, ConsoleLine, cl);
|
|
BLO_write_raw(writer, (size_t)cl->len + 1, cl->line);
|
|
}
|
|
BLO_write_struct(writer, SpaceConsole, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_TOPBAR) {
|
|
BLO_write_struct(writer, SpaceTopBar, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_STATUSBAR) {
|
|
BLO_write_struct(writer, SpaceStatusBar, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_USERPREF) {
|
|
BLO_write_struct(writer, SpaceUserPref, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_CLIP) {
|
|
BLO_write_struct(writer, SpaceClip, sl);
|
|
}
|
|
else if (sl->spacetype == SPACE_INFO) {
|
|
BLO_write_struct(writer, SpaceInfo, sl);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BKE_screen_area_map_blend_write(BlendWriter *writer, ScrAreaMap *area_map)
|
|
{
|
|
BLO_write_struct_list(writer, ScrVert, &area_map->vertbase);
|
|
BLO_write_struct_list(writer, ScrEdge, &area_map->edgebase);
|
|
LISTBASE_FOREACH (ScrArea *, area, &area_map->areabase) {
|
|
area->butspacetype = area->spacetype; /* Just for compatibility, will be reset below. */
|
|
|
|
BLO_write_struct(writer, ScrArea, area);
|
|
|
|
BLO_write_struct(writer, ScrGlobalAreaData, area->global);
|
|
|
|
write_area_regions(writer, area);
|
|
|
|
area->butspacetype = SPACE_EMPTY; /* Unset again, was changed above. */
|
|
}
|
|
}
|
|
|
|
static void direct_link_panel_list(BlendDataReader *reader, ListBase *lb)
|
|
{
|
|
BLO_read_list(reader, lb);
|
|
|
|
LISTBASE_FOREACH (Panel *, panel, lb) {
|
|
panel->runtime_flag = 0;
|
|
panel->activedata = NULL;
|
|
panel->type = NULL;
|
|
panel->runtime.custom_data_ptr = NULL;
|
|
direct_link_panel_list(reader, &panel->children);
|
|
}
|
|
}
|
|
|
|
static void direct_link_region(BlendDataReader *reader, ARegion *region, int spacetype)
|
|
{
|
|
direct_link_panel_list(reader, ®ion->panels);
|
|
|
|
BLO_read_list(reader, ®ion->panels_category_active);
|
|
|
|
BLO_read_list(reader, ®ion->ui_lists);
|
|
|
|
/* The area's search filter is runtime only, so we need to clear the active flag on read. */
|
|
region->flag &= ~RGN_FLAG_SEARCH_FILTER_ACTIVE;
|
|
|
|
LISTBASE_FOREACH (uiList *, ui_list, ®ion->ui_lists) {
|
|
ui_list->type = NULL;
|
|
ui_list->dyn_data = NULL;
|
|
BLO_read_data_address(reader, &ui_list->properties);
|
|
IDP_BlendDataRead(reader, &ui_list->properties);
|
|
}
|
|
|
|
BLO_read_list(reader, ®ion->ui_previews);
|
|
|
|
if (spacetype == SPACE_EMPTY) {
|
|
/* unknown space type, don't leak regiondata */
|
|
region->regiondata = NULL;
|
|
}
|
|
else if (region->flag & RGN_FLAG_TEMP_REGIONDATA) {
|
|
/* Runtime data, don't use. */
|
|
region->regiondata = NULL;
|
|
}
|
|
else {
|
|
BLO_read_data_address(reader, ®ion->regiondata);
|
|
if (region->regiondata) {
|
|
if (spacetype == SPACE_VIEW3D) {
|
|
RegionView3D *rv3d = region->regiondata;
|
|
|
|
BLO_read_data_address(reader, &rv3d->localvd);
|
|
BLO_read_data_address(reader, &rv3d->clipbb);
|
|
|
|
rv3d->depths = NULL;
|
|
rv3d->render_engine = NULL;
|
|
rv3d->sms = NULL;
|
|
rv3d->smooth_timer = NULL;
|
|
|
|
rv3d->rflag &= ~(RV3D_NAVIGATING | RV3D_PAINTING);
|
|
rv3d->runtime_viewlock = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
region->v2d.sms = NULL;
|
|
region->v2d.alpha_hor = region->v2d.alpha_vert = 255; /* visible by default */
|
|
BLI_listbase_clear(®ion->panels_category);
|
|
BLI_listbase_clear(®ion->handlers);
|
|
BLI_listbase_clear(®ion->uiblocks);
|
|
region->headerstr = NULL;
|
|
region->visible = 0;
|
|
region->type = NULL;
|
|
region->do_draw = 0;
|
|
region->gizmo_map = NULL;
|
|
region->regiontimer = NULL;
|
|
region->draw_buffer = NULL;
|
|
memset(®ion->drawrct, 0, sizeof(region->drawrct));
|
|
}
|
|
|
|
/* for the saved 2.50 files without regiondata */
|
|
/* and as patch for 2.48 and older */
|
|
void BKE_screen_view3d_do_versions_250(View3D *v3d, ListBase *regions)
|
|
{
|
|
LISTBASE_FOREACH (ARegion *, region, regions) {
|
|
if (region->regiontype == RGN_TYPE_WINDOW && region->regiondata == NULL) {
|
|
RegionView3D *rv3d;
|
|
|
|
rv3d = region->regiondata = MEM_callocN(sizeof(RegionView3D), "region v3d patch");
|
|
rv3d->persp = (char)v3d->persp;
|
|
rv3d->view = (char)v3d->view;
|
|
rv3d->dist = v3d->dist;
|
|
copy_v3_v3(rv3d->ofs, v3d->ofs);
|
|
copy_qt_qt(rv3d->viewquat, v3d->viewquat);
|
|
}
|
|
}
|
|
|
|
/* this was not initialized correct always */
|
|
if (v3d->gridsubdiv == 0) {
|
|
v3d->gridsubdiv = 10;
|
|
}
|
|
}
|
|
|
|
static void direct_link_area(BlendDataReader *reader, ScrArea *area)
|
|
{
|
|
BLO_read_list(reader, &(area->spacedata));
|
|
BLO_read_list(reader, &(area->regionbase));
|
|
|
|
BLI_listbase_clear(&area->handlers);
|
|
area->type = NULL; /* spacetype callbacks */
|
|
|
|
/* Should always be unset so that rna_Area_type_get works correctly. */
|
|
area->butspacetype = SPACE_EMPTY;
|
|
|
|
area->region_active_win = -1;
|
|
|
|
area->flag &= ~AREA_FLAG_ACTIVE_TOOL_UPDATE;
|
|
|
|
BLO_read_data_address(reader, &area->global);
|
|
|
|
/* if we do not have the spacetype registered we cannot
|
|
* free it, so don't allocate any new memory for such spacetypes. */
|
|
if (!BKE_spacetype_exists(area->spacetype)) {
|
|
/* Hint for versioning code to replace deprecated space types. */
|
|
area->butspacetype = area->spacetype;
|
|
|
|
area->spacetype = SPACE_EMPTY;
|
|
}
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, &area->regionbase) {
|
|
direct_link_region(reader, region, area->spacetype);
|
|
}
|
|
|
|
/* accident can happen when read/save new file with older version */
|
|
/* 2.50: we now always add spacedata for info */
|
|
if (area->spacedata.first == NULL) {
|
|
SpaceInfo *sinfo = MEM_callocN(sizeof(SpaceInfo), "spaceinfo");
|
|
area->spacetype = sinfo->spacetype = SPACE_INFO;
|
|
BLI_addtail(&area->spacedata, sinfo);
|
|
}
|
|
/* add local view3d too */
|
|
else if (area->spacetype == SPACE_VIEW3D) {
|
|
BKE_screen_view3d_do_versions_250(area->spacedata.first, &area->regionbase);
|
|
}
|
|
|
|
LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) {
|
|
BLO_read_list(reader, &(sl->regionbase));
|
|
|
|
/* if we do not have the spacetype registered we cannot
|
|
* free it, so don't allocate any new memory for such spacetypes. */
|
|
if (!BKE_spacetype_exists(sl->spacetype)) {
|
|
sl->spacetype = SPACE_EMPTY;
|
|
}
|
|
|
|
LISTBASE_FOREACH (ARegion *, region, &sl->regionbase) {
|
|
direct_link_region(reader, region, sl->spacetype);
|
|
}
|
|
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
View3D *v3d = (View3D *)sl;
|
|
|
|
v3d->flag |= V3D_INVALID_BACKBUF;
|
|
|
|
if (v3d->gpd) {
|
|
BLO_read_data_address(reader, &v3d->gpd);
|
|
BKE_gpencil_blend_read_data(reader, v3d->gpd);
|
|
}
|
|
BLO_read_data_address(reader, &v3d->localvd);
|
|
|
|
/* Runtime data */
|
|
v3d->runtime.properties_storage = NULL;
|
|
v3d->runtime.flag = 0;
|
|
|
|
/* render can be quite heavy, set to solid on load */
|
|
if (v3d->shading.type == OB_RENDER) {
|
|
v3d->shading.type = OB_SOLID;
|
|
}
|
|
v3d->shading.prev_type = OB_SOLID;
|
|
|
|
BKE_screen_view3d_shading_blend_read_data(reader, &v3d->shading);
|
|
|
|
BKE_screen_view3d_do_versions_250(v3d, &sl->regionbase);
|
|
}
|
|
else if (sl->spacetype == SPACE_GRAPH) {
|
|
SpaceGraph *sipo = (SpaceGraph *)sl;
|
|
|
|
BLO_read_data_address(reader, &sipo->ads);
|
|
BLI_listbase_clear(&sipo->runtime.ghost_curves);
|
|
}
|
|
else if (sl->spacetype == SPACE_NLA) {
|
|
SpaceNla *snla = (SpaceNla *)sl;
|
|
|
|
BLO_read_data_address(reader, &snla->ads);
|
|
}
|
|
else if (sl->spacetype == SPACE_OUTLINER) {
|
|
SpaceOutliner *space_outliner = (SpaceOutliner *)sl;
|
|
|
|
/* use #BLO_read_get_new_data_address_no_us and do not free old memory avoiding double
|
|
* frees and use of freed memory. this could happen because of a
|
|
* bug fixed in revision 58959 where the treestore memory address
|
|
* was not unique */
|
|
TreeStore *ts = BLO_read_get_new_data_address_no_us(reader, space_outliner->treestore);
|
|
space_outliner->treestore = NULL;
|
|
if (ts) {
|
|
TreeStoreElem *elems = BLO_read_get_new_data_address_no_us(reader, ts->data);
|
|
|
|
space_outliner->treestore = BLI_mempool_create(
|
|
sizeof(TreeStoreElem), ts->usedelem, 512, BLI_MEMPOOL_ALLOW_ITER);
|
|
if (ts->usedelem && elems) {
|
|
for (int i = 0; i < ts->usedelem; i++) {
|
|
TreeStoreElem *new_elem = BLI_mempool_alloc(space_outliner->treestore);
|
|
*new_elem = elems[i];
|
|
}
|
|
}
|
|
/* we only saved what was used */
|
|
space_outliner->storeflag |= SO_TREESTORE_CLEANUP; /* at first draw */
|
|
}
|
|
space_outliner->tree.first = space_outliner->tree.last = NULL;
|
|
space_outliner->runtime = NULL;
|
|
}
|
|
else if (sl->spacetype == SPACE_IMAGE) {
|
|
SpaceImage *sima = (SpaceImage *)sl;
|
|
|
|
sima->iuser.scene = NULL;
|
|
sima->iuser.ok = 1;
|
|
sima->scopes.waveform_1 = NULL;
|
|
sima->scopes.waveform_2 = NULL;
|
|
sima->scopes.waveform_3 = NULL;
|
|
sima->scopes.vecscope = NULL;
|
|
sima->scopes.ok = 0;
|
|
|
|
/* WARNING: gpencil data is no longer stored directly in sima after 2.5
|
|
* so sacrifice a few old files for now to avoid crashes with new files!
|
|
* committed: r28002 */
|
|
#if 0
|
|
sima->gpd = newdataadr(fd, sima->gpd);
|
|
if (sima->gpd) {
|
|
BKE_gpencil_blend_read_data(fd, sima->gpd);
|
|
}
|
|
#endif
|
|
}
|
|
else if (sl->spacetype == SPACE_NODE) {
|
|
SpaceNode *snode = (SpaceNode *)sl;
|
|
|
|
if (snode->gpd) {
|
|
BLO_read_data_address(reader, &snode->gpd);
|
|
BKE_gpencil_blend_read_data(reader, snode->gpd);
|
|
}
|
|
|
|
BLO_read_list(reader, &snode->treepath);
|
|
snode->edittree = NULL;
|
|
snode->iofsd = NULL;
|
|
BLI_listbase_clear(&snode->linkdrag);
|
|
}
|
|
else if (sl->spacetype == SPACE_TEXT) {
|
|
SpaceText *st = (SpaceText *)sl;
|
|
memset(&st->runtime, 0, sizeof(st->runtime));
|
|
}
|
|
else if (sl->spacetype == SPACE_SEQ) {
|
|
SpaceSeq *sseq = (SpaceSeq *)sl;
|
|
|
|
/* grease pencil data is not a direct data and can't be linked from direct_link*
|
|
* functions, it should be linked from lib_link* functions instead
|
|
*
|
|
* otherwise it'll lead to lost grease data on open because it'll likely be
|
|
* read from file after all other users of grease pencil and newdataadr would
|
|
* simple return NULL here (sergey)
|
|
*/
|
|
#if 0
|
|
if (sseq->gpd) {
|
|
sseq->gpd = newdataadr(fd, sseq->gpd);
|
|
BKE_gpencil_blend_read_data(fd, sseq->gpd);
|
|
}
|
|
#endif
|
|
sseq->scopes.reference_ibuf = NULL;
|
|
sseq->scopes.zebra_ibuf = NULL;
|
|
sseq->scopes.waveform_ibuf = NULL;
|
|
sseq->scopes.sep_waveform_ibuf = NULL;
|
|
sseq->scopes.vector_ibuf = NULL;
|
|
sseq->scopes.histogram_ibuf = NULL;
|
|
}
|
|
else if (sl->spacetype == SPACE_PROPERTIES) {
|
|
SpaceProperties *sbuts = (SpaceProperties *)sl;
|
|
|
|
sbuts->path = NULL;
|
|
sbuts->texuser = NULL;
|
|
sbuts->mainbo = sbuts->mainb;
|
|
sbuts->mainbuser = sbuts->mainb;
|
|
sbuts->runtime = NULL;
|
|
}
|
|
else if (sl->spacetype == SPACE_CONSOLE) {
|
|
SpaceConsole *sconsole = (SpaceConsole *)sl;
|
|
|
|
BLO_read_list(reader, &sconsole->scrollback);
|
|
BLO_read_list(reader, &sconsole->history);
|
|
|
|
/* comma expressions, (e.g. expr1, expr2, expr3) evaluate each expression,
|
|
* from left to right. the right-most expression sets the result of the comma
|
|
* expression as a whole*/
|
|
LISTBASE_FOREACH_MUTABLE (ConsoleLine *, cl, &sconsole->history) {
|
|
BLO_read_data_address(reader, &cl->line);
|
|
if (cl->line) {
|
|
/* the allocted length is not written, so reset here */
|
|
cl->len_alloc = cl->len + 1;
|
|
}
|
|
else {
|
|
BLI_remlink(&sconsole->history, cl);
|
|
MEM_freeN(cl);
|
|
}
|
|
}
|
|
}
|
|
else if (sl->spacetype == SPACE_FILE) {
|
|
SpaceFile *sfile = (SpaceFile *)sl;
|
|
|
|
/* this sort of info is probably irrelevant for reloading...
|
|
* plus, it isn't saved to files yet!
|
|
*/
|
|
sfile->folders_prev = sfile->folders_next = NULL;
|
|
BLI_listbase_clear(&sfile->folder_histories);
|
|
sfile->files = NULL;
|
|
sfile->layout = NULL;
|
|
sfile->op = NULL;
|
|
sfile->previews_timer = NULL;
|
|
sfile->tags = 0;
|
|
BLO_read_data_address(reader, &sfile->params);
|
|
BLO_read_data_address(reader, &sfile->asset_params);
|
|
}
|
|
else if (sl->spacetype == SPACE_CLIP) {
|
|
SpaceClip *sclip = (SpaceClip *)sl;
|
|
|
|
sclip->scopes.track_search = NULL;
|
|
sclip->scopes.track_preview = NULL;
|
|
sclip->scopes.ok = 0;
|
|
}
|
|
}
|
|
|
|
BLI_listbase_clear(&area->actionzones);
|
|
|
|
BLO_read_data_address(reader, &area->v1);
|
|
BLO_read_data_address(reader, &area->v2);
|
|
BLO_read_data_address(reader, &area->v3);
|
|
BLO_read_data_address(reader, &area->v4);
|
|
}
|
|
|
|
/**
|
|
* \return false on error.
|
|
*/
|
|
bool BKE_screen_area_map_blend_read_data(BlendDataReader *reader, ScrAreaMap *area_map)
|
|
{
|
|
BLO_read_list(reader, &area_map->vertbase);
|
|
BLO_read_list(reader, &area_map->edgebase);
|
|
BLO_read_list(reader, &area_map->areabase);
|
|
LISTBASE_FOREACH (ScrArea *, area, &area_map->areabase) {
|
|
direct_link_area(reader, area);
|
|
}
|
|
|
|
/* edges */
|
|
LISTBASE_FOREACH (ScrEdge *, se, &area_map->edgebase) {
|
|
BLO_read_data_address(reader, &se->v1);
|
|
BLO_read_data_address(reader, &se->v2);
|
|
BKE_screen_sort_scrvert(&se->v1, &se->v2);
|
|
|
|
if (se->v1 == NULL) {
|
|
BLI_remlink(&area_map->edgebase, se);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void BKE_screen_area_blend_read_lib(BlendLibReader *reader, ID *parent_id, ScrArea *area)
|
|
{
|
|
BLO_read_id_address(reader, parent_id->lib, &area->full);
|
|
|
|
memset(&area->runtime, 0x0, sizeof(area->runtime));
|
|
|
|
LISTBASE_FOREACH (SpaceLink *, sl, &area->spacedata) {
|
|
switch (sl->spacetype) {
|
|
case SPACE_VIEW3D: {
|
|
View3D *v3d = (View3D *)sl;
|
|
|
|
BLO_read_id_address(reader, parent_id->lib, &v3d->camera);
|
|
BLO_read_id_address(reader, parent_id->lib, &v3d->ob_center);
|
|
|
|
if (v3d->localvd) {
|
|
BLO_read_id_address(reader, parent_id->lib, &v3d->localvd->camera);
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_GRAPH: {
|
|
SpaceGraph *sipo = (SpaceGraph *)sl;
|
|
bDopeSheet *ads = sipo->ads;
|
|
|
|
if (ads) {
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->source);
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->filter_grp);
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_PROPERTIES: {
|
|
SpaceProperties *sbuts = (SpaceProperties *)sl;
|
|
BLO_read_id_address(reader, parent_id->lib, &sbuts->pinid);
|
|
if (sbuts->pinid == NULL) {
|
|
sbuts->flag &= ~SB_PIN_CONTEXT;
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_FILE: {
|
|
SpaceFile *sfile = (SpaceFile *)sl;
|
|
sfile->tags |= FILE_TAG_REBUILD_MAIN_FILES;
|
|
break;
|
|
}
|
|
case SPACE_ACTION: {
|
|
SpaceAction *saction = (SpaceAction *)sl;
|
|
bDopeSheet *ads = &saction->ads;
|
|
|
|
if (ads) {
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->source);
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->filter_grp);
|
|
}
|
|
|
|
BLO_read_id_address(reader, parent_id->lib, &saction->action);
|
|
break;
|
|
}
|
|
case SPACE_IMAGE: {
|
|
SpaceImage *sima = (SpaceImage *)sl;
|
|
|
|
BLO_read_id_address(reader, parent_id->lib, &sima->image);
|
|
BLO_read_id_address(reader, parent_id->lib, &sima->mask_info.mask);
|
|
|
|
/* NOTE: pre-2.5, this was local data not lib data, but now we need this as lib data
|
|
* so fingers crossed this works fine!
|
|
*/
|
|
BLO_read_id_address(reader, parent_id->lib, &sima->gpd);
|
|
break;
|
|
}
|
|
case SPACE_SEQ: {
|
|
SpaceSeq *sseq = (SpaceSeq *)sl;
|
|
|
|
/* NOTE: pre-2.5, this was local data not lib data, but now we need this as lib data
|
|
* so fingers crossed this works fine!
|
|
*/
|
|
BLO_read_id_address(reader, parent_id->lib, &sseq->gpd);
|
|
break;
|
|
}
|
|
case SPACE_NLA: {
|
|
SpaceNla *snla = (SpaceNla *)sl;
|
|
bDopeSheet *ads = snla->ads;
|
|
|
|
if (ads) {
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->source);
|
|
BLO_read_id_address(reader, parent_id->lib, &ads->filter_grp);
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_TEXT: {
|
|
SpaceText *st = (SpaceText *)sl;
|
|
|
|
BLO_read_id_address(reader, parent_id->lib, &st->text);
|
|
break;
|
|
}
|
|
case SPACE_SCRIPT: {
|
|
SpaceScript *scpt = (SpaceScript *)sl;
|
|
/*scpt->script = NULL; - 2.45 set to null, better re-run the script */
|
|
if (scpt->script) {
|
|
BLO_read_id_address(reader, parent_id->lib, &scpt->script);
|
|
if (scpt->script) {
|
|
SCRIPT_SET_NULL(scpt->script);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_OUTLINER: {
|
|
SpaceOutliner *space_outliner = (SpaceOutliner *)sl;
|
|
BLO_read_id_address(reader, NULL, &space_outliner->search_tse.id);
|
|
|
|
if (space_outliner->treestore) {
|
|
TreeStoreElem *tselem;
|
|
BLI_mempool_iter iter;
|
|
|
|
BLI_mempool_iternew(space_outliner->treestore, &iter);
|
|
while ((tselem = BLI_mempool_iterstep(&iter))) {
|
|
BLO_read_id_address(reader, NULL, &tselem->id);
|
|
}
|
|
/* rebuild hash table, because it depends on ids too */
|
|
space_outliner->storeflag |= SO_TREESTORE_REBUILD;
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_NODE: {
|
|
SpaceNode *snode = (SpaceNode *)sl;
|
|
|
|
/* node tree can be stored locally in id too, link this first */
|
|
BLO_read_id_address(reader, parent_id->lib, &snode->id);
|
|
BLO_read_id_address(reader, parent_id->lib, &snode->from);
|
|
|
|
bNodeTree *ntree = snode->id ? ntreeFromID(snode->id) : NULL;
|
|
if (ntree) {
|
|
snode->nodetree = ntree;
|
|
}
|
|
else {
|
|
BLO_read_id_address(reader, parent_id->lib, &snode->nodetree);
|
|
}
|
|
|
|
bNodeTreePath *path;
|
|
for (path = snode->treepath.first; path; path = path->next) {
|
|
if (path == snode->treepath.first) {
|
|
/* first nodetree in path is same as snode->nodetree */
|
|
path->nodetree = snode->nodetree;
|
|
}
|
|
else {
|
|
BLO_read_id_address(reader, parent_id->lib, &path->nodetree);
|
|
}
|
|
|
|
if (!path->nodetree) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* remaining path entries are invalid, remove */
|
|
bNodeTreePath *path_next;
|
|
for (; path; path = path_next) {
|
|
path_next = path->next;
|
|
|
|
BLI_remlink(&snode->treepath, path);
|
|
MEM_freeN(path);
|
|
}
|
|
|
|
/* edittree is just the last in the path,
|
|
* set this directly since the path may have been shortened above */
|
|
if (snode->treepath.last) {
|
|
path = snode->treepath.last;
|
|
snode->edittree = path->nodetree;
|
|
}
|
|
else {
|
|
snode->edittree = NULL;
|
|
}
|
|
break;
|
|
}
|
|
case SPACE_CLIP: {
|
|
SpaceClip *sclip = (SpaceClip *)sl;
|
|
BLO_read_id_address(reader, parent_id->lib, &sclip->clip);
|
|
BLO_read_id_address(reader, parent_id->lib, &sclip->mask_info.mask);
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|