| 
									
										
										
										
											2011-02-23 10:52:22 +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 | 
					
						
							| 
									
										
										
										
											2018-06-01 18:19:39 +02: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 wm | 
					
						
							| 
									
										
										
										
											2014-01-19 23:14:24 +11:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |  * User level access for blend file read/write, file-history and user-preferences | 
					
						
							|  |  |  |  * (including relevant operators). | 
					
						
							| 
									
										
										
										
											2011-02-25 14:04:21 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 01:24:16 +00:00
										 |  |  | /* placed up here because of crappy
 | 
					
						
							|  |  |  |  * winsock stuff. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2011-04-12 15:55:38 +00:00
										 |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-05 18:50:17 +00:00
										 |  |  | #include "zlib.h" /* wm_read_exotic() */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #ifdef WIN32
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | /* Need to include windows.h so _WIN32_IE is defined. */ | 
					
						
							| 
									
										
										
										
											2019-01-14 16:15:33 +11:00
										 |  |  | #  include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2012-04-15 07:54:07 +00:00
										 |  |  | #  ifndef _WIN32_IE
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | /* Minimal requirements for SHGetSpecialFolderPath on MINGW MSVC has this defined already. */ | 
					
						
							| 
									
										
										
										
											2019-01-14 16:15:33 +11:00
										 |  |  | #    define _WIN32_IE 0x0400
 | 
					
						
							| 
									
										
										
										
											2012-04-15 07:54:07 +00:00
										 |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | /* For SHGetSpecialFolderPath, has to be done before BLI_winstuff
 | 
					
						
							| 
									
										
										
										
											2019-04-18 07:21:26 +02:00
										 |  |  |  * because 'near' is disabled through BLI_windstuff */ | 
					
						
							| 
									
										
										
										
											2012-04-15 07:54:07 +00:00
										 |  |  | #  include "BLI_winstuff.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #  include <shlobj.h>
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-05 19:30:37 +00:00
										 |  |  | #include "MEM_CacheLimiterC-Api.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_blenlib.h"
 | 
					
						
							| 
									
										
										
										
											2020-10-15 10:54:06 +11:00
										 |  |  | #include "BLI_fileops_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BLI_linklist.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-02 11:43:54 +02:00
										 |  |  | #include "BLI_system.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_threads.h"
 | 
					
						
							|  |  |  | #include "BLI_timer.h"
 | 
					
						
							|  |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-02 11:43:54 +02:00
										 |  |  | #include BLI_SYSTEM_PID_H
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-16 17:32:01 +10:00
										 |  |  | #include "BLT_translation.h"
 | 
					
						
							| 
									
										
										
										
											2011-06-15 08:28:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-26 14:38:47 +02:00
										 |  |  | #include "BLF_api.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							| 
									
										
											  
											
												Giant commit!
A full detailed description of this will be done later... is several days
of work. Here's a summary:
Render:
- Full cleanup of render code, removing *all* globals and bad level calls
  all over blender. Render module is now not called abusive anymore
- API-fied calls to rendering
- Full recode of internal render pipeline. Is now rendering tiles by
  default, prepared for much smarter 'bucket' render later.
- Each thread now can render a full part
- Renders were tested with 4 threads, goes fine, apart from some lookup
  tables in softshadow and AO still
- Rendering is prepared to do multiple layers and passes
- No single 32 bits trick in render code anymore, all 100% floats now.
Writing images/movies
- moved writing images to blender kernel (bye bye 'schrijfplaatje'!)
- made a new Movie handle system, also in kernel. This will enable much
  easier use of movies in Blender
PreviewRender:
- Using new render API, previewrender (in buttons) now uses regular render
  code to generate images.
- new datafile 'preview.blend.c' has the preview scenes in it
- previews get rendered in exact displayed size (1 pixel = 1 pixel)
3D Preview render
- new; press Pkey in 3d window, for a panel that continuously renders
  (pkey is for games, i know... but we dont do that in orange now!)
- this render works nearly identical to buttons-preview render, so it stops
  rendering on any event (mouse, keyboard, etc)
- on moving/scaling the panel, the render code doesn't recreate all geometry
- same for shifting/panning view
- all other operations (now) regenerate the full render database still.
- this is WIP... but big fun, especially for simple scenes!
Compositor
- Using same node system as now in use for shaders, you can composit images
- works pretty straightforward... needs much more options/tools and integration
  with rendering still
- is not threaded yet, nor is so smart to only recalculate changes... will be
  done soon!
- the "Render Result" node will get all layers/passes as output sockets
- The "Output" node renders to a builtin image, which you can view in the Image
  window. (yes, output nodes to render-result, and to files, is on the list!)
The Bad News
- "Unified Render" is removed. It might come back in some stage, but this
  system should be built from scratch. I can't really understand this code...
  I expect it is not much needed, especially with advanced layer/passes
  control
- Panorama render, Field render, Motion blur, is not coded yet... (I had to
  recode every single feature in render, so...!)
- Lens Flare is also not back... needs total revision, might become composit
  effect though (using zbuffer for visibility)
- Part render is gone! (well, thats obvious, its default now).
- The render window is only restored with limited functionality... I am going
  to check first the option to render to a Image window, so Blender can become
  a true single-window application. :)
  For example, the 'Spare render buffer' (jkey) doesnt work.
- Render with border, now default creates a smaller image
- No zbuffers are written yet... on the todo!
- Scons files and MSVC will need work to get compiling again
OK... thats what I can quickly recall. Now go compiling!
											
										 
											2006-01-23 22:05:47 +00:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2006-07-04 12:10:37 +00:00
										 |  |  | #include "DNA_screen_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_space_types.h"
 | 
					
						
							|  |  |  | #include "DNA_userdef_types.h"
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | #include "DNA_windowmanager_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-13 12:02:08 +02:00
										 |  |  | #include "DNA_workspace_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-02 23:23:57 +10:00
										 |  |  | #include "BKE_addon.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-23 14:37:13 +01:00
										 |  |  | #include "BKE_appdir.h"
 | 
					
						
							| 
									
										
										
										
											2013-06-18 18:11:52 +00:00
										 |  |  | #include "BKE_autoexec.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_blender.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:42:41 +10:00
										 |  |  | #include "BKE_blendfile.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-05 15:52:38 +02:00
										 |  |  | #include "BKE_callbacks.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-18 02:56:48 +00:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_global.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | #include "BKE_idprop.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-05 16:17:14 +01:00
										 |  |  | #include "BKE_lib_id.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 12:58:59 +01:00
										 |  |  | #include "BKE_lib_override.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_main.h"
 | 
					
						
							|  |  |  | #include "BKE_packedFile.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-19 00:50:21 +00:00
										 |  |  | #include "BKE_report.h"
 | 
					
						
							| 
									
										
										
										
											2015-10-08 15:56:46 +11:00
										 |  |  | #include "BKE_scene.h"
 | 
					
						
							| 
									
										
										
										
											2012-04-17 20:51:11 +00:00
										 |  |  | #include "BKE_screen.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BKE_sound.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-19 14:17:59 +01:00
										 |  |  | #include "BKE_undo_system.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
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLO_readfile.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #include "BLO_undofile.h" /* to save from an undo memfile */
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLO_writefile.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 10:00:46 +00:00
										 |  |  | #include "RNA_access.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | #include "RNA_define.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-07 10:00:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | #include "IMB_imbuf.h"
 | 
					
						
							|  |  |  | #include "IMB_imbuf_types.h"
 | 
					
						
							|  |  |  | #include "IMB_thumbs.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												
Port of part of the Interface code to 2.50.
This is based on the current trunk version, so these files should not need
merges. There's two things (clipboard and intptr_t) that are missing in 2.50
and commented out with XXX 2.48, these can be enabled again once trunk is
merged into this branch.
Further this is not all interface code, there are many parts commented out:
* interface.c: nearly all button types, missing: links, chartab, keyevent.
* interface_draw.c: almost all code, with some small exceptions.
* interface_ops.c: this replaces ui_do_but and uiDoBlocks with two operators,
  making it non-blocking. 
* interface_regions: this is a part of interface.c, split off, contains code to
  create regions for tooltips, menus, pupmenu (that one is crashing currently),
  color chooser, basically regions with buttons which is fairly independent of
  core interface code.
* interface_panel.c and interface_icons.c: not ported over, so no panels and
  icons yet. Panels should probably become (free floating) regions? 
* text.c: (formerly language.c) for drawing text and translation. this works
  but is using bad globals still and could be cleaned up.
Header Files:
* ED_datafiles.h now has declarations for datatoc_ files, so those extern
  declarations can be #included instead of repeated.
* The user interface code is in UI_interface.h and other UI_* files.
Core:
* The API for creating blocks, buttons, etc is nearly the same still. Blocks
  are now created per region instead of per area.
* The code was made non-blocking, which means that any changes and redraws
  should be possible while editing a button. That means though that we need
  some sort of persistence even though the blender model is to recreate buttons
  for each redraw. So when a new block is created, some matching happens to
  find out which buttons correspond to buttons in the previously created block,
  and for activated buttons some data is then copied over to the new button.
* Added UI_init/UI_init_userdef/UI_exit functions that should initialize code
  in this module, instead of multiple function calls in the windowmanager.
* Removed most static/globals from interface.c.
* Removed UIafterfunc_ I don't think it's needed anymore, and not sure how it
  would integrate here?
* Currently only full window redraws are used, this should become per region
  and maybe per button later.
Operators:
* Events are currently handled through two operators: button activate and menu
  handle. Operators may not be the best way to implement this, since there are
  currently some issues with events being missed, but they can become a special
  handler type instead, this should not be a big change.
* The button activate operator runs as long as a button is active, and will
  handle all interaction with that button until the button is not activated
  anymore. This means clicking, text editing, number dragging, opening menu
  blocks, etc.
* Since this operator has to be non-blocking, the ui_do_but code needed to made
  non-blocking. That means variables that were previously on the stack, now
  need to be stored away in a struct such that they can be accessed again when
  the operator receives more events.
* Additionally the place in the ui_do_but code indicated the state, now that
  needs to be set explicit in order to handle the right events in the right
  state. So an activated button can be in one of these states: init, highlight,
  wait_flash, wait_release, wait_key_event, num_editing, text_editing,
  text_selecting, block_open, exit.
* For each button type an ui_apply_but_* function has also been separated out
  from ui_do_but. This makes it possible to continuously apply the button as
  text is being typed for example, and there is an option in the code to enable
  this. Since the code non-blocking and can deal with the button being deleted
  even, it should be safe to do this.
* When editing text, dragging numbers, etc, the actual data (but->poin) is not
  being edited, since that would mean data is being edited without correct
  updates happening, while some other part of blender may be accessing that
  data in the meantime. So data values, strings, vectors are written to a
  temporary location and only flush in the apply function.
Regions:
* Menus, color chooser, tooltips etc all create screen level regions. Such menu
  blocks give a handle to the button that creates it, which will contain the
  results of the menu block once a MESSAGE event is received from that menu
  block.
* For this type of menu block the coordinates used to be in window space. They
  are still created that way and ui_positionblock still works with window
  coordinates, but after that the block and buttons are brought back to region
  coordinates since these are now contained in a region.
* The flush/overdraw frontbuffer drawing code was removed, the windowmanager
  should have enough information with these screen level regions to have full
  control over what gets drawn when and to then do correct compositing.
Testing:
* The header in the time space currently has some buttons to test the UI code.
											
										 
											2008-11-11 18:31:32 +00:00
										 |  |  | #include "ED_datafiles.h"
 | 
					
						
							| 
									
										
										
										
											2013-12-20 17:39:22 +01:00
										 |  |  | #include "ED_fileselect.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | #include "ED_image.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-28 20:46:50 -06:00
										 |  |  | #include "ED_outliner.h"
 | 
					
						
							| 
									
										
										
										
											2008-11-27 17:58:46 +00:00
										 |  |  | #include "ED_screen.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "ED_undo.h"
 | 
					
						
							|  |  |  | #include "ED_util.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | #include "ED_view3d.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-06 12:08:03 +01:00
										 |  |  | #include "ED_view3d_offscreen.h"
 | 
					
						
							| 
									
										
										
										
											2008-11-27 17:58:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-23 17:42:36 +00:00
										 |  |  | #include "GHOST_C-api.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-05 14:56:10 +00:00
										 |  |  | #include "GHOST_Path-api.h"
 | 
					
						
							| 
									
										
										
										
											2009-01-23 17:42:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												
Port of part of the Interface code to 2.50.
This is based on the current trunk version, so these files should not need
merges. There's two things (clipboard and intptr_t) that are missing in 2.50
and commented out with XXX 2.48, these can be enabled again once trunk is
merged into this branch.
Further this is not all interface code, there are many parts commented out:
* interface.c: nearly all button types, missing: links, chartab, keyevent.
* interface_draw.c: almost all code, with some small exceptions.
* interface_ops.c: this replaces ui_do_but and uiDoBlocks with two operators,
  making it non-blocking. 
* interface_regions: this is a part of interface.c, split off, contains code to
  create regions for tooltips, menus, pupmenu (that one is crashing currently),
  color chooser, basically regions with buttons which is fairly independent of
  core interface code.
* interface_panel.c and interface_icons.c: not ported over, so no panels and
  icons yet. Panels should probably become (free floating) regions? 
* text.c: (formerly language.c) for drawing text and translation. this works
  but is using bad globals still and could be cleaned up.
Header Files:
* ED_datafiles.h now has declarations for datatoc_ files, so those extern
  declarations can be #included instead of repeated.
* The user interface code is in UI_interface.h and other UI_* files.
Core:
* The API for creating blocks, buttons, etc is nearly the same still. Blocks
  are now created per region instead of per area.
* The code was made non-blocking, which means that any changes and redraws
  should be possible while editing a button. That means though that we need
  some sort of persistence even though the blender model is to recreate buttons
  for each redraw. So when a new block is created, some matching happens to
  find out which buttons correspond to buttons in the previously created block,
  and for activated buttons some data is then copied over to the new button.
* Added UI_init/UI_init_userdef/UI_exit functions that should initialize code
  in this module, instead of multiple function calls in the windowmanager.
* Removed most static/globals from interface.c.
* Removed UIafterfunc_ I don't think it's needed anymore, and not sure how it
  would integrate here?
* Currently only full window redraws are used, this should become per region
  and maybe per button later.
Operators:
* Events are currently handled through two operators: button activate and menu
  handle. Operators may not be the best way to implement this, since there are
  currently some issues with events being missed, but they can become a special
  handler type instead, this should not be a big change.
* The button activate operator runs as long as a button is active, and will
  handle all interaction with that button until the button is not activated
  anymore. This means clicking, text editing, number dragging, opening menu
  blocks, etc.
* Since this operator has to be non-blocking, the ui_do_but code needed to made
  non-blocking. That means variables that were previously on the stack, now
  need to be stored away in a struct such that they can be accessed again when
  the operator receives more events.
* Additionally the place in the ui_do_but code indicated the state, now that
  needs to be set explicit in order to handle the right events in the right
  state. So an activated button can be in one of these states: init, highlight,
  wait_flash, wait_release, wait_key_event, num_editing, text_editing,
  text_selecting, block_open, exit.
* For each button type an ui_apply_but_* function has also been separated out
  from ui_do_but. This makes it possible to continuously apply the button as
  text is being typed for example, and there is an option in the code to enable
  this. Since the code non-blocking and can deal with the button being deleted
  even, it should be safe to do this.
* When editing text, dragging numbers, etc, the actual data (but->poin) is not
  being edited, since that would mean data is being edited without correct
  updates happening, while some other part of blender may be accessing that
  data in the meantime. So data values, strings, vectors are written to a
  temporary location and only flush in the apply function.
Regions:
* Menus, color chooser, tooltips etc all create screen level regions. Such menu
  blocks give a handle to the button that creates it, which will contain the
  results of the menu block once a MESSAGE event is received from that menu
  block.
* For this type of menu block the coordinates used to be in window space. They
  are still created that way and ui_positionblock still works with window
  coordinates, but after that the block and buttons are brought back to region
  coordinates since these are now contained in a region.
* The flush/overdraw frontbuffer drawing code was removed, the windowmanager
  should have enough information with these screen level regions to have full
  control over what gets drawn when and to then do correct compositing.
Testing:
* The header in the time space currently has some buttons to test the UI code.
											
										 
											2008-11-11 18:31:32 +00:00
										 |  |  | #include "UI_interface.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | #include "UI_resources.h"
 | 
					
						
							| 
									
										
										
										
											2014-01-20 12:43:56 +11:00
										 |  |  | #include "UI_view2d.h"
 | 
					
						
							| 
									
										
											  
											
												Giant commit!
A full detailed description of this will be done later... is several days
of work. Here's a summary:
Render:
- Full cleanup of render code, removing *all* globals and bad level calls
  all over blender. Render module is now not called abusive anymore
- API-fied calls to rendering
- Full recode of internal render pipeline. Is now rendering tiles by
  default, prepared for much smarter 'bucket' render later.
- Each thread now can render a full part
- Renders were tested with 4 threads, goes fine, apart from some lookup
  tables in softshadow and AO still
- Rendering is prepared to do multiple layers and passes
- No single 32 bits trick in render code anymore, all 100% floats now.
Writing images/movies
- moved writing images to blender kernel (bye bye 'schrijfplaatje'!)
- made a new Movie handle system, also in kernel. This will enable much
  easier use of movies in Blender
PreviewRender:
- Using new render API, previewrender (in buttons) now uses regular render
  code to generate images.
- new datafile 'preview.blend.c' has the preview scenes in it
- previews get rendered in exact displayed size (1 pixel = 1 pixel)
3D Preview render
- new; press Pkey in 3d window, for a panel that continuously renders
  (pkey is for games, i know... but we dont do that in orange now!)
- this render works nearly identical to buttons-preview render, so it stops
  rendering on any event (mouse, keyboard, etc)
- on moving/scaling the panel, the render code doesn't recreate all geometry
- same for shifting/panning view
- all other operations (now) regenerate the full render database still.
- this is WIP... but big fun, especially for simple scenes!
Compositor
- Using same node system as now in use for shaders, you can composit images
- works pretty straightforward... needs much more options/tools and integration
  with rendering still
- is not threaded yet, nor is so smart to only recalculate changes... will be
  done soon!
- the "Render Result" node will get all layers/passes as output sockets
- The "Output" node renders to a builtin image, which you can view in the Image
  window. (yes, output nodes to render-result, and to files, is on the list!)
The Bad News
- "Unified Render" is removed. It might come back in some stage, but this
  system should be built from scratch. I can't really understand this code...
  I expect it is not much needed, especially with advanced layer/passes
  control
- Panorama render, Field render, Motion blur, is not coded yet... (I had to
  recode every single feature in render, so...!)
- Lens Flare is also not back... needs total revision, might become composit
  effect though (using zbuffer for visibility)
- Part render is gone! (well, thats obvious, its default now).
- The render window is only restored with limited functionality... I am going
  to check first the option to render to a Image window, so Blender can become
  a true single-window application. :)
  For example, the 'Spare render buffer' (jkey) doesnt work.
- Render with border, now default creates a smaller image
- No zbuffers are written yet... on the todo!
- Scons files and MSVC will need work to get compiling again
OK... thats what I can quickly recall. Now go compiling!
											
										 
											2006-01-23 22:05:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-22 13:30:12 +10:00
										 |  |  | /* only to report a missing engine */ | 
					
						
							|  |  |  | #include "RE_engine.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-31 04:11:39 +00:00
										 |  |  | #ifdef WITH_PYTHON
 | 
					
						
							| 
									
										
										
										
											2020-08-17 17:20:12 +10:00
										 |  |  | #  include "BPY_extern_python.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-17 17:46:06 +10:00
										 |  |  | #  include "BPY_extern_run.h"
 | 
					
						
							| 
									
										
										
										
											2010-09-12 12:09:31 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 16:44:32 +02:00
										 |  |  | #include "DEG_depsgraph.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | #include "WM_api.h"
 | 
					
						
							| 
									
										
										
										
											2017-11-13 19:43:34 +11:00
										 |  |  | #include "WM_message.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-23 08:20:46 +02:00
										 |  |  | #include "WM_toolsystem.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "WM_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-23 08:20:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | #include "wm.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "wm_event_system.h"
 | 
					
						
							| 
									
										
										
										
											2011-02-14 18:20:10 +00:00
										 |  |  | #include "wm_files.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-13 13:33:01 +00:00
										 |  |  | #include "wm_window.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | static RecentFile *wm_file_history_find(const char *filepath); | 
					
						
							|  |  |  | static void wm_history_file_free(RecentFile *recent); | 
					
						
							| 
									
										
										
										
											2021-02-21 17:29:52 +01:00
										 |  |  | static void wm_history_files_free(void); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | static void wm_history_file_update(void); | 
					
						
							|  |  |  | static void wm_history_file_write(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  | static void wm_test_autorun_revert_action_exec(bContext *C); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Misc Utility Functions
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_file_tag_modified(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   wmWindowManager *wm = G_MAIN->wm.first; | 
					
						
							|  |  |  |   if (wm->file_saved) { | 
					
						
							|  |  |  |     wm->file_saved = 0; | 
					
						
							|  |  |  |     /* notifier that data changed, for save-over warning or header */ | 
					
						
							|  |  |  |     WM_main_add_notifier(NC_WM | ND_DATACHANGED, NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool wm_file_or_image_is_modified(const Main *bmain, const wmWindowManager *wm) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return !wm->file_saved || ED_image_should_save_modified(bmain); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Window Matching for File Reading
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * To be able to read files without windows closing, opening, moving | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  |  * we try to prepare for worst case: | 
					
						
							|  |  |  |  * - active window gets active screen from file | 
					
						
							|  |  |  |  * - restoring the screens from non-active windows | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  |  * Best case is all screens match, in that case they get assigned to proper window. | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | static void wm_window_match_init(bContext *C, ListBase *wmlist) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   *wmlist = G_MAIN->wm; | 
					
						
							|  |  |  |   BLI_listbase_clear(&G_MAIN->wm); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |   wmWindow *active_win = CTX_wm_window(C); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* first wrap up running stuff */ | 
					
						
							|  |  |  |   /* code copied from wm_init_exit.c */ | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |   LISTBASE_FOREACH (wmWindowManager *, wm, wmlist) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WM_jobs_kill_all(wm); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |     LISTBASE_FOREACH (wmWindow *, win, &wm->windows) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       CTX_wm_window_set(C, win); /* needed by operator close callbacks */ | 
					
						
							|  |  |  |       WM_event_remove_handlers(C, &win->handlers); | 
					
						
							|  |  |  |       WM_event_remove_handlers(C, &win->modalhandlers); | 
					
						
							|  |  |  |       ED_screen_exit(C, win, WM_window_get_active_screen(win)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* reset active window */ | 
					
						
							|  |  |  |   CTX_wm_window_set(C, active_win); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* XXX Hack! We have to clear context menu here, because removing all modalhandlers
 | 
					
						
							|  |  |  |    * above frees the active menu (at least, in the 'startup splash' case), | 
					
						
							|  |  |  |    * causing use-after-free error in later handling of the button callbacks in UI code | 
					
						
							|  |  |  |    * (see ui_apply_but_funcs_after()). | 
					
						
							|  |  |  |    * Tried solving this by always NULL-ing context's menu when setting wm/win/etc., | 
					
						
							|  |  |  |    * but it broke popups refreshing (see T47632), | 
					
						
							|  |  |  |    * so for now just handling this specific case here. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   CTX_wm_menu_set(C, NULL); | 
					
						
							| 
									
										
										
										
											2010-01-13 21:37:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ED_editors_exit(G_MAIN, true); | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2006-08-20 15:22:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void wm_window_substitute_old(wmWindowManager *oldwm, | 
					
						
							|  |  |  |                                      wmWindowManager *wm, | 
					
						
							|  |  |  |                                      wmWindow *oldwin, | 
					
						
							|  |  |  |                                      wmWindow *win) | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   win->ghostwin = oldwin->ghostwin; | 
					
						
							|  |  |  |   win->gpuctx = oldwin->gpuctx; | 
					
						
							|  |  |  |   win->active = oldwin->active; | 
					
						
							|  |  |  |   if (win->active) { | 
					
						
							|  |  |  |     wm->winactive = win; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (oldwm->windrawable == oldwin) { | 
					
						
							|  |  |  |     oldwm->windrawable = NULL; | 
					
						
							|  |  |  |     wm->windrawable = win; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* File loading in background mode still calls this. */ | 
					
						
							|  |  |  |   if (!G.background) { | 
					
						
							|  |  |  |     /* Pointer back. */ | 
					
						
							|  |  |  |     GHOST_SetWindowUserData(win->ghostwin, win); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   oldwin->ghostwin = NULL; | 
					
						
							|  |  |  |   oldwin->gpuctx = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   win->eventstate = oldwin->eventstate; | 
					
						
							|  |  |  |   oldwin->eventstate = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ensure proper screen rescaling */ | 
					
						
							|  |  |  |   win->sizex = oldwin->sizex; | 
					
						
							|  |  |  |   win->sizey = oldwin->sizey; | 
					
						
							|  |  |  |   win->posx = oldwin->posx; | 
					
						
							|  |  |  |   win->posy = oldwin->posy; | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void wm_window_match_keep_current_wm(const bContext *C, | 
					
						
							|  |  |  |                                             ListBase *current_wm_list, | 
					
						
							|  |  |  |                                             const bool load_ui, | 
					
						
							|  |  |  |                                             ListBase *r_new_wm_list) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   wmWindowManager *wm = current_wm_list->first; | 
					
						
							|  |  |  |   bScreen *screen = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* match oldwm to new dbase, only old files */ | 
					
						
							| 
									
										
										
										
											2020-08-01 13:02:21 +10:00
										 |  |  |   wm->initialized &= ~WM_WINDOW_IS_INIT; | 
					
						
							| 
									
										
										
										
											2009-01-27 17:12:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* when loading without UI, no matching needed */ | 
					
						
							|  |  |  |   if (load_ui && (screen = CTX_wm_screen(C))) { | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |     LISTBASE_FOREACH (wmWindow *, win, &wm->windows) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       WorkSpace *workspace; | 
					
						
							| 
									
										
										
										
											2011-09-28 19:05:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       BKE_workspace_layout_find_global(bmain, screen, &workspace); | 
					
						
							|  |  |  |       BKE_workspace_active_set(win->workspace_hook, workspace); | 
					
						
							|  |  |  |       win->scene = CTX_data_scene(C); | 
					
						
							| 
									
										
											  
											
												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
										 |  |  |       /* all windows get active screen from file */ | 
					
						
							|  |  |  |       if (screen->winid == 0) { | 
					
						
							|  |  |  |         WM_window_set_active_screen(win, workspace, screen); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         WorkSpaceLayout *layout_old = WM_window_get_active_layout(win); | 
					
						
							|  |  |  |         WorkSpaceLayout *layout_new = ED_workspace_layout_duplicate( | 
					
						
							|  |  |  |             bmain, workspace, layout_old, win); | 
					
						
							| 
									
										
										
										
											2009-09-28 15:34:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         WM_window_set_active_layout(win, workspace, layout_new); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2010-06-26 13:24:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       bScreen *win_screen = WM_window_get_active_screen(win); | 
					
						
							|  |  |  |       win_screen->winid = win->winid; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *r_new_wm_list = *current_wm_list; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void wm_window_match_replace_by_file_wm(bContext *C, | 
					
						
							|  |  |  |                                                ListBase *current_wm_list, | 
					
						
							|  |  |  |                                                ListBase *readfile_wm_list, | 
					
						
							|  |  |  |                                                ListBase *r_new_wm_list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   wmWindowManager *oldwm = current_wm_list->first; | 
					
						
							|  |  |  |   wmWindowManager *wm = readfile_wm_list->first; /* will become our new WM */ | 
					
						
							|  |  |  |   bool has_match = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* this code could move to setup_appdata */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* preserve key configurations in new wm, to preserve their keymaps */ | 
					
						
							|  |  |  |   wm->keyconfigs = oldwm->keyconfigs; | 
					
						
							|  |  |  |   wm->addonconf = oldwm->addonconf; | 
					
						
							|  |  |  |   wm->defaultconf = oldwm->defaultconf; | 
					
						
							|  |  |  |   wm->userconf = oldwm->userconf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_listbase_clear(&oldwm->keyconfigs); | 
					
						
							|  |  |  |   oldwm->addonconf = NULL; | 
					
						
							|  |  |  |   oldwm->defaultconf = NULL; | 
					
						
							|  |  |  |   oldwm->userconf = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ensure making new keymaps and set space types */ | 
					
						
							|  |  |  |   wm->initialized = 0; | 
					
						
							|  |  |  |   wm->winactive = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Clearing drawable of before deleting any context
 | 
					
						
							|  |  |  |    * to avoid clearing the wrong wm. */ | 
					
						
							|  |  |  |   wm_window_clear_drawable(oldwm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* only first wm in list has ghostwins */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (wmWindow *, win, &wm->windows) { | 
					
						
							|  |  |  |     LISTBASE_FOREACH (wmWindow *, oldwin, &oldwm->windows) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (oldwin->winid == win->winid) { | 
					
						
							|  |  |  |         has_match = true; | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         wm_window_substitute_old(oldwm, wm, oldwin, win); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* make sure at least one window is kept open so we don't lose the context, check T42303 */ | 
					
						
							|  |  |  |   if (!has_match) { | 
					
						
							|  |  |  |     wm_window_substitute_old(oldwm, wm, oldwm->windows.first, wm->windows.first); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-11-09 04:57:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_close_and_free_all(C, current_wm_list); | 
					
						
							| 
									
										
										
										
											2014-10-19 20:50:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   *r_new_wm_list = *readfile_wm_list; | 
					
						
							| 
									
										
										
										
											2017-11-09 04:57:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Match old WM with new, 4 cases: | 
					
						
							|  |  |  |  * 1) No current WM, no WM in file: Make new default. | 
					
						
							|  |  |  |  * 2) No current WM, but WM in file: Keep current WM, do nothing else. | 
					
						
							|  |  |  |  * 3) Current WM, but not in file: Keep current WM, update windows with screens from file. | 
					
						
							|  |  |  |  * 4) Current WM, and WM in file: Try to keep current GHOST windows, use WM from file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param r_new_wm_list: Return argument for the wm list to be used from now on. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void wm_window_match_do(bContext *C, | 
					
						
							|  |  |  |                                ListBase *current_wm_list, | 
					
						
							|  |  |  |                                ListBase *readfile_wm_list, | 
					
						
							|  |  |  |                                ListBase *r_new_wm_list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (BLI_listbase_is_empty(current_wm_list)) { | 
					
						
							|  |  |  |     /* case 1 */ | 
					
						
							|  |  |  |     if (BLI_listbase_is_empty(readfile_wm_list)) { | 
					
						
							|  |  |  |       Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |       /* Neither current, no newly read file have a WM -> add the default one. */ | 
					
						
							|  |  |  |       wm_add_default(bmain, C); | 
					
						
							|  |  |  |       *r_new_wm_list = bmain->wm; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* case 2 */ | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       *r_new_wm_list = *readfile_wm_list; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* case 3 */ | 
					
						
							|  |  |  |     if (BLI_listbase_is_empty(readfile_wm_list)) { | 
					
						
							|  |  |  |       /* We've read file without wm, keep current one entirely alive.
 | 
					
						
							|  |  |  |        * Happens when reading pre 2.5 files (no WM back then) */ | 
					
						
							|  |  |  |       wm_window_match_keep_current_wm( | 
					
						
							|  |  |  |           C, current_wm_list, (G.fileflags & G_FILE_NO_UI) == 0, r_new_wm_list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* case 4 */ | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       wm_window_match_replace_by_file_wm(C, current_wm_list, readfile_wm_list, r_new_wm_list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Preferences Initialization & Versioning
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-02 23:23:57 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * In case #UserDef was read, re-initialize values that depend on it. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-07-31 21:46:13 +10:00
										 |  |  | static void wm_init_userdef(Main *bmain) | 
					
						
							| 
									
										
											  
											
												2.5
Summary of ain features:
- Themes and Styles are now editable.
- CTRL+U "Save user defaults" now goes to new .B25.blend, so you
  can use 2.4x and 2.5x next to each other. If B25 doesn't exist, it
  reads the regular .B.blend
- Press Tkey in 3d window for (unfinished) toolbar WIP. It now only
  shows the last operator, if appropriate.
  Nkey properties moved to the other side.
A lot of work was done on removing old themes for good and properly
getting it work with the 2.5 region system. Here's some notes;
- Buttons now all have a complete set of colors, based on button classifications
  (See outliner -> user prefs -> Interface
- Theme colors have been extended with basic colors for region types. 
  Currently colors are defined for Window, Header, List/Channels and
  for Button/Tool views. 
  The screen manager handles this btw, so a TH_BACK will always pick the
  right backdrop color.
- Menu backdrops are in in Button theme colors. Floating Panels will be in
  the per-space type Themes.
- Styles were added in RNA too, but only for the font settings now.
  Only Panel font, widget font and widget-label work now. The 'group label'
  will be for templates mostly.
  Style settings will be expanded with spacing defaults, label conventions, 
  etc.
- Label text colors are stored in per-space Theme too, to make sure they fit.
  Same goes for Panel title color.
Note that 'shadow' for fonts can conflict with text colors; shadow color is
currently stored in Style... shadow code needs a bit of work still.
											
										 
											2009-04-27 13:44:11 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-10-02 23:23:57 +10:00
										 |  |  |   /* Not versioning, just avoid errors. */ | 
					
						
							|  |  |  | #ifndef WITH_CYCLES
 | 
					
						
							|  |  |  |   BKE_addon_remove_safe(&U.addons, "cycles"); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   UNUSED_VARS(BKE_addon_remove_safe); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-02 20:11:51 +10:00
										 |  |  |   UI_init_userdef(); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-30 20:09:02 +10:00
										 |  |  |   /* needed so loading a file from the command line respects user-pref T26156. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(G.fileflags, U.flag & USER_FILENOUI, G_FILE_NO_UI); | 
					
						
							| 
									
										
										
										
											2011-02-21 02:40:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* set the python auto-execute setting from user prefs */ | 
					
						
							|  |  |  |   /* enabled by default, unless explicitly enabled in the command line which overrides */ | 
					
						
							|  |  |  |   if ((G.f & G_FLAG_SCRIPT_OVERRIDE_PREF) == 0) { | 
					
						
							|  |  |  |     SET_FLAG_FROM_TEST(G.f, (U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0, G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-10-21 17:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 12:06:57 +02:00
										 |  |  |   MEM_CacheLimiter_set_maximum(((size_t)U.memcachelimit) * 1024 * 1024); | 
					
						
							|  |  |  |   BKE_sound_init(bmain); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-05 01:25:20 +11:00
										 |  |  |   /* Update the temporary directory from the preferences or fallback to the system default. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BKE_tempdir_init(U.tempdir); | 
					
						
							| 
									
										
										
										
											2020-04-08 12:16:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Update tablet API preference. */ | 
					
						
							|  |  |  |   WM_init_tablet_api(); | 
					
						
							| 
									
										
										
										
											2020-10-02 23:23:57 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   BLO_sanitize_experimental_features_userpref_blend(&U); | 
					
						
							| 
									
										
											  
											
												2.5
Summary of ain features:
- Themes and Styles are now editable.
- CTRL+U "Save user defaults" now goes to new .B25.blend, so you
  can use 2.4x and 2.5x next to each other. If B25 doesn't exist, it
  reads the regular .B.blend
- Press Tkey in 3d window for (unfinished) toolbar WIP. It now only
  shows the last operator, if appropriate.
  Nkey properties moved to the other side.
A lot of work was done on removing old themes for good and properly
getting it work with the 2.5 region system. Here's some notes;
- Buttons now all have a complete set of colors, based on button classifications
  (See outliner -> user prefs -> Interface
- Theme colors have been extended with basic colors for region types. 
  Currently colors are defined for Window, Header, List/Channels and
  for Button/Tool views. 
  The screen manager handles this btw, so a TH_BACK will always pick the
  right backdrop color.
- Menu backdrops are in in Button theme colors. Floating Panels will be in
  the per-space type Themes.
- Styles were added in RNA too, but only for the font settings now.
  Only Panel font, widget font and widget-label work now. The 'group label'
  will be for templates mostly.
  Style settings will be expanded with spacing defaults, label conventions, 
  etc.
- Label text colors are stored in per-space Theme too, to make sure they fit.
  Same goes for Panel title color.
Note that 'shadow' for fonts can conflict with text colors; shadow color is
currently stored in Style... shadow code needs a bit of work still.
											
										 
											2009-04-27 13:44:11 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | /* return codes */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define BKE_READ_EXOTIC_FAIL_PATH -3   /* file format is not supported */
 | 
					
						
							|  |  |  | #define BKE_READ_EXOTIC_FAIL_FORMAT -2 /* file format is not supported */
 | 
					
						
							|  |  |  | #define BKE_READ_EXOTIC_FAIL_OPEN -1   /* Can't open the file */
 | 
					
						
							|  |  |  | #define BKE_READ_EXOTIC_OK_BLEND 0     /* .blend file */
 | 
					
						
							| 
									
										
										
										
											2015-10-08 14:22:16 +11:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #  define BKE_READ_EXOTIC_OK_OTHER 1 /* other supported formats */
 | 
					
						
							| 
									
										
										
										
											2015-10-08 14:22:16 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Exotic File Formats
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Currently only supports '.blend' files, | 
					
						
							|  |  |  |  * we could support registering other file formats and their loaders. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | /* intended to check for non-blender formats but for now it only reads blends */ | 
					
						
							| 
									
										
										
										
											2015-10-08 14:22:16 +11:00
										 |  |  | static int wm_read_exotic(const char *name) | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int len; | 
					
						
							|  |  |  |   gzFile gzfile; | 
					
						
							|  |  |  |   char header[7]; | 
					
						
							|  |  |  |   int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* make sure we're not trying to read a directory.... */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   len = strlen(name); | 
					
						
							|  |  |  |   if (len > 0 && ELEM(name[len - 1], '/', '\\')) { | 
					
						
							|  |  |  |     retval = BKE_READ_EXOTIC_FAIL_PATH; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     gzfile = BLI_gzopen(name, "rb"); | 
					
						
							|  |  |  |     if (gzfile == NULL) { | 
					
						
							|  |  |  |       retval = BKE_READ_EXOTIC_FAIL_OPEN; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       len = gzread(gzfile, header, sizeof(header)); | 
					
						
							|  |  |  |       gzclose(gzfile); | 
					
						
							|  |  |  |       if (len == sizeof(header) && STREQLEN(header, "BLENDER", 7)) { | 
					
						
							|  |  |  |         retval = BKE_READ_EXOTIC_OK_BLEND; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* We may want to support loading other file formats
 | 
					
						
							|  |  |  |          * from their header bytes or file extension. | 
					
						
							|  |  |  |          * This used to be supported in the code below and may be added | 
					
						
							|  |  |  |          * back at some point. */ | 
					
						
							| 
									
										
										
										
											2019-04-12 00:58:54 +10:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         WM_cursor_wait(true); | 
					
						
							| 
									
										
										
										
											2012-05-21 09:00:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         if (is_foo_format(name)) { | 
					
						
							|  |  |  |           read_foo(name); | 
					
						
							|  |  |  |           retval = BKE_READ_EXOTIC_OK_OTHER; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |           retval = BKE_READ_EXOTIC_FAIL_FORMAT; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-21 09:00:35 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         WM_cursor_wait(false); | 
					
						
							| 
									
										
										
										
											2012-05-21 09:00:35 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return retval; | 
					
						
							| 
									
										
										
										
											2011-05-05 14:21:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Blend-File Shared Utilities
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-18 18:11:52 +00:00
										 |  |  | void WM_file_autoexec_init(const char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G.f & G_FLAG_SCRIPT_OVERRIDE_PREF) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-06-18 18:11:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G.f & G_FLAG_SCRIPT_AUTOEXEC) { | 
					
						
							|  |  |  |     char path[FILE_MAX]; | 
					
						
							|  |  |  |     BLI_split_dir_part(filepath, path, sizeof(path)); | 
					
						
							|  |  |  |     if (BKE_autoexec_match(path)) { | 
					
						
							|  |  |  |       G.f &= ~G_FLAG_SCRIPT_AUTOEXEC; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-06-18 18:11:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 12:34:13 +02:00
										 |  |  | void wm_file_read_report(bContext *C, Main *bmain) | 
					
						
							| 
									
										
										
										
											2015-09-22 16:45:23 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ReportList *reports = NULL; | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |   LISTBASE_FOREACH (Scene *, scene, &bmain->scenes) { | 
					
						
							|  |  |  |     if (scene->r.engine[0] && | 
					
						
							|  |  |  |         BLI_findstring(&R_engines, scene->r.engine, offsetof(RenderEngineType, idname)) == NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (reports == NULL) { | 
					
						
							|  |  |  |         reports = CTX_wm_reports(C); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-09-22 16:45:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       BKE_reportf(reports, | 
					
						
							|  |  |  |                   RPT_ERROR, | 
					
						
							|  |  |  |                   "Engine '%s' not available for scene '%s' (an add-on may need to be installed " | 
					
						
							|  |  |  |                   "or enabled)", | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |                   scene->r.engine, | 
					
						
							|  |  |  |                   scene->id.name + 2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-09-22 16:45:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (reports) { | 
					
						
							|  |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       WM_report_banner_show(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-09-22 16:45:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:02:12 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Logic shared between #WM_file_read & #wm_homefile_read, | 
					
						
							|  |  |  |  * call before loading a file. | 
					
						
							|  |  |  |  * \note In the case of #WM_file_read the file may fail to load. | 
					
						
							|  |  |  |  * Change here shouldn't cause user-visible changes in that case. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void wm_file_read_pre(bContext *C, bool use_data, bool UNUSED(use_userdef)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (use_data) { | 
					
						
							|  |  |  |     BKE_callback_exec_null(CTX_data_main(C), BKE_CB_EVT_LOAD_PRE); | 
					
						
							|  |  |  |     BLI_timer_on_file_load(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Always do this as both startup and preferences may have loaded in many font's
 | 
					
						
							|  |  |  |    * at a different zoom level to the file being loaded. */ | 
					
						
							|  |  |  |   UI_view2d_zoom_cache_reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Logic shared between #WM_file_read & #wm_homefile_read, | 
					
						
							|  |  |  |  * updates to make after reading a file. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void wm_file_read_post(bContext *C, | 
					
						
							|  |  |  |                               const bool is_startup_file, | 
					
						
							|  |  |  |                               const bool is_factory_startup, | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |                               const bool use_data, | 
					
						
							| 
									
										
										
										
											2019-05-22 14:28:10 +10:00
										 |  |  |                               const bool use_userdef, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                               const bool reset_app_template) | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool addons_loaded = false; | 
					
						
							|  |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							| 
									
										
										
										
											2015-10-07 00:27:27 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       /* remove windows which failed to be added via WM_check */ | 
					
						
							|  |  |  |       wm_window_ghostwindows_remove_invalid(C, wm); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     CTX_wm_window_set(C, wm->windows.first); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-10-07 00:27:27 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | #ifdef WITH_PYTHON
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (is_startup_file) { | 
					
						
							| 
									
										
										
										
											2021-02-16 15:51:04 +11:00
										 |  |  |     /* On startup (by default), Python won't have been initialized.
 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * The following block handles data & preferences being reloaded | 
					
						
							|  |  |  |      * which requires resetting some internal variables. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (CTX_py_init_get(C)) { | 
					
						
							| 
									
										
										
										
											2020-06-03 12:45:27 +10:00
										 |  |  |       bool reset_all = use_userdef; | 
					
						
							|  |  |  |       if (use_userdef || reset_app_template) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         /* Only run when we have a template path found. */ | 
					
						
							|  |  |  |         if (BKE_appdir_app_template_any()) { | 
					
						
							| 
									
										
										
										
											2020-08-17 18:16:54 +10:00
										 |  |  |           BPY_run_string_eval( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |               C, (const char *[]){"bl_app_template_utils", NULL}, "bl_app_template_utils.reset()"); | 
					
						
							| 
									
										
										
										
											2020-06-03 12:45:27 +10:00
										 |  |  |           reset_all = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-03 12:45:27 +10:00
										 |  |  |       } | 
					
						
							|  |  |  |       if (reset_all) { | 
					
						
							| 
									
										
										
										
											2021-02-16 15:51:04 +11:00
										 |  |  |         BPY_run_string_exec( | 
					
						
							|  |  |  |             C, | 
					
						
							|  |  |  |             (const char *[]){"bpy", "addon_utils", NULL}, | 
					
						
							|  |  |  |             /* Refresh scripts as the preferences may have changed the user-scripts path.
 | 
					
						
							|  |  |  |              * | 
					
						
							|  |  |  |              * This is needed when loading settings from the previous version, | 
					
						
							|  |  |  |              * otherwise the script path stored in the preferences would be ignored. */ | 
					
						
							|  |  |  |             "bpy.utils.refresh_script_paths()\n" | 
					
						
							|  |  |  |             /* Sync add-ons, these may have changed from the defaults. */ | 
					
						
							|  |  |  |             "addon_utils.reset_all()"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |       if (use_data) { | 
					
						
							|  |  |  |         BPY_python_reset(C); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       addons_loaded = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* run any texts that were loaded in and flagged as modules */ | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     if (use_data) { | 
					
						
							|  |  |  |       BPY_python_reset(C); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     addons_loaded = true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-03-02 11:09:24 +11:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   UNUSED_VARS(is_startup_file, reset_app_template); | 
					
						
							|  |  |  | #endif /* WITH_PYTHON */
 | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:28:10 +10:00
										 |  |  |   if (use_userdef) { | 
					
						
							|  |  |  |     if (is_factory_startup) { | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |       BKE_callback_exec_null(bmain, BKE_CB_EVT_LOAD_FACTORY_USERDEF_POST); | 
					
						
							| 
									
										
										
										
											2019-05-22 14:28:10 +10:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:28:10 +10:00
										 |  |  |   if (use_data) { | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     /* important to do before NULL'ing the context */ | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |     BKE_callback_exec_null(bmain, BKE_CB_EVT_VERSION_UPDATE); | 
					
						
							|  |  |  |     BKE_callback_exec_null(bmain, BKE_CB_EVT_LOAD_POST); | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     if (is_factory_startup) { | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |       BKE_callback_exec_null(bmain, BKE_CB_EVT_LOAD_FACTORY_STARTUP_POST); | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-22 14:28:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (use_data) { | 
					
						
							|  |  |  |     WM_operatortype_last_properties_clear_all(); | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* After load post, so for example the driver namespace can be filled
 | 
					
						
							|  |  |  |      * before evaluating the depsgraph. */ | 
					
						
							| 
									
										
										
										
											2019-06-05 18:31:04 +02:00
										 |  |  |     wm_event_do_depsgraph(C, true); | 
					
						
							| 
									
										
										
										
											2019-03-07 11:40:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     ED_editors_init(C); | 
					
						
							| 
									
										
										
										
											2019-03-07 11:40:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-13 19:43:34 +11:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     WM_event_add_notifier(C, NC_WM | ND_FILEREAD, NULL); | 
					
						
							| 
									
										
										
										
											2017-11-13 19:43:34 +11:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     WM_msg_publish_static(CTX_wm_message_bus(C), WM_MSG_STATICTYPE_FILE_READ); | 
					
						
							| 
									
										
										
										
											2017-11-13 19:43:34 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* report any errors.
 | 
					
						
							|  |  |  |    * currently disabled if addons aren't yet loaded */ | 
					
						
							|  |  |  |   if (addons_loaded) { | 
					
						
							|  |  |  |     wm_file_read_report(C, bmain); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       if (wm->undo_stack == NULL) { | 
					
						
							|  |  |  |         wm->undo_stack = BKE_undosys_stack_create(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         BKE_undosys_stack_clear(wm->undo_stack); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       BKE_undosys_stack_init_from_main(wm->undo_stack, bmain); | 
					
						
							|  |  |  |       BKE_undosys_stack_init_from_context(wm->undo_stack, C); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       /* in background mode this makes it hard to load
 | 
					
						
							|  |  |  |        * a blend file and do anything since the screen | 
					
						
							|  |  |  |        * won't be set to a valid value again */ | 
					
						
							|  |  |  |       CTX_wm_window_set(C, NULL); /* exits queues */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  |       /* Ensure auto-run action is not used from a previous blend file load. */ | 
					
						
							|  |  |  |       wm_test_autorun_revert_action_set(NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |       /* Ensure tools are registered. */ | 
					
						
							|  |  |  |       WM_toolsystem_init(C); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-09-22 12:50:41 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Main Blend-File API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-29 05:33:05 +11:00
										 |  |  | bool WM_file_read(bContext *C, const char *filepath, ReportList *reports) | 
					
						
							| 
									
										
										
										
											2007-12-24 18:27:28 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* assume automated tasks with background, don't write recent file list */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   const bool do_history_file_update = (G.background == false) && | 
					
						
							|  |  |  |                                       (CTX_wm_manager(C)->op_undo_depth == 0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool success = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:02:12 +10:00
										 |  |  |   const bool use_data = true; | 
					
						
							|  |  |  |   const bool use_userdef = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* so we can get the error message */ | 
					
						
							|  |  |  |   errno = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 10:36:57 +01:00
										 |  |  |   WM_cursor_wait(true); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:02:12 +10:00
										 |  |  |   wm_file_read_pre(C, use_data, use_userdef); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* first try to append data from exotic file formats... */ | 
					
						
							|  |  |  |   /* it throws error box when file doesn't exist and returns -1 */ | 
					
						
							|  |  |  |   /* note; it should set some error message somewhere... (ton) */ | 
					
						
							| 
									
										
										
										
											2019-07-30 10:26:03 +10:00
										 |  |  |   const int retval = wm_read_exotic(filepath); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* we didn't succeed, now try to read Blender file */ | 
					
						
							|  |  |  |   if (retval == BKE_READ_EXOTIC_OK_BLEND) { | 
					
						
							|  |  |  |     const int G_f_orig = G.f; | 
					
						
							|  |  |  |     ListBase wmbase; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* put aside screens to match with persistent windows later */ | 
					
						
							|  |  |  |     /* also exit screens and editors */ | 
					
						
							|  |  |  |     wm_window_match_init(C, &wmbase); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 00:34:21 +11:00
										 |  |  |     const struct BlendFileReadParams params = { | 
					
						
							|  |  |  |         .is_startup = false, | 
					
						
							|  |  |  |         /* Loading preferences when the user intended to load a regular file is a security
 | 
					
						
							|  |  |  |          * risk, because the excluded path list is also loaded. Further it's just confusing | 
					
						
							|  |  |  |          * if a user loads a file and various preferences change. */ | 
					
						
							|  |  |  |         .skip_flags = BLO_READ_SKIP_USERDEF, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct BlendFileData *bfd = BKE_blendfile_read(filepath, ¶ms, reports); | 
					
						
							|  |  |  |     if (bfd != NULL) { | 
					
						
							|  |  |  |       BKE_blendfile_read_setup(C, bfd, ¶ms, reports); | 
					
						
							|  |  |  |       success = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* BKE_file_read sets new Main into context. */ | 
					
						
							|  |  |  |     Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 00:34:19 +11:00
										 |  |  |     /* When recovering a session from an unsaved file, this can have a blank path. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (BKE_main_blendfile_path(bmain)[0] != '\0') { | 
					
						
							|  |  |  |       G.save_over = 1; | 
					
						
							| 
									
										
										
										
											2021-03-12 00:34:19 +11:00
										 |  |  |       G.relbase_valid = 1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       G.save_over = 0; | 
					
						
							|  |  |  |       G.relbase_valid = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* this flag is initialized by the operator but overwritten on read.
 | 
					
						
							|  |  |  |      * need to re-enable it here else drivers + registered scripts wont work. */ | 
					
						
							|  |  |  |     if (G.f != G_f_orig) { | 
					
						
							|  |  |  |       const int flags_keep = G_FLAG_ALL_RUNTIME; | 
					
						
							|  |  |  |       G.f &= G_FLAG_ALL_READFILE; | 
					
						
							|  |  |  |       G.f = (G.f & ~flags_keep) | (G_f_orig & flags_keep); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* match the read WM with current WM */ | 
					
						
							|  |  |  |     wm_window_match_do(C, &wmbase, &bmain->wm, &bmain->wm); | 
					
						
							|  |  |  |     WM_check(C); /* opens window(s), checks keymaps */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-30 10:26:03 +10:00
										 |  |  |     if (success) { | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |       if (do_history_file_update) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         wm_history_file_update(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |     wm_file_read_post(C, false, false, use_data, use_userdef, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-10-08 14:22:16 +11:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-05-31 23:21:16 +10:00
										 |  |  |   else if (retval == BKE_READ_EXOTIC_OK_OTHER) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BKE_undo_write(C, "Import file"); | 
					
						
							| 
									
										
										
										
											2019-05-31 23:21:16 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-10-08 14:22:16 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   else if (retval == BKE_READ_EXOTIC_FAIL_OPEN) { | 
					
						
							|  |  |  |     BKE_reportf(reports, | 
					
						
							|  |  |  |                 RPT_ERROR, | 
					
						
							|  |  |  |                 "Cannot read file '%s': %s", | 
					
						
							|  |  |  |                 filepath, | 
					
						
							|  |  |  |                 errno ? strerror(errno) : TIP_("unable to open the file")); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (retval == BKE_READ_EXOTIC_FAIL_FORMAT) { | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_ERROR, "File format is not supported in file '%s'", filepath); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (retval == BKE_READ_EXOTIC_FAIL_PATH) { | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_ERROR, "File path '%s' invalid", filepath); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_ERROR, "Unknown error loading '%s'", filepath); | 
					
						
							|  |  |  |     BLI_assert(!"invalid 'retval'"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (success == false) { | 
					
						
							|  |  |  |     /* remove from recent files list */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |     if (do_history_file_update) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       RecentFile *recent = wm_file_history_find(filepath); | 
					
						
							|  |  |  |       if (recent) { | 
					
						
							|  |  |  |         wm_history_file_free(recent); | 
					
						
							|  |  |  |         wm_history_file_write(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 10:36:57 +01:00
										 |  |  |   WM_cursor_wait(false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return success; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 10:07:30 +11:00
										 |  |  | static struct { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char app_template[64]; | 
					
						
							|  |  |  |   bool override; | 
					
						
							| 
									
										
										
										
											2018-09-28 09:46:12 +02:00
										 |  |  | } wm_init_state_app_template = {{0}}; | 
					
						
							| 
									
										
										
										
											2018-08-17 16:34:51 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Used for setting app-template from the command line: | 
					
						
							|  |  |  |  * - non-empty string: overrides. | 
					
						
							|  |  |  |  * - empty string: override, using no app template. | 
					
						
							|  |  |  |  * - NULL: clears override. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void WM_init_state_app_template_set(const char *app_template) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (app_template) { | 
					
						
							|  |  |  |     STRNCPY(wm_init_state_app_template.app_template, app_template); | 
					
						
							|  |  |  |     wm_init_state_app_template.override = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     wm_init_state_app_template.app_template[0] = '\0'; | 
					
						
							|  |  |  |     wm_init_state_app_template.override = false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-08-17 16:34:51 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *WM_init_state_app_template_get(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return wm_init_state_app_template.override ? wm_init_state_app_template.app_template : NULL; | 
					
						
							| 
									
										
										
										
											2018-08-17 16:34:51 +10:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2006-07-04 12:10:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Startup & Preferences Blend-File API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 21:15:27 +11:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2017-03-17 00:37:28 +11:00
										 |  |  |  * Called on startup, (context entirely filled with NULLs) | 
					
						
							|  |  |  |  * or called for 'New File' both startup.blend and userpref.blend are checked. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |  * \param use_factory_settings: | 
					
						
							|  |  |  |  * Ignore on-disk startup file, use bundled ``datatoc_startup_blend`` instead. | 
					
						
							| 
									
										
										
										
											2017-03-17 00:37:28 +11:00
										 |  |  |  * Used for "Restore Factory Settings". | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2017-08-11 09:29:25 +10:00
										 |  |  |  * \param use_userdef: Load factory settings as well as startup file. | 
					
						
							|  |  |  |  * Disabled for "File New" we don't want to reload preferences. | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \param filepath_startup_override: | 
					
						
							|  |  |  |  * Optional path pointing to an alternative blend file (may be NULL). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param app_template_override: | 
					
						
							|  |  |  |  * Template to use instead of the template defined in user-preferences. | 
					
						
							| 
									
										
										
										
											2017-03-25 09:29:51 +11:00
										 |  |  |  * When not-null, this is written into the user preferences. | 
					
						
							| 
									
										
										
										
											2015-01-25 21:15:27 +11:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void wm_homefile_read(bContext *C, | 
					
						
							|  |  |  |                       ReportList *reports, | 
					
						
							|  |  |  |                       bool use_factory_settings, | 
					
						
							| 
									
										
										
										
											2020-06-03 00:54:00 +10:00
										 |  |  |                       bool use_empty_data, | 
					
						
							|  |  |  |                       bool use_data, | 
					
						
							|  |  |  |                       bool use_userdef, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                       const char *filepath_startup_override, | 
					
						
							|  |  |  |                       const char *app_template_override, | 
					
						
							|  |  |  |                       bool *r_is_factory_startup) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Main *bmain = G_MAIN; /* Context does not always have valid main pointer here... */ | 
					
						
							|  |  |  |   ListBase wmbase; | 
					
						
							|  |  |  |   bool success = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool filepath_startup_is_factory = true; | 
					
						
							|  |  |  |   char filepath_startup[FILE_MAX]; | 
					
						
							|  |  |  |   char filepath_userdef[FILE_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* When 'app_template' is set:
 | 
					
						
							|  |  |  |    * '{BLENDER_USER_CONFIG}/{app_template}' */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char app_template_system[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* When 'app_template' is set:
 | 
					
						
							|  |  |  |    * '{BLENDER_SYSTEM_SCRIPTS}/startup/bl_app_templates_system/{app_template}' */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char app_template_config[FILE_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   eBLOReadSkip skip_flags = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (use_data == false) { | 
					
						
							|  |  |  |     skip_flags |= BLO_READ_SKIP_DATA; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (use_userdef == false) { | 
					
						
							|  |  |  |     skip_flags |= BLO_READ_SKIP_USERDEF; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* True if we load startup.blend from memory
 | 
					
						
							|  |  |  |    * or use app-template startup.blend which the user hasn't saved. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool is_factory_startup = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 12:45:27 +10:00
										 |  |  |   const char *app_template = NULL; | 
					
						
							|  |  |  |   bool update_defaults = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (filepath_startup_override != NULL) { | 
					
						
							|  |  |  |     /* pass */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (app_template_override) { | 
					
						
							|  |  |  |     /* This may be clearing the current template by setting to an empty string. */ | 
					
						
							|  |  |  |     app_template = app_template_override; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (!use_factory_settings && U.app_template[0]) { | 
					
						
							|  |  |  |     app_template = U.app_template; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const bool reset_app_template = ((!app_template && U.app_template[0]) || | 
					
						
							|  |  |  |                                    (app_template && !STREQ(app_template, U.app_template))); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* options exclude eachother */ | 
					
						
							|  |  |  |   BLI_assert((use_factory_settings && filepath_startup_override) == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ((G.f & G_FLAG_SCRIPT_OVERRIDE_PREF) == 0) { | 
					
						
							|  |  |  |     SET_FLAG_FROM_TEST(G.f, (U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0, G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 14:05:18 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     if (reset_app_template) { | 
					
						
							|  |  |  |       /* Always load UI when switching to another template. */ | 
					
						
							|  |  |  |       G.fileflags &= ~G_FILE_NO_UI; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 14:34:18 +10:00
										 |  |  |   if (use_userdef || reset_app_template) { | 
					
						
							|  |  |  | #ifdef WITH_PYTHON
 | 
					
						
							|  |  |  |     /* This only runs once Blender has already started. */ | 
					
						
							|  |  |  |     if (CTX_py_init_get(C)) { | 
					
						
							|  |  |  |       /* This is restored by 'wm_file_read_post', disable before loading any preferences
 | 
					
						
							|  |  |  |        * so an add-on can read their own preferences when un-registering, | 
					
						
							|  |  |  |        * and use new preferences if/when re-registering, see T67577. | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        * Note that this fits into 'wm_file_read_pre' function but gets messy | 
					
						
							|  |  |  |        * since we need to know if 'reset_app_template' is true. */ | 
					
						
							| 
									
										
										
										
											2020-08-17 18:16:54 +10:00
										 |  |  |       BPY_run_string_eval(C, (const char *[]){"addon_utils", NULL}, "addon_utils.disable_all()"); | 
					
						
							| 
									
										
										
										
											2020-06-03 14:34:18 +10:00
										 |  |  |     } | 
					
						
							|  |  |  | #endif /* WITH_PYTHON */
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:02:12 +10:00
										 |  |  |   wm_file_read_pre(C, use_data, use_userdef); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:02:12 +10:00
										 |  |  |   if (use_data) { | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |     G.relbase_valid = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |     /* put aside screens to match with persistent windows later */ | 
					
						
							|  |  |  |     wm_window_match_init(C, &wmbase); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   filepath_startup[0] = '\0'; | 
					
						
							|  |  |  |   filepath_userdef[0] = '\0'; | 
					
						
							|  |  |  |   app_template_system[0] = '\0'; | 
					
						
							|  |  |  |   app_template_config[0] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const char *const cfgdir = BKE_appdir_folder_id(BLENDER_USER_CONFIG, NULL); | 
					
						
							|  |  |  |   if (!use_factory_settings) { | 
					
						
							|  |  |  |     if (cfgdir) { | 
					
						
							|  |  |  |       BLI_path_join( | 
					
						
							|  |  |  |           filepath_startup, sizeof(filepath_startup), cfgdir, BLENDER_STARTUP_FILE, NULL); | 
					
						
							|  |  |  |       filepath_startup_is_factory = false; | 
					
						
							|  |  |  |       if (use_userdef) { | 
					
						
							|  |  |  |         BLI_path_join( | 
					
						
							|  |  |  |             filepath_userdef, sizeof(filepath_startup), cfgdir, BLENDER_USERPREF_FILE, NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       use_factory_settings = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (filepath_startup_override) { | 
					
						
							|  |  |  |       BLI_strncpy(filepath_startup, filepath_startup_override, FILE_MAX); | 
					
						
							|  |  |  |       filepath_startup_is_factory = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* load preferences before startup.blend */ | 
					
						
							|  |  |  |   if (use_userdef) { | 
					
						
							|  |  |  |     if (!use_factory_settings && BLI_exists(filepath_userdef)) { | 
					
						
							|  |  |  |       UserDef *userdef = BKE_blendfile_userdef_read(filepath_userdef, NULL); | 
					
						
							|  |  |  |       if (userdef != NULL) { | 
					
						
							|  |  |  |         BKE_blender_userdef_data_set_and_free(userdef); | 
					
						
							|  |  |  |         userdef = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         skip_flags |= BLO_READ_SKIP_USERDEF; | 
					
						
							|  |  |  |         printf("Read prefs: %s\n", filepath_userdef); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ((app_template != NULL) && (app_template[0] != '\0')) { | 
					
						
							|  |  |  |     if (!BKE_appdir_app_template_id_search( | 
					
						
							|  |  |  |             app_template, app_template_system, sizeof(app_template_system))) { | 
					
						
							|  |  |  |       /* Can safely continue with code below, just warn it's not found. */ | 
					
						
							|  |  |  |       BKE_reportf(reports, RPT_WARNING, "Application Template '%s' not found", app_template); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Insert template name into startup file. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* note that the path is being set even when 'use_factory_settings == true'
 | 
					
						
							|  |  |  |      * this is done so we can load a templates factory-settings */ | 
					
						
							|  |  |  |     if (!use_factory_settings) { | 
					
						
							|  |  |  |       BLI_path_join(app_template_config, sizeof(app_template_config), cfgdir, app_template, NULL); | 
					
						
							|  |  |  |       BLI_path_join(filepath_startup, | 
					
						
							|  |  |  |                     sizeof(filepath_startup), | 
					
						
							|  |  |  |                     app_template_config, | 
					
						
							|  |  |  |                     BLENDER_STARTUP_FILE, | 
					
						
							|  |  |  |                     NULL); | 
					
						
							|  |  |  |       filepath_startup_is_factory = false; | 
					
						
							|  |  |  |       if (BLI_access(filepath_startup, R_OK) != 0) { | 
					
						
							|  |  |  |         filepath_startup[0] = '\0'; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       filepath_startup[0] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (filepath_startup[0] == '\0') { | 
					
						
							|  |  |  |       BLI_path_join(filepath_startup, | 
					
						
							|  |  |  |                     sizeof(filepath_startup), | 
					
						
							|  |  |  |                     app_template_system, | 
					
						
							|  |  |  |                     BLENDER_STARTUP_FILE, | 
					
						
							|  |  |  |                     NULL); | 
					
						
							|  |  |  |       filepath_startup_is_factory = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Update defaults only for system templates. */ | 
					
						
							|  |  |  |       update_defaults = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!use_factory_settings || (filepath_startup[0] != '\0')) { | 
					
						
							|  |  |  |     if (BLI_access(filepath_startup, R_OK) == 0) { | 
					
						
							| 
									
										
										
										
											2021-03-12 00:34:21 +11:00
										 |  |  |       const struct BlendFileReadParams params = { | 
					
						
							|  |  |  |           .is_startup = true, | 
					
						
							|  |  |  |           .skip_flags = skip_flags | BLO_READ_SKIP_USERDEF, | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       struct BlendFileData *bfd = BKE_blendfile_read(filepath_startup, ¶ms, NULL); | 
					
						
							|  |  |  |       if (bfd != NULL) { | 
					
						
							|  |  |  |         BKE_blendfile_read_setup_ex( | 
					
						
							|  |  |  |             C, bfd, ¶ms, NULL, update_defaults && use_data, app_template); | 
					
						
							|  |  |  |         success = true; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (success) { | 
					
						
							|  |  |  |       is_factory_startup = filepath_startup_is_factory; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-05 19:09:21 +10:00
										 |  |  |   if (use_userdef) { | 
					
						
							|  |  |  |     if ((skip_flags & BLO_READ_SKIP_USERDEF) == 0) { | 
					
						
							|  |  |  |       UserDef *userdef_default = BKE_blendfile_userdef_from_defaults(); | 
					
						
							|  |  |  |       BKE_blender_userdef_data_set_and_free(userdef_default); | 
					
						
							|  |  |  |       skip_flags |= BLO_READ_SKIP_USERDEF; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (success == false && filepath_startup_override && reports) { | 
					
						
							|  |  |  |     /* We can not return from here because wm is already reset */ | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_ERROR, "Could not read '%s'", filepath_startup_override); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (success == false) { | 
					
						
							| 
									
										
										
										
											2021-03-12 00:34:21 +11:00
										 |  |  |     const struct BlendFileReadParams params = { | 
					
						
							|  |  |  |         .is_startup = true, | 
					
						
							|  |  |  |         .skip_flags = skip_flags, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     struct BlendFileData *bfd = BKE_blendfile_read_from_memory( | 
					
						
							|  |  |  |         datatoc_startup_blend, datatoc_startup_blend_size, ¶ms, NULL); | 
					
						
							|  |  |  |     if (bfd != NULL) { | 
					
						
							|  |  |  |       BKE_blendfile_read_setup_ex(C, bfd, ¶ms, NULL, true, NULL); | 
					
						
							|  |  |  |       success = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-30 11:04:02 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 19:49:44 +02:00
										 |  |  |     if (use_data && BLI_listbase_is_empty(&wmbase)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       wm_clear_default_size(C); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (use_empty_data) { | 
					
						
							|  |  |  |     BKE_blendfile_read_make_empty(C); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Load template preferences,
 | 
					
						
							|  |  |  |    * unlike regular preferences we only use some of the settings, | 
					
						
							|  |  |  |    * see: BKE_blender_userdef_set_app_template */ | 
					
						
							|  |  |  |   if (app_template_system[0] != '\0') { | 
					
						
							|  |  |  |     char temp_path[FILE_MAX]; | 
					
						
							|  |  |  |     temp_path[0] = '\0'; | 
					
						
							|  |  |  |     if (!use_factory_settings) { | 
					
						
							|  |  |  |       BLI_path_join( | 
					
						
							|  |  |  |           temp_path, sizeof(temp_path), app_template_config, BLENDER_USERPREF_FILE, NULL); | 
					
						
							|  |  |  |       if (BLI_access(temp_path, R_OK) != 0) { | 
					
						
							|  |  |  |         temp_path[0] = '\0'; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (temp_path[0] == '\0') { | 
					
						
							|  |  |  |       BLI_path_join( | 
					
						
							|  |  |  |           temp_path, sizeof(temp_path), app_template_system, BLENDER_USERPREF_FILE, NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (use_userdef) { | 
					
						
							|  |  |  |       UserDef *userdef_template = NULL; | 
					
						
							|  |  |  |       /* just avoids missing file warning */ | 
					
						
							|  |  |  |       if (BLI_exists(temp_path)) { | 
					
						
							|  |  |  |         userdef_template = BKE_blendfile_userdef_read(temp_path, NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (userdef_template == NULL) { | 
					
						
							|  |  |  |         /* we need to have preferences load to overwrite preferences from previous template */ | 
					
						
							| 
									
										
										
										
											2019-07-30 11:04:02 +10:00
										 |  |  |         userdef_template = BKE_blendfile_userdef_from_defaults(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |       if (userdef_template) { | 
					
						
							|  |  |  |         BKE_blender_userdef_app_template_data_set_and_free(userdef_template); | 
					
						
							|  |  |  |         userdef_template = NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (app_template_override) { | 
					
						
							|  |  |  |     BLI_strncpy(U.app_template, app_template_override, sizeof(U.app_template)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (use_userdef) { | 
					
						
							|  |  |  |     /* check userdef before open window, keymaps etc */ | 
					
						
							| 
									
										
										
										
											2019-07-31 21:46:13 +10:00
										 |  |  |     wm_init_userdef(bmain); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     /* match the read WM with current WM */ | 
					
						
							|  |  |  |     wm_window_match_do(C, &wmbase, &bmain->wm, &bmain->wm); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-25 16:31:02 +10:00
										 |  |  |   if (use_userdef) { | 
					
						
							| 
									
										
										
										
											2020-03-29 16:33:51 +11:00
										 |  |  |     /* Clear keymaps because the current default keymap may have been initialized
 | 
					
						
							|  |  |  |      * from user preferences, which have been reset. */ | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |     LISTBASE_FOREACH (wmWindowManager *, wm, &bmain->wm) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (wm->defaultconf) { | 
					
						
							|  |  |  |         wm->defaultconf->flag &= ~KEYCONF_INIT_DEFAULT; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   if (use_data) { | 
					
						
							|  |  |  |     WM_check(C); /* opens window(s), checks keymaps */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |     bmain->name[0] = '\0'; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |     /* start with save preference untitled.blend */ | 
					
						
							|  |  |  |     G.save_over = 0; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-22 14:05:22 +10:00
										 |  |  |   wm_file_read_post(C, true, is_factory_startup, use_data, use_userdef, reset_app_template); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (r_is_factory_startup) { | 
					
						
							|  |  |  |     *r_is_factory_startup = is_factory_startup; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \name Blend-File History API
 | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_history_file_read(void) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char name[FILE_MAX]; | 
					
						
							|  |  |  |   LinkNode *l, *lines; | 
					
						
							|  |  |  |   struct RecentFile *recent; | 
					
						
							|  |  |  |   const char *line; | 
					
						
							|  |  |  |   int num; | 
					
						
							|  |  |  |   const char *const cfgdir = BKE_appdir_folder_id(BLENDER_USER_CONFIG, NULL); | 
					
						
							| 
									
										
										
										
											2010-07-04 15:35:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!cfgdir) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-07-04 15:35:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 13:23:26 +11:00
										 |  |  |   BLI_join_dirfile(name, sizeof(name), cfgdir, BLENDER_HISTORY_FILE); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   lines = BLI_file_read_as_lines(name); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:29:52 +01:00
										 |  |  |   wm_history_files_free(); | 
					
						
							| 
									
										
										
										
											2006-05-30 13:38:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* read list of recent opened files from recent-files.txt to memory */ | 
					
						
							|  |  |  |   for (l = lines, num = 0; l && (num < U.recent_files); l = l->next) { | 
					
						
							|  |  |  |     line = l->link; | 
					
						
							|  |  |  |     /* don't check if files exist, causes slow startup for remote/external drives */ | 
					
						
							|  |  |  |     if (line[0]) { | 
					
						
							|  |  |  |       recent = (RecentFile *)MEM_mallocN(sizeof(RecentFile), "RecentFile"); | 
					
						
							|  |  |  |       BLI_addtail(&(G.recent_files), recent); | 
					
						
							|  |  |  |       recent->filepath = BLI_strdup(line); | 
					
						
							|  |  |  |       num++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_file_free_lines(lines); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | static RecentFile *wm_history_file_new(const char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RecentFile *recent = MEM_mallocN(sizeof(RecentFile), "RecentFile"); | 
					
						
							|  |  |  |   recent->filepath = BLI_strdup(filepath); | 
					
						
							|  |  |  |   return recent; | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_history_file_free(RecentFile *recent) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(BLI_findindex(&G.recent_files, recent) != -1); | 
					
						
							|  |  |  |   MEM_freeN(recent->filepath); | 
					
						
							|  |  |  |   BLI_freelinkN(&G.recent_files, recent); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:29:52 +01:00
										 |  |  | static void wm_history_files_free(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   LISTBASE_FOREACH_MUTABLE (RecentFile *, recent, &G.recent_files) { | 
					
						
							|  |  |  |     wm_history_file_free(recent); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | static RecentFile *wm_file_history_find(const char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return BLI_findstring_ptr(&G.recent_files, filepath, offsetof(RecentFile, filepath)); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Write #BLENDER_HISTORY_FILE as-is, without checking the environment | 
					
						
							| 
									
										
										
										
											2018-09-24 17:27:41 +02:00
										 |  |  |  * (that's handled by #wm_history_file_update). | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  |  */ | 
					
						
							|  |  |  | static void wm_history_file_write(void) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const char *user_config_dir; | 
					
						
							|  |  |  |   char name[FILE_MAX]; | 
					
						
							|  |  |  |   FILE *fp; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* will be NULL in background mode */ | 
					
						
							|  |  |  |   user_config_dir = BKE_appdir_folder_id_create(BLENDER_USER_CONFIG, NULL); | 
					
						
							|  |  |  |   if (!user_config_dir) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-20 23:13:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 13:23:26 +11:00
										 |  |  |   BLI_join_dirfile(name, sizeof(name), user_config_dir, BLENDER_HISTORY_FILE); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   fp = BLI_fopen(name, "w"); | 
					
						
							|  |  |  |   if (fp) { | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |     LISTBASE_FOREACH (RecentFile *, recent, &G.recent_files) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       fprintf(fp, "%s\n", recent->filepath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     fclose(fp); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Run after saving a file to refresh the #BLENDER_HISTORY_FILE list. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void wm_history_file_update(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RecentFile *recent; | 
					
						
							|  |  |  |   const char *blendfile_name = BKE_main_blendfile_path_from_global(); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* no write history for recovered startup files */ | 
					
						
							|  |  |  |   if (blendfile_name[0] == '\0') { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   recent = G.recent_files.first; | 
					
						
							|  |  |  |   /* refresh recent-files.txt of recent opened files, when current file was changed */ | 
					
						
							|  |  |  |   if (!(recent) || (BLI_path_cmp(recent->filepath, blendfile_name) != 0)) { | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     recent = wm_file_history_find(blendfile_name); | 
					
						
							|  |  |  |     if (recent) { | 
					
						
							|  |  |  |       BLI_remlink(&G.recent_files, recent); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       RecentFile *recent_next; | 
					
						
							|  |  |  |       for (recent = BLI_findlink(&G.recent_files, U.recent_files - 1); recent; | 
					
						
							|  |  |  |            recent = recent_next) { | 
					
						
							|  |  |  |         recent_next = recent->next; | 
					
						
							|  |  |  |         wm_history_file_free(recent); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       recent = wm_history_file_new(blendfile_name); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-05 14:56:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* add current file to the beginning of list */ | 
					
						
							|  |  |  |     BLI_addhead(&(G.recent_files), recent); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* write current file to recent-files.txt */ | 
					
						
							|  |  |  |     wm_history_file_write(); | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* also update most recent files on System */ | 
					
						
							|  |  |  |     GHOST_addToSystemRecentFiles(blendfile_name); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 12:53:09 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \name Save Main Blend-File (internal)
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-17 20:51:11 +00:00
										 |  |  | /* screen can be NULL */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static ImBuf *blend_file_thumb(const bContext *C, | 
					
						
							|  |  |  |                                Scene *scene, | 
					
						
							|  |  |  |                                bScreen *screen, | 
					
						
							|  |  |  |                                BlendThumbnail **thumb_pt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* will be scaled down, but gives some nice oversampling */ | 
					
						
							|  |  |  |   ImBuf *ibuf; | 
					
						
							|  |  |  |   BlendThumbnail *thumb; | 
					
						
							| 
									
										
											  
											
												UI: File Browser Design Overhaul
This is a general redesign of the File Browser GUI and interaction
methods. For screenshots, check patch D5601.
Main changes in short:
* File Browser as floating window
* New layout of regions
* Popovers for view and filter options
* Vertical list view with interactive column header
* New and updated icons
* Keymap consistency fixes
* Many tweaks and fixes to the drawing of views
----
General:
* The file browser now opens as temporary floating window. It closes on
  Esc. The header is hidden then.
* When the file browser is opened as regular editor, the header remains
  visible.
* All file browser regions are now defined in Python (the button
  layout).
* Adjusted related operator UI names.
Keymap:
Keymap is now consistent with other list-based views in Blender, such as
the Outliner.
* Left click to select, double-click to open
* Right-click context menus
* Shift-click to fill selection
* Ctrl-click to extend selection
Operator options:
These previously overlapped with the source list, which caused numerous
issues with resizing and presenting many settings in a small panel area.
It was also generally inconsistent with Blender.
* Moved to new sidebar, which can easily be shown or hidden using a
  prominent Options toggle.
* IO operators have new layouts to match this new sidebar, using
  sub-panels. This will have to be committed separately (Add-on
  repository).
* If operators want to show the options by default, they have the option
  to do so (see `WM_FILESEL_SHOW_PROPS`, `hide_props_region`), otherwise
  they are hidden by default.
General Layout:
The layout has been changed to be simpler, more standard, and fits
better in with Blender 2.8.
* More conventional layout (file path at top, file name at the bottom,
  execute/cancel buttons in bottom right).
* Use of popovers to group controls, and allow for more descriptive
  naming.
* Search box is always live now, just like Outliner.
Views:
* Date Modified column combines both date and time, also uses user
  friendly strings for recent dates (i.e. "Yesterday", "Today").
* Details columns (file size, modification date/time) are now toggleable
  for all display types, they are not hardcoded per display type.
* File sizes now show as B, KB, MB, ... rather than B, KiB, MiB, … They
  are now also calculated using base 10 of course.
* Option to sort in inverse order.
Vertical List View:
* This view now used a much simpler single vertical list with columns
  for information.
* Users can click on the headers of these columns to order by that
  category, and click again to reverse the ordering.
Icons:
* Updated icons by Jendrzych, with better centering.
* Files and folders have new icons in Icon view.
* Both files and folders have reworked superimposed icons that show
  users the file/folder type.
* 3D file documents correctly use the 3d file icon, which was unused
  previously.
* Workspaces now show their icon on Link/Append - also when listed in
  the Outliner.
Minor Python-API breakage:
* `bpy.types.FileSelectParams.display_type`: `LIST_SHORT` and
  `LIST_LONG` are replaced by `LIST_VERTICAL` and `LIST_HORIZONTAL`.
Removes the feature where directories would automatically be created if
they are entered into the file path text button, but don't exist. We
were not sure if users use it enough to keep it. We can definitely bring
it back.
----
//Combined effort by @billreynish, @harley, @jendrzych, my university
colleague Brian Meisenheimer and myself.//
Differential Revision: https://developer.blender.org/D5601
Reviewers: Brecht, Bastien
											
										 
											2019-09-03 15:43:38 +02:00
										 |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							| 
									
										
										
										
											2020-05-20 14:58:57 +10:00
										 |  |  |   const float pixelsize_old = U.pixelsize; | 
					
						
							| 
									
										
											  
											
												UI: File Browser Design Overhaul
This is a general redesign of the File Browser GUI and interaction
methods. For screenshots, check patch D5601.
Main changes in short:
* File Browser as floating window
* New layout of regions
* Popovers for view and filter options
* Vertical list view with interactive column header
* New and updated icons
* Keymap consistency fixes
* Many tweaks and fixes to the drawing of views
----
General:
* The file browser now opens as temporary floating window. It closes on
  Esc. The header is hidden then.
* When the file browser is opened as regular editor, the header remains
  visible.
* All file browser regions are now defined in Python (the button
  layout).
* Adjusted related operator UI names.
Keymap:
Keymap is now consistent with other list-based views in Blender, such as
the Outliner.
* Left click to select, double-click to open
* Right-click context menus
* Shift-click to fill selection
* Ctrl-click to extend selection
Operator options:
These previously overlapped with the source list, which caused numerous
issues with resizing and presenting many settings in a small panel area.
It was also generally inconsistent with Blender.
* Moved to new sidebar, which can easily be shown or hidden using a
  prominent Options toggle.
* IO operators have new layouts to match this new sidebar, using
  sub-panels. This will have to be committed separately (Add-on
  repository).
* If operators want to show the options by default, they have the option
  to do so (see `WM_FILESEL_SHOW_PROPS`, `hide_props_region`), otherwise
  they are hidden by default.
General Layout:
The layout has been changed to be simpler, more standard, and fits
better in with Blender 2.8.
* More conventional layout (file path at top, file name at the bottom,
  execute/cancel buttons in bottom right).
* Use of popovers to group controls, and allow for more descriptive
  naming.
* Search box is always live now, just like Outliner.
Views:
* Date Modified column combines both date and time, also uses user
  friendly strings for recent dates (i.e. "Yesterday", "Today").
* Details columns (file size, modification date/time) are now toggleable
  for all display types, they are not hardcoded per display type.
* File sizes now show as B, KB, MB, ... rather than B, KiB, MiB, … They
  are now also calculated using base 10 of course.
* Option to sort in inverse order.
Vertical List View:
* This view now used a much simpler single vertical list with columns
  for information.
* Users can click on the headers of these columns to order by that
  category, and click again to reverse the ordering.
Icons:
* Updated icons by Jendrzych, with better centering.
* Files and folders have new icons in Icon view.
* Both files and folders have reworked superimposed icons that show
  users the file/folder type.
* 3D file documents correctly use the 3d file icon, which was unused
  previously.
* Workspaces now show their icon on Link/Append - also when listed in
  the Outliner.
Minor Python-API breakage:
* `bpy.types.FileSelectParams.display_type`: `LIST_SHORT` and
  `LIST_LONG` are replaced by `LIST_VERTICAL` and `LIST_HORIZONTAL`.
Removes the feature where directories would automatically be created if
they are entered into the file path text button, but don't exist. We
were not sure if users use it enough to keep it. We can definitely bring
it back.
----
//Combined effort by @billreynish, @harley, @jendrzych, my university
colleague Brian Meisenheimer and myself.//
Differential Revision: https://developer.blender.org/D5601
Reviewers: Brecht, Bastien
											
										 
											2019-09-03 15:43:38 +02:00
										 |  |  |   wmWindow *windrawable_old = wm->windrawable; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char err_out[256] = "unknown"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* screen if no camera found */ | 
					
						
							| 
									
										
										
										
											2020-04-03 13:25:03 +02:00
										 |  |  |   ScrArea *area = NULL; | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |   ARegion *region = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   View3D *v3d = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* In case we are given a valid thumbnail data, just generate image from it. */ | 
					
						
							|  |  |  |   if (*thumb_pt) { | 
					
						
							|  |  |  |     thumb = *thumb_pt; | 
					
						
							|  |  |  |     return BKE_main_thumbnail_to_imbuf(NULL, thumb); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* scene can be NULL if running a script at startup and calling the save operator */ | 
					
						
							|  |  |  |   if (G.background || scene == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ((scene->camera == NULL) && (screen != NULL)) { | 
					
						
							| 
									
										
										
										
											2020-04-03 13:25:03 +02:00
										 |  |  |     area = BKE_screen_find_big_area(screen, SPACE_VIEW3D, 0); | 
					
						
							|  |  |  |     region = BKE_area_find_region_type(area, RGN_TYPE_WINDOW); | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |     if (region) { | 
					
						
							| 
									
										
										
										
											2020-04-03 13:25:03 +02:00
										 |  |  |       v3d = area->spacedata.first; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (scene->camera == NULL && v3d == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* gets scaled to BLEN_THUMB_SIZE */ | 
					
						
							| 
									
										
										
										
											2019-07-25 16:36:22 +02:00
										 |  |  |   Depsgraph *depsgraph = CTX_data_ensure_evaluated_depsgraph(C); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-20 14:58:57 +10:00
										 |  |  |   /* Note that with scaling, this ends up being 0.5,
 | 
					
						
							|  |  |  |    * as it's a thumbnail, we don't need object centers and friends to be 1:1 size. */ | 
					
						
							|  |  |  |   U.pixelsize = 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (scene->camera) { | 
					
						
							|  |  |  |     ibuf = ED_view3d_draw_offscreen_imbuf_simple(depsgraph, | 
					
						
							|  |  |  |                                                  scene, | 
					
						
							| 
									
										
										
										
											2019-04-25 16:24:06 +02:00
										 |  |  |                                                  NULL, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                                  OB_SOLID, | 
					
						
							|  |  |  |                                                  scene->camera, | 
					
						
							|  |  |  |                                                  BLEN_THUMB_SIZE * 2, | 
					
						
							|  |  |  |                                                  BLEN_THUMB_SIZE * 2, | 
					
						
							|  |  |  |                                                  IB_rect, | 
					
						
							|  |  |  |                                                  V3D_OFSDRAW_NONE, | 
					
						
							|  |  |  |                                                  R_ALPHAPREMUL, | 
					
						
							|  |  |  |                                                  NULL, | 
					
						
							|  |  |  |                                                  NULL, | 
					
						
							|  |  |  |                                                  err_out); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     ibuf = ED_view3d_draw_offscreen_imbuf(depsgraph, | 
					
						
							|  |  |  |                                           scene, | 
					
						
							|  |  |  |                                           OB_SOLID, | 
					
						
							|  |  |  |                                           v3d, | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |                                           region, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                           BLEN_THUMB_SIZE * 2, | 
					
						
							|  |  |  |                                           BLEN_THUMB_SIZE * 2, | 
					
						
							|  |  |  |                                           IB_rect, | 
					
						
							|  |  |  |                                           R_ALPHAPREMUL, | 
					
						
							|  |  |  |                                           NULL, | 
					
						
							| 
									
										
										
										
											2021-02-10 12:46:43 +01:00
										 |  |  |                                           true, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                           NULL, | 
					
						
							|  |  |  |                                           err_out); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-20 14:58:57 +10:00
										 |  |  |   U.pixelsize = pixelsize_old; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												UI: File Browser Design Overhaul
This is a general redesign of the File Browser GUI and interaction
methods. For screenshots, check patch D5601.
Main changes in short:
* File Browser as floating window
* New layout of regions
* Popovers for view and filter options
* Vertical list view with interactive column header
* New and updated icons
* Keymap consistency fixes
* Many tweaks and fixes to the drawing of views
----
General:
* The file browser now opens as temporary floating window. It closes on
  Esc. The header is hidden then.
* When the file browser is opened as regular editor, the header remains
  visible.
* All file browser regions are now defined in Python (the button
  layout).
* Adjusted related operator UI names.
Keymap:
Keymap is now consistent with other list-based views in Blender, such as
the Outliner.
* Left click to select, double-click to open
* Right-click context menus
* Shift-click to fill selection
* Ctrl-click to extend selection
Operator options:
These previously overlapped with the source list, which caused numerous
issues with resizing and presenting many settings in a small panel area.
It was also generally inconsistent with Blender.
* Moved to new sidebar, which can easily be shown or hidden using a
  prominent Options toggle.
* IO operators have new layouts to match this new sidebar, using
  sub-panels. This will have to be committed separately (Add-on
  repository).
* If operators want to show the options by default, they have the option
  to do so (see `WM_FILESEL_SHOW_PROPS`, `hide_props_region`), otherwise
  they are hidden by default.
General Layout:
The layout has been changed to be simpler, more standard, and fits
better in with Blender 2.8.
* More conventional layout (file path at top, file name at the bottom,
  execute/cancel buttons in bottom right).
* Use of popovers to group controls, and allow for more descriptive
  naming.
* Search box is always live now, just like Outliner.
Views:
* Date Modified column combines both date and time, also uses user
  friendly strings for recent dates (i.e. "Yesterday", "Today").
* Details columns (file size, modification date/time) are now toggleable
  for all display types, they are not hardcoded per display type.
* File sizes now show as B, KB, MB, ... rather than B, KiB, MiB, … They
  are now also calculated using base 10 of course.
* Option to sort in inverse order.
Vertical List View:
* This view now used a much simpler single vertical list with columns
  for information.
* Users can click on the headers of these columns to order by that
  category, and click again to reverse the ordering.
Icons:
* Updated icons by Jendrzych, with better centering.
* Files and folders have new icons in Icon view.
* Both files and folders have reworked superimposed icons that show
  users the file/folder type.
* 3D file documents correctly use the 3d file icon, which was unused
  previously.
* Workspaces now show their icon on Link/Append - also when listed in
  the Outliner.
Minor Python-API breakage:
* `bpy.types.FileSelectParams.display_type`: `LIST_SHORT` and
  `LIST_LONG` are replaced by `LIST_VERTICAL` and `LIST_HORIZONTAL`.
Removes the feature where directories would automatically be created if
they are entered into the file path text button, but don't exist. We
were not sure if users use it enough to keep it. We can definitely bring
it back.
----
//Combined effort by @billreynish, @harley, @jendrzych, my university
colleague Brian Meisenheimer and myself.//
Differential Revision: https://developer.blender.org/D5601
Reviewers: Brecht, Bastien
											
										 
											2019-09-03 15:43:38 +02:00
										 |  |  |   /* Reset to old drawable. */ | 
					
						
							|  |  |  |   if (windrawable_old) { | 
					
						
							|  |  |  |     wm_window_make_drawable(wm, windrawable_old); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     wm_window_clear_drawable(wm); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ibuf) { | 
					
						
							|  |  |  |     /* dirty oversampling */ | 
					
						
							|  |  |  |     IMB_scaleImBuf(ibuf, BLEN_THUMB_SIZE, BLEN_THUMB_SIZE); | 
					
						
							|  |  |  |     thumb = BKE_main_thumbnail_from_imbuf(NULL, ibuf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* '*thumb_pt' needs to stay NULL to prevent a bad thumbnail from being handled */ | 
					
						
							|  |  |  |     fprintf(stderr, "blend_file_thumb failed to create thumbnail: %s\n", err_out); | 
					
						
							|  |  |  |     thumb = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* must be freed by caller */ | 
					
						
							|  |  |  |   *thumb_pt = thumb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ibuf; | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-22 08:38:12 +00:00
										 |  |  | /* easy access from gdb */ | 
					
						
							| 
									
										
										
										
											2013-12-05 17:26:03 +11:00
										 |  |  | bool write_crash_blend(void) | 
					
						
							| 
									
										
										
										
											2010-06-22 08:38:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char path[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2020-06-18 12:21:38 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_strncpy(path, BKE_main_blendfile_path_from_global(), sizeof(path)); | 
					
						
							|  |  |  |   BLI_path_extension_replace(path, sizeof(path), "_crash.blend"); | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   if (BLO_write_file(G_MAIN, path, G.fileflags, &(const struct BlendFileWriteParams){0}, NULL)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     printf("written: %s\n", path); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   printf("failed: %s\n", path); | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2010-06-22 08:38:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-20 12:12:45 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \see #wm_homefile_write_exec wraps #BLO_write_file in a similar way. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-18 15:25:22 +10:00
										 |  |  | static bool wm_file_write(bContext *C, | 
					
						
							|  |  |  |                           const char *filepath, | 
					
						
							|  |  |  |                           int fileflags, | 
					
						
							|  |  |  |                           eBLO_WritePathRemap remap_mode, | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |                           bool use_save_as_copy, | 
					
						
							| 
									
										
										
										
											2020-06-18 15:25:22 +10:00
										 |  |  |                           ReportList *reports) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   int len; | 
					
						
							|  |  |  |   int ok = false; | 
					
						
							|  |  |  |   BlendThumbnail *thumb, *main_thumb; | 
					
						
							|  |  |  |   ImBuf *ibuf_thumb = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   len = strlen(filepath); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (len == 0) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_ERROR, "Path is empty, cannot save"); | 
					
						
							|  |  |  |     return ok; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-16 05:24:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (len >= FILE_MAX) { | 
					
						
							|  |  |  |     BKE_report(reports, RPT_ERROR, "Path too long, cannot save"); | 
					
						
							|  |  |  |     return ok; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Check if file write permission is ok */ | 
					
						
							|  |  |  |   if (BLI_exists(filepath) && !BLI_file_is_writable(filepath)) { | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_ERROR, "Cannot save blend file, path '%s' is not writable", filepath); | 
					
						
							|  |  |  |     return ok; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-12 13:35:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* note: used to replace the file extension (to ensure '.blend'),
 | 
					
						
							|  |  |  |    * no need to now because the operator ensures, | 
					
						
							|  |  |  |    * its handy for scripts to save to a predefined name without blender editing it */ | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* send the OnSave event */ | 
					
						
							| 
									
										
										
										
											2020-12-04 13:50:53 -06:00
										 |  |  |   LISTBASE_FOREACH (Library *, li, &bmain->libraries) { | 
					
						
							| 
									
										
										
										
											2020-06-23 09:54:07 +10:00
										 |  |  |     if (BLI_path_cmp(li->filepath_abs, filepath) == 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       BKE_reportf(reports, RPT_ERROR, "Cannot overwrite used library '%.240s'", filepath); | 
					
						
							|  |  |  |       return ok; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 00:54:22 +10:00
										 |  |  |   /* Call pre-save callbacks before writing preview,
 | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |    * that way you can generate custom file thumbnail. */ | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |   BKE_callback_exec_null(bmain, BKE_CB_EVT_SAVE_PRE); | 
					
						
							| 
									
										
											  
											
												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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 14:32:17 +02:00
										 |  |  |   /* Enforce full override check/generation on file save. */ | 
					
						
							| 
									
										
										
										
											2020-02-10 18:05:19 +01:00
										 |  |  |   BKE_lib_override_library_main_operations_create(bmain, true); | 
					
						
							| 
									
										
										
										
											2019-08-22 14:32:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* blend file thumbnail */ | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* Save before exit_editmode, otherwise derivedmeshes for shared data corrupt T27765. */ | 
					
						
							|  |  |  |   /* Main now can store a '.blend' thumbnail, useful for background mode
 | 
					
						
							|  |  |  |    * or thumbnail customization. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   main_thumb = thumb = bmain->blen_thumb; | 
					
						
							|  |  |  |   if ((U.flag & USER_SAVE_PREVIEWS) && BLI_thread_is_main()) { | 
					
						
							|  |  |  |     ibuf_thumb = blend_file_thumb(C, CTX_data_scene(C), CTX_wm_screen(C), &thumb); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-07-13 19:16:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* operator now handles overwrite checks */ | 
					
						
							| 
									
										
										
										
											2010-09-17 08:53:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G.fileflags & G_FILE_AUTOPACK) { | 
					
						
							| 
									
										
										
										
											2019-07-07 23:57:35 +10:00
										 |  |  |     BKE_packedfile_pack_all(bmain, reports, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-03-21 14:12:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* don't forget not to return without! */ | 
					
						
							| 
									
										
										
										
											2021-03-05 10:36:57 +01:00
										 |  |  |   WM_cursor_wait(true); | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-30 16:46:09 +11:00
										 |  |  |   ED_editors_flush_edits(bmain); | 
					
						
							| 
									
										
										
										
											2014-01-20 12:12:45 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* first time saving */ | 
					
						
							|  |  |  |   /* XXX temp solution to solve bug, real fix coming (ton) */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   if ((BKE_main_blendfile_path(bmain)[0] == '\0') && (use_save_as_copy == false)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BLI_strncpy(bmain->name, filepath, sizeof(bmain->name)); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-10-09 20:27:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* XXX temp solution to solve bug, real fix coming (ton) */ | 
					
						
							|  |  |  |   bmain->recovered = 0; | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   if (BLO_write_file(CTX_data_main(C), | 
					
						
							|  |  |  |                      filepath, | 
					
						
							|  |  |  |                      fileflags, | 
					
						
							|  |  |  |                      &(const struct BlendFileWriteParams){ | 
					
						
							|  |  |  |                          .remap_mode = remap_mode, | 
					
						
							|  |  |  |                          .use_save_versions = true, | 
					
						
							|  |  |  |                          .use_save_as_copy = use_save_as_copy, | 
					
						
							|  |  |  |                          .thumb = thumb, | 
					
						
							|  |  |  |                      }, | 
					
						
							|  |  |  |                      reports)) { | 
					
						
							|  |  |  |     const bool do_history_file_update = (G.background == false) && | 
					
						
							|  |  |  |                                         (CTX_wm_manager(C)->op_undo_depth == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (use_save_as_copy == false) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       G.relbase_valid = 1; | 
					
						
							|  |  |  |       BLI_strncpy(bmain->name, filepath, sizeof(bmain->name)); /* is guaranteed current file */ | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       G.save_over = 1; /* disable untitled.blend convention */ | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-23 21:35:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     SET_FLAG_FROM_TEST(G.fileflags, fileflags & G_FILE_COMPRESS, G_FILE_COMPRESS); | 
					
						
							| 
									
										
										
										
											2010-03-05 10:37:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* prevent background mode scripts from clobbering history */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |     if (do_history_file_update) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       wm_history_file_update(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-24 21:52:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |     BKE_callback_exec_null(bmain, BKE_CB_EVT_SAVE_POST); | 
					
						
							| 
									
										
										
										
											2011-06-24 16:54:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* run this function after because the file cant be written before the blend is */ | 
					
						
							|  |  |  |     if (ibuf_thumb) { | 
					
						
							|  |  |  |       IMB_thumb_delete(filepath, THB_FAIL); /* without this a failed thumb overrides */ | 
					
						
							|  |  |  |       ibuf_thumb = IMB_thumb_create(filepath, THB_LARGE, THB_SOURCE_BLEND, ibuf_thumb); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-30 19:21:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 13:19:09 +10:00
										 |  |  |     /* Without this there is no feedback the file was saved. */ | 
					
						
							|  |  |  |     BKE_reportf(reports, RPT_INFO, "Saved \"%s\"", BLI_path_basename(filepath)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* Success. */ | 
					
						
							|  |  |  |     ok = true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
											  
											
												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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ibuf_thumb) { | 
					
						
							|  |  |  |     IMB_freeImBuf(ibuf_thumb); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (thumb && thumb != main_thumb) { | 
					
						
							|  |  |  |     MEM_freeN(thumb); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 10:36:57 +01:00
										 |  |  |   WM_cursor_wait(false); | 
					
						
							| 
									
										
											  
											
												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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ok; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Auto-Save API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-18 06:27:32 +00:00
										 |  |  | void wm_autosave_location(char *filepath) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int pid = abs(getpid()); | 
					
						
							|  |  |  |   char path[1024]; | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | #ifdef WIN32
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const char *savedir; | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G_MAIN && G.relbase_valid) { | 
					
						
							|  |  |  |     const char *basename = BLI_path_basename(BKE_main_blendfile_path_from_global()); | 
					
						
							|  |  |  |     int len = strlen(basename) - 6; | 
					
						
							| 
									
										
										
										
											2019-04-17 16:54:28 +02:00
										 |  |  |     BLI_snprintf(path, sizeof(path), "%.*s_%d_autosave.blend", len, basename, pid); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2019-04-17 16:54:28 +02:00
										 |  |  |     BLI_snprintf(path, sizeof(path), "%d_autosave.blend", pid); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-02-12 16:54:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | #ifdef WIN32
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* XXX Need to investigate how to handle default location of '/tmp/'
 | 
					
						
							|  |  |  |    * This is a relative directory on Windows, and it may be | 
					
						
							|  |  |  |    * found. Example: | 
					
						
							|  |  |  |    * Blender installed on D:\ drive, D:\ drive has D:\tmp\ | 
					
						
							|  |  |  |    * Now, BLI_exists() will find '/tmp/' exists, but | 
					
						
							|  |  |  |    * BLI_make_file_string will create string that has it most likely on C:\ | 
					
						
							| 
									
										
										
										
											2020-08-26 15:50:48 +02:00
										 |  |  |    * through BLI_windows_get_default_root_dir(). | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |    * If there is no C:\tmp autosave fails. */ | 
					
						
							|  |  |  |   if (!BLI_exists(BKE_tempdir_base())) { | 
					
						
							|  |  |  |     savedir = BKE_appdir_folder_id_create(BLENDER_USER_AUTOSAVE, NULL); | 
					
						
							|  |  |  |     BLI_make_file_string("/", filepath, savedir, path); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-09-28 19:11:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 13:23:26 +11:00
										 |  |  |   BLI_join_dirfile(filepath, FILE_MAX, BKE_tempdir_base(), path); | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 17:23:48 +00:00
										 |  |  | void WM_autosave_init(wmWindowManager *wm) | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_autosave_timer_ended(wm); | 
					
						
							| 
									
										
										
										
											2008-12-19 00:50:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (U.flag & USER_AUTOSAVE) { | 
					
						
							|  |  |  |     wm->autosavetimer = WM_event_add_timer(wm, NULL, TIMERAUTOSAVE, U.savetime * 60.0); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  | void wm_autosave_timer(Main *bmain, wmWindowManager *wm, wmTimer *UNUSED(wt)) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filepath[FILE_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WM_event_remove_timer(wm, NULL, wm->autosavetimer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-16 23:37:13 +02:00
										 |  |  |   /* If a modal operator is running, don't autosave because we might not be in
 | 
					
						
							|  |  |  |    * a valid state to save. But try again in 10ms. */ | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (wmWindow *, win, &wm->windows) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     LISTBASE_FOREACH (wmEventHandler *, handler_base, &win->modalhandlers) { | 
					
						
							|  |  |  |       if (handler_base->type == WM_HANDLER_TYPE_OP) { | 
					
						
							|  |  |  |         wmEventHandler_Op *handler = (wmEventHandler_Op *)handler_base; | 
					
						
							|  |  |  |         if (handler->op) { | 
					
						
							| 
									
										
										
										
											2020-06-16 23:37:13 +02:00
										 |  |  |           wm->autosavetimer = WM_event_add_timer(wm, NULL, TIMERAUTOSAVE, 0.01); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wm_autosave_location(filepath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (U.uiflag & USER_GLOBALUNDO) { | 
					
						
							|  |  |  |     /* fast save of last undobuffer, now with UI */ | 
					
						
							|  |  |  |     struct MemFile *memfile = ED_undosys_stack_memfile_get_active(wm->undo_stack); | 
					
						
							|  |  |  |     if (memfile) { | 
					
						
							|  |  |  |       BLO_memfile_write_file(memfile, filepath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2020-03-29 16:33:51 +11:00
										 |  |  |     /* Save as regular blend file. */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |     const int fileflags = G.fileflags & ~G_FILE_COMPRESS; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-30 16:46:09 +11:00
										 |  |  |     ED_editors_flush_edits(bmain); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |     /* Error reporting into console. */ | 
					
						
							|  |  |  |     BLO_write_file(bmain, filepath, fileflags, &(const struct BlendFileWriteParams){0}, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* do timer after file write, just in case file write takes a long time */ | 
					
						
							|  |  |  |   wm->autosavetimer = WM_event_add_timer(wm, NULL, TIMERAUTOSAVE, U.savetime * 60.0); | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_autosave_timer_ended(wmWindowManager *wm) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (wm->autosavetimer) { | 
					
						
							|  |  |  |     WM_event_remove_timer(wm, NULL, wm->autosavetimer); | 
					
						
							|  |  |  |     wm->autosavetimer = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_autosave_delete(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filename[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2018-06-07 16:43:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_autosave_location(filename); | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (BLI_exists(filename)) { | 
					
						
							|  |  |  |     char str[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2020-03-07 13:23:26 +11:00
										 |  |  |     BLI_join_dirfile(str, sizeof(str), BKE_tempdir_base(), BLENDER_QUIT_FILE); | 
					
						
							| 
									
										
											  
											
												Another step in the undo evolution.
- Made unified API for undo calls, to be found in space.c
  BIF_undo_push(char *str)
  BIF_undo(void)
  BIF_redo(void)
  These calls will do all undo levels, including editmode and vpaint.
  The transition is work in progress, because mesh undo needs recode.
- New global hotkey CTR+Z for undo
  Note: 'shaded draw mode' still is SHIFT+Z, the old CTRL+Z was to recalc
  the lighting in shaded mode, which already became much more interactive,
  like during/after any transform().
  Recalc hotkey now is SHIFT+ALT+Z
  CTRL+<any modifier>+Z is redo.
- For OSX users; the Apple-key ("Command") now maps to CTRL as well. This
  disables the one-mouse-button hack for rightmouse btw, will be fixed in
  next commit. At least we can use Apple-Z :)
- Old Ukey for undo is still there, as a training period... my preference is
  to restore Ukey to "reload original data" as in past, and only use new
  CTRL+Z for undo.
- Added undo_push() for all of editobject.c and editview.c. Meaning we can
  start using/testing global undo in the 3d window. Please dont comment on
  missing parts for now, first I want someone to volunteer to tackle all of
  that.
- Since the global undo has a full 'file' in memory, it can save extremely
  fast on exit to <temp dir>/quit.blend. That's default now when global undo
  is enabled. It prints "Saved session recovery to ..." in console then.
- In file menu, a new option is added "Recover Last Session". Note that this
  reads the undo-save, which is without UI.
- With such nice new features we then can also kill the disputed
  Cancel/Confirm menu on Q-KEY.
- Added fix which initializes seam/normal theme color on saved themes.
  They showed black now.... (Note: that's in usiblender.c!)
											
										 
											2004-09-18 12:12:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* if global undo; remove tempsave, otherwise rename */ | 
					
						
							|  |  |  |     if (U.uiflag & USER_GLOBALUNDO) { | 
					
						
							|  |  |  |       BLI_delete(filename, false, false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       BLI_rename(filename, str); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | void wm_autosave_read(bContext *C, ReportList *reports) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filename[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_autosave_location(filename); | 
					
						
							|  |  |  |   WM_file_read(C, filename, reports); | 
					
						
							| 
									
										
										
										
											2009-10-20 13:58:53 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2014-04-03 11:04:02 +11:00
										 |  |  | /** \name Initialize WM_OT_open_xxx properties
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Check if load_ui was set by the caller. | 
					
						
							|  |  |  |  * Fall back to user preference when file flags not specified. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_open_init_load_ui(wmOperator *op, bool use_prefs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PropertyRNA *prop = RNA_struct_find_property(op->ptr, "load_ui"); | 
					
						
							|  |  |  |   if (!RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |     bool value = use_prefs ? ((U.flag & USER_FILENOUI) == 0) : ((G.fileflags & G_FILE_NO_UI) == 0); | 
					
						
							| 
									
										
										
										
											2014-04-03 11:04:02 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     RNA_property_boolean_set(op->ptr, prop, value); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-04-03 11:04:02 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_open_init_use_scripts(wmOperator *op, bool use_prefs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PropertyRNA *prop = RNA_struct_find_property(op->ptr, "use_scripts"); | 
					
						
							|  |  |  |   if (!RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |     /* use G_FLAG_SCRIPT_AUTOEXEC rather than the userpref because this means if
 | 
					
						
							|  |  |  |      * the flag has been disabled from the command line, then opening | 
					
						
							|  |  |  |      * from the menu wont enable this setting. */ | 
					
						
							|  |  |  |     bool value = use_prefs ? ((U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0) : | 
					
						
							|  |  |  |                              ((G.f & G_FLAG_SCRIPT_AUTOEXEC) != 0); | 
					
						
							| 
									
										
										
										
											2014-04-03 11:04:02 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     RNA_property_boolean_set(op->ptr, prop, value); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-04-03 11:04:02 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2015-10-28 04:10:00 +05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \name Startup File Save Operator
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * \see #wm_file_write wraps #BLO_write_file in a similar way. | 
					
						
							| 
									
										
										
										
											2018-10-22 15:08:26 +11:00
										 |  |  |  * \return success. | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | static int wm_homefile_write_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							|  |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   char filepath[FILE_MAX]; | 
					
						
							|  |  |  |   int fileflags; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const char *app_template = U.app_template[0] ? U.app_template : NULL; | 
					
						
							|  |  |  |   const char *const cfgdir = BKE_appdir_folder_id_create(BLENDER_USER_CONFIG, app_template); | 
					
						
							|  |  |  |   if (cfgdir == NULL) { | 
					
						
							|  |  |  |     BKE_report(op->reports, RPT_ERROR, "Unable to create user config path"); | 
					
						
							|  |  |  |     return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-25 09:29:51 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |   BKE_callback_exec_null(bmain, BKE_CB_EVT_SAVE_PRE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* check current window and close it if temp */ | 
					
						
							|  |  |  |   if (win && WM_window_is_temp_screen(win)) { | 
					
						
							|  |  |  |     wm_window_close(C, wm, win); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* update keymaps in user preferences */ | 
					
						
							|  |  |  |   WM_keyconfig_update(wm); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_path_join(filepath, sizeof(filepath), cfgdir, BLENDER_STARTUP_FILE, NULL); | 
					
						
							| 
									
										
										
										
											2017-03-25 09:29:51 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("Writing homefile: '%s' ", filepath); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-30 16:46:09 +11:00
										 |  |  |   ED_editors_flush_edits(bmain); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 16:33:51 +11:00
										 |  |  |   /* Force save as regular blend file. */ | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   fileflags = G.fileflags & ~G_FILE_COMPRESS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (BLO_write_file(bmain, | 
					
						
							|  |  |  |                      filepath, | 
					
						
							|  |  |  |                      fileflags, | 
					
						
							|  |  |  |                      &(const struct BlendFileWriteParams){ | 
					
						
							| 
									
										
										
										
											2021-03-04 20:52:03 +11:00
										 |  |  |                          /* Make all paths absolute when saving the startup file.
 | 
					
						
							|  |  |  |                           * On load the `G.relbase_valid` will be false so the paths | 
					
						
							|  |  |  |                           * wont have a base for resolving the relative paths. */ | 
					
						
							|  |  |  |                          .remap_mode = BLO_WRITE_PATH_REMAP_ABSOLUTE, | 
					
						
							| 
									
										
										
										
											2021-03-04 20:56:17 +11:00
										 |  |  |                          /* Don't apply any path changes to the current blend file. */ | 
					
						
							|  |  |  |                          .use_save_as_copy = true, | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |                      }, | 
					
						
							|  |  |  |                      op->reports) == 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     printf("fail\n"); | 
					
						
							|  |  |  |     return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("ok\n"); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   G.save_over = 0; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-09 10:25:04 +02:00
										 |  |  |   BKE_callback_exec_null(bmain, BKE_CB_EVT_SAVE_POST); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_save_homefile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Save Startup File"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_save_homefile"; | 
					
						
							|  |  |  |   ot->description = "Make the current file the default .blend file"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_homefile_write_exec; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Write Preferences Operator
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | /* Only save the prefs block. operator entry */ | 
					
						
							|  |  |  | static int wm_userpref_write_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-10 15:46:31 +10:00
										 |  |  |   /* Update keymaps in user preferences. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_keyconfig_update(wm); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-10 15:46:31 +10:00
										 |  |  |   const bool ok = BKE_blendfile_userdef_write_all(op->reports); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return ok ? OPERATOR_FINISHED : OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_save_userpref(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Save Preferences"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_save_userpref"; | 
					
						
							| 
									
										
										
										
											2019-07-30 11:04:02 +10:00
										 |  |  |   ot->description = "Make the current preferences default"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_userpref_write_exec; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Preferences Operator
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * When reading preferences, there are some exceptions for values which are reset. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-06-12 12:21:21 +10:00
										 |  |  | static void wm_userpref_read_exceptions(UserDef *userdef_curr, const UserDef *userdef_prev) | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  | { | 
					
						
							|  |  |  | #define USERDEF_RESTORE(member) \
 | 
					
						
							|  |  |  |   { \ | 
					
						
							|  |  |  |     userdef_curr->member = userdef_prev->member; \ | 
					
						
							|  |  |  |   } \ | 
					
						
							|  |  |  |   ((void)0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Current visible preferences category. */ | 
					
						
							| 
									
										
										
										
											2019-09-05 03:16:52 +10:00
										 |  |  |   USERDEF_RESTORE(space_data.section_active); | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef USERDEF_RESTORE
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  | static void rna_struct_update_when_changed(bContext *C, | 
					
						
							|  |  |  |                                            Main *bmain, | 
					
						
							|  |  |  |                                            PointerRNA *ptr_a, | 
					
						
							|  |  |  |                                            PointerRNA *ptr_b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CollectionPropertyIterator iter; | 
					
						
							|  |  |  |   PropertyRNA *iterprop = RNA_struct_iterator_property(ptr_a->type); | 
					
						
							|  |  |  |   BLI_assert(ptr_a->type == ptr_b->type); | 
					
						
							|  |  |  |   RNA_property_collection_begin(ptr_a, iterprop, &iter); | 
					
						
							|  |  |  |   for (; iter.valid; RNA_property_collection_next(&iter)) { | 
					
						
							|  |  |  |     PropertyRNA *prop = iter.ptr.data; | 
					
						
							|  |  |  |     if (STREQ(RNA_property_identifier(prop), "rna_type")) { | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (RNA_property_type(prop)) { | 
					
						
							|  |  |  |       case PROP_POINTER: { | 
					
						
							|  |  |  |         PointerRNA ptr_sub_a = RNA_property_pointer_get(ptr_a, prop); | 
					
						
							|  |  |  |         PointerRNA ptr_sub_b = RNA_property_pointer_get(ptr_b, prop); | 
					
						
							|  |  |  |         rna_struct_update_when_changed(C, bmain, &ptr_sub_a, &ptr_sub_b); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case PROP_COLLECTION: | 
					
						
							|  |  |  |         /* Don't handle collections. */ | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: { | 
					
						
							|  |  |  |         if (!RNA_property_equals(bmain, ptr_a, ptr_b, prop, RNA_EQ_STRICT)) { | 
					
						
							|  |  |  |           RNA_property_update(C, ptr_b, prop); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   RNA_property_collection_end(&iter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_userpref_update_when_changed(bContext *C, | 
					
						
							|  |  |  |                                             Main *bmain, | 
					
						
							|  |  |  |                                             UserDef *userdef_prev, | 
					
						
							|  |  |  |                                             UserDef *userdef_curr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   PointerRNA ptr_a, ptr_b; | 
					
						
							|  |  |  |   RNA_pointer_create(NULL, &RNA_Preferences, userdef_prev, &ptr_a); | 
					
						
							|  |  |  |   RNA_pointer_create(NULL, &RNA_Preferences, userdef_curr, &ptr_b); | 
					
						
							| 
									
										
										
										
											2019-05-14 11:12:02 +10:00
										 |  |  |   const bool is_dirty = userdef_curr->runtime.is_dirty; | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   rna_struct_update_when_changed(C, bmain, &ptr_a, &ptr_b); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-13 04:44:25 +10:00
										 |  |  |   WM_reinit_gizmomap_all(bmain); | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   WM_keyconfig_reload(C); | 
					
						
							| 
									
										
										
										
											2019-08-13 04:44:25 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 11:12:02 +10:00
										 |  |  |   userdef_curr->runtime.is_dirty = is_dirty; | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_userpref_read_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const bool use_data = false; | 
					
						
							|  |  |  |   const bool use_userdef = true; | 
					
						
							| 
									
										
										
										
											2019-05-13 13:25:51 +10:00
										 |  |  |   const bool use_factory_settings = STREQ(op->type->idname, "WM_OT_read_factory_userpref"); | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   UserDef U_backup = U; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wm_homefile_read(C, | 
					
						
							|  |  |  |                    op->reports, | 
					
						
							|  |  |  |                    use_factory_settings, | 
					
						
							|  |  |  |                    false, | 
					
						
							|  |  |  |                    use_data, | 
					
						
							|  |  |  |                    use_userdef, | 
					
						
							|  |  |  |                    NULL, | 
					
						
							|  |  |  |                    WM_init_state_app_template_get(), | 
					
						
							|  |  |  |                    NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-12 12:21:21 +10:00
										 |  |  |   wm_userpref_read_exceptions(&U, &U_backup); | 
					
						
							| 
									
										
										
										
											2019-06-12 17:53:18 +10:00
										 |  |  |   SET_FLAG_FROM_TEST(G.f, use_factory_settings, G_FLAG_USERPREF_NO_SAVE_ON_EXIT); | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wm_userpref_update_when_changed(C, bmain, &U_backup, &U); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 15:51:49 +10:00
										 |  |  |   if (use_factory_settings) { | 
					
						
							|  |  |  |     U.runtime.is_dirty = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 14:42:09 +10:00
										 |  |  |   /* Needed to recalculate UI scaling values (eg, #UserDef.inv_dpi_fac). */ | 
					
						
							|  |  |  |   wm_window_clear_drawable(bmain->wm.first); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   WM_event_add_notifier(C, NC_WINDOW, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_read_userpref(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ot->name = "Load Preferences"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_read_userpref"; | 
					
						
							|  |  |  |   ot->description = "Load last saved preferences"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_userpref_read_exec; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 13:25:51 +10:00
										 |  |  | void WM_OT_read_factory_userpref(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ot->name = "Load Factory Preferences"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_read_factory_userpref"; | 
					
						
							| 
									
										
										
										
											2019-07-17 09:29:28 +10:00
										 |  |  |   ot->description = | 
					
						
							|  |  |  |       "Load factory default preferences. " | 
					
						
							|  |  |  |       "To make changes to preferences permanent, use \"Save Preferences\""; | 
					
						
							| 
									
										
										
										
											2019-05-13 13:25:51 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_userpref_read_exec; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read File History Operator
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | static int wm_history_file_read_exec(bContext *UNUSED(C), wmOperator *UNUSED(op)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ED_file_read_bookmarks(); | 
					
						
							|  |  |  |   wm_history_file_read(); | 
					
						
							|  |  |  |   return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_read_history(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Reload History File"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_read_history"; | 
					
						
							|  |  |  |   ot->description = "Reloads history and bookmarks"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_history_file_read_exec; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* this operator is only used for loading settings from a previous blender install */ | 
					
						
							|  |  |  |   ot->flag = OPTYPE_INTERNAL; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Read Startup & Preferences Operator
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Both #WM_OT_read_homefile & #WM_OT_read_factory_settings. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | static int wm_homefile_read_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 17:44:08 +10:00
										 |  |  |   const bool use_factory_startup_and_userdef = STREQ(op->type->idname, | 
					
						
							|  |  |  |                                                      "WM_OT_read_factory_settings"); | 
					
						
							|  |  |  |   const bool use_factory_settings = use_factory_startup_and_userdef || | 
					
						
							|  |  |  |                                     RNA_boolean_get(op->ptr, "use_factory_startup"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool use_userdef = false; | 
					
						
							|  |  |  |   char filepath_buf[FILE_MAX]; | 
					
						
							|  |  |  |   const char *filepath = NULL; | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  |   UserDef U_backup = U; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!use_factory_settings) { | 
					
						
							|  |  |  |     PropertyRNA *prop = RNA_struct_find_property(op->ptr, "filepath"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* This can be used when loading of a start-up file should only change
 | 
					
						
							|  |  |  |      * the scene content but keep the blender UI as it is. */ | 
					
						
							|  |  |  |     wm_open_init_load_ui(op, true); | 
					
						
							|  |  |  |     SET_FLAG_FROM_TEST(G.fileflags, !RNA_boolean_get(op->ptr, "load_ui"), G_FILE_NO_UI); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |       RNA_property_string_get(op->ptr, prop, filepath_buf); | 
					
						
							|  |  |  |       filepath = filepath_buf; | 
					
						
							|  |  |  |       if (BLI_access(filepath, R_OK)) { | 
					
						
							|  |  |  |         BKE_reportf( | 
					
						
							|  |  |  |             op->reports, RPT_ERROR, "Can't read alternative start-up file: '%s'", filepath); | 
					
						
							|  |  |  |         return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2020-09-02 17:44:08 +10:00
										 |  |  |     if (use_factory_startup_and_userdef) { | 
					
						
							|  |  |  |       /* always load UI for factory settings (prefs will re-init) */ | 
					
						
							|  |  |  |       G.fileflags &= ~G_FILE_NO_UI; | 
					
						
							|  |  |  |       /* Always load preferences with factory settings. */ | 
					
						
							|  |  |  |       use_userdef = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char app_template_buf[sizeof(U.app_template)]; | 
					
						
							|  |  |  |   const char *app_template; | 
					
						
							|  |  |  |   PropertyRNA *prop_app_template = RNA_struct_find_property(op->ptr, "app_template"); | 
					
						
							|  |  |  |   const bool use_splash = !use_factory_settings && RNA_boolean_get(op->ptr, "use_splash"); | 
					
						
							|  |  |  |   const bool use_empty_data = RNA_boolean_get(op->ptr, "use_empty"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (prop_app_template && RNA_property_is_set(op->ptr, prop_app_template)) { | 
					
						
							|  |  |  |     RNA_property_string_get(op->ptr, prop_app_template, app_template_buf); | 
					
						
							|  |  |  |     app_template = app_template_buf; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-05 19:09:21 +10:00
										 |  |  |     if (!use_factory_settings) { | 
					
						
							|  |  |  |       /* Always load preferences when switching templates with own preferences. */ | 
					
						
							|  |  |  |       use_userdef = BKE_appdir_app_template_has_userpref(app_template) || | 
					
						
							|  |  |  |                     BKE_appdir_app_template_has_userpref(U.app_template); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Turn override off, since we're explicitly loading a different app-template. */ | 
					
						
							|  |  |  |     WM_init_state_app_template_set(NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Normally NULL, only set when overriding from the command-line. */ | 
					
						
							|  |  |  |     app_template = WM_init_state_app_template_get(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |   bool use_data = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_homefile_read(C, | 
					
						
							|  |  |  |                    op->reports, | 
					
						
							|  |  |  |                    use_factory_settings, | 
					
						
							|  |  |  |                    use_empty_data, | 
					
						
							| 
									
										
										
										
											2019-05-13 12:44:08 +10:00
										 |  |  |                    use_data, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                    use_userdef, | 
					
						
							|  |  |  |                    filepath, | 
					
						
							|  |  |  |                    app_template, | 
					
						
							|  |  |  |                    NULL); | 
					
						
							|  |  |  |   if (use_splash) { | 
					
						
							|  |  |  |     WM_init_splash(C); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-14 12:23:55 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  |   if (use_userdef) { | 
					
						
							| 
									
										
										
										
											2019-06-12 12:21:21 +10:00
										 |  |  |     wm_userpref_read_exceptions(&U, &U_backup); | 
					
						
							| 
									
										
										
										
											2019-06-12 17:53:18 +10:00
										 |  |  |     SET_FLAG_FROM_TEST(G.f, use_factory_settings, G_FLAG_USERPREF_NO_SAVE_ON_EXIT); | 
					
						
							| 
									
										
										
										
											2019-08-11 18:03:17 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (use_factory_settings) { | 
					
						
							|  |  |  |       U.runtime.is_dirty = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-11 21:49:05 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-28 20:46:50 -06:00
										 |  |  |   if (G.fileflags & G_FILE_NO_UI) { | 
					
						
							|  |  |  |     ED_outliner_select_sync_from_all_tag(C); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | static void wm_homefile_read_after_dialog_callback(bContext *C, void *user_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WM_operator_name_call_with_properties( | 
					
						
							|  |  |  |       C, "WM_OT_read_homefile", WM_OP_EXEC_DEFAULT, (IDProperty *)user_data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_free_operator_properties_callback(void *user_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   IDProperty *properties = (IDProperty *)user_data; | 
					
						
							|  |  |  |   IDP_FreeProperty(properties); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-24 17:45:34 +02:00
										 |  |  | static int wm_homefile_read_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   if (U.uiflag & USER_SAVE_PROMPT && | 
					
						
							|  |  |  |       wm_file_or_image_is_modified(CTX_data_main(C), CTX_wm_manager(C))) { | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  |     wmGenericCallback *callback = MEM_callocN(sizeof(*callback), __func__); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |     callback->exec = wm_homefile_read_after_dialog_callback; | 
					
						
							|  |  |  |     callback->user_data = IDP_CopyProperty(op->properties); | 
					
						
							|  |  |  |     callback->free_user_data = wm_free_operator_properties_callback; | 
					
						
							|  |  |  |     wm_close_file_dialog(C, callback); | 
					
						
							|  |  |  |     return OPERATOR_INTERFACE; | 
					
						
							| 
									
										
										
										
											2019-04-24 17:45:34 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   return wm_homefile_read_exec(C, op); | 
					
						
							| 
									
										
										
										
											2019-04-24 17:45:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 12:23:55 +10:00
										 |  |  | static void read_homefile_props(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   PropertyRNA *prop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prop = RNA_def_string(ot->srna, "app_template", "Template", sizeof(U.app_template), "", ""); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prop = RNA_def_boolean(ot->srna, "use_empty", false, "Empty", ""); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | void WM_OT_read_homefile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PropertyRNA *prop; | 
					
						
							|  |  |  |   ot->name = "Reload Start-Up File"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_read_homefile"; | 
					
						
							|  |  |  |   ot->description = "Open the default file (doesn't save the current file)"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-24 17:45:34 +02:00
										 |  |  |   ot->invoke = wm_homefile_read_invoke; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->exec = wm_homefile_read_exec; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   prop = RNA_def_string_file_path( | 
					
						
							|  |  |  |       ot->srna, "filepath", NULL, FILE_MAX, "File Path", "Path to an alternative start-up file"); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* So scripts can use an alternative start-up file without the UI */ | 
					
						
							|  |  |  |   prop = RNA_def_boolean( | 
					
						
							|  |  |  |       ot->srna, "load_ui", true, "Load UI", "Load user interface setup from the .blend file"); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* So the splash can be kept open after loading a file (for templates). */ | 
					
						
							|  |  |  |   prop = RNA_def_boolean(ot->srna, "use_splash", false, "Splash", ""); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							| 
									
										
										
										
											2017-03-14 21:05:00 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 17:44:08 +10:00
										 |  |  |   /* So scripts can load factory-startup without resetting preferences
 | 
					
						
							|  |  |  |    * (which has other implications such as reloading all add-ons). | 
					
						
							|  |  |  |    * Match naming for `--factory-startup` command line argument. */ | 
					
						
							|  |  |  |   prop = RNA_def_boolean(ot->srna, "use_factory_startup", false, "Factory Startup", ""); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 12:23:55 +10:00
										 |  |  |   read_homefile_props(ot); | 
					
						
							| 
									
										
										
										
											2017-03-25 09:29:51 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* omit poll to run in background mode */ | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_read_factory_settings(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Load Factory Settings"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_read_factory_settings"; | 
					
						
							| 
									
										
										
										
											2019-07-17 09:29:28 +10:00
										 |  |  |   ot->description = | 
					
						
							|  |  |  |       "Load factory default startup file and preferences. " | 
					
						
							|  |  |  |       "To make changes permanent, use \"Save Startup File\" and \"Save Preferences\""; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							|  |  |  |   ot->exec = wm_homefile_read_exec; | 
					
						
							| 
									
										
										
										
											2017-03-25 09:29:51 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 12:23:55 +10:00
										 |  |  |   read_homefile_props(ot); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* omit poll to run in background mode */ | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Open Main .blend File Utilities
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Wrap #WM_file_read, shared by file reading operators. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | static bool wm_file_read_opwrap(bContext *C, const char *filepath, ReportList *reports) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool success; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* XXX wm in context is not set correctly after WM_file_read -> crash */ | 
					
						
							|  |  |  |   /* do it before for now, but is this correct with multiple windows? */ | 
					
						
							|  |  |  |   WM_event_add_notifier(C, NC_WINDOW, NULL); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   /* Set by the "use_scripts" property on file load. */ | 
					
						
							|  |  |  |   if ((G.f & G_FLAG_SCRIPT_AUTOEXEC) == 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     WM_file_autoexec_init(filepath); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   success = WM_file_read(C, filepath, reports); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return success; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* Generic operator state utilities */ | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void create_operator_state(wmOperatorType *ot, int first_state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   PropertyRNA *prop = RNA_def_int( | 
					
						
							|  |  |  |       ot->srna, "state", first_state, INT32_MIN, INT32_MAX, "State", "", INT32_MIN, INT32_MAX); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_SKIP_SAVE); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int get_operator_state(wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return RNA_int_get(op->ptr, "state"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_next_operator_state(wmOperator *op, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   RNA_int_set(op->ptr, "state", state); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct OperatorDispatchTarget { | 
					
						
							|  |  |  |   int state; | 
					
						
							|  |  |  |   int (*run)(bContext *C, wmOperator *op); | 
					
						
							|  |  |  | } OperatorDispatchTarget; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int operator_state_dispatch(bContext *C, wmOperator *op, OperatorDispatchTarget *targets) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int state = get_operator_state(op); | 
					
						
							|  |  |  |   for (int i = 0; targets[i].run; i++) { | 
					
						
							|  |  |  |     OperatorDispatchTarget target = targets[i]; | 
					
						
							|  |  |  |     if (target.state == state) { | 
					
						
							|  |  |  |       return target.run(C, op); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   BLI_assert(false); | 
					
						
							|  |  |  |   return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Open Main .blend File Operator
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |   OPEN_MAINFILE_STATE_DISCARD_CHANGES, | 
					
						
							|  |  |  |   OPEN_MAINFILE_STATE_SELECT_FILE_PATH, | 
					
						
							|  |  |  |   OPEN_MAINFILE_STATE_OPEN, | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | static int wm_open_mainfile_dispatch(bContext *C, wmOperator *op); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | static void wm_open_mainfile_after_dialog_callback(bContext *C, void *user_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WM_operator_name_call_with_properties( | 
					
						
							|  |  |  |       C, "WM_OT_open_mainfile", WM_OP_INVOKE_DEFAULT, (IDProperty *)user_data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | static int wm_open_mainfile__discard_changes(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  |   if (RNA_boolean_get(op->ptr, "display_file_selector")) { | 
					
						
							|  |  |  |     set_next_operator_state(op, OPEN_MAINFILE_STATE_SELECT_FILE_PATH); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     set_next_operator_state(op, OPEN_MAINFILE_STATE_OPEN); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   if (U.uiflag & USER_SAVE_PROMPT && | 
					
						
							|  |  |  |       wm_file_or_image_is_modified(CTX_data_main(C), CTX_wm_manager(C))) { | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  |     wmGenericCallback *callback = MEM_callocN(sizeof(*callback), __func__); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |     callback->exec = wm_open_mainfile_after_dialog_callback; | 
					
						
							|  |  |  |     callback->user_data = IDP_CopyProperty(op->properties); | 
					
						
							|  |  |  |     callback->free_user_data = wm_free_operator_properties_callback; | 
					
						
							|  |  |  |     wm_close_file_dialog(C, callback); | 
					
						
							|  |  |  |     return OPERATOR_INTERFACE; | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   return wm_open_mainfile_dispatch(C, op); | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_open_mainfile__select_file_path(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   set_next_operator_state(op, OPEN_MAINFILE_STATE_OPEN); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   const char *openname = BKE_main_blendfile_path(bmain); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (CTX_wm_window(C) == NULL) { | 
					
						
							|  |  |  |     /* in rare cases this could happen, when trying to invoke in background
 | 
					
						
							|  |  |  |      * mode on load for example. Don't use poll for this because exec() | 
					
						
							|  |  |  |      * can still run without a window */ | 
					
						
							|  |  |  |     BKE_report(op->reports, RPT_ERROR, "Context window not set"); | 
					
						
							|  |  |  |     return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* if possible, get the name of the most recently used .blend file */ | 
					
						
							|  |  |  |   if (G.recent_files.first) { | 
					
						
							|  |  |  |     struct RecentFile *recent = G.recent_files.first; | 
					
						
							|  |  |  |     openname = recent->filepath; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_string_set(op->ptr, "filepath", openname); | 
					
						
							|  |  |  |   wm_open_init_load_ui(op, true); | 
					
						
							|  |  |  |   wm_open_init_use_scripts(op, true); | 
					
						
							|  |  |  |   op->customdata = NULL; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_event_add_fileselect(C, op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_RUNNING_MODAL; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | static int wm_open_mainfile__open(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filepath[FILE_MAX]; | 
					
						
							|  |  |  |   bool success; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_string_get(op->ptr, "filepath", filepath); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* re-use last loaded setting so we can reload a file without changing */ | 
					
						
							|  |  |  |   wm_open_init_load_ui(op, false); | 
					
						
							|  |  |  |   wm_open_init_use_scripts(op, false); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   SET_FLAG_FROM_TEST(G.fileflags, !RNA_boolean_get(op->ptr, "load_ui"), G_FILE_NO_UI); | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(G.f, RNA_boolean_get(op->ptr, "use_scripts"), G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							|  |  |  |   success = wm_file_read_opwrap(C, filepath, op->reports); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* for file open also popup for warnings, not only errors */ | 
					
						
							|  |  |  |   BKE_report_print_level_set(op->reports, RPT_WARNING); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (success) { | 
					
						
							| 
									
										
										
										
											2019-08-28 20:46:50 -06:00
										 |  |  |     if (G.fileflags & G_FILE_NO_UI) { | 
					
						
							|  |  |  |       ED_outliner_select_sync_from_all_tag(C); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-19 14:25:29 -03:00
										 |  |  |     ED_view3d_local_collections_reset(C, (G.fileflags & G_FILE_NO_UI) != 0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return OPERATOR_FINISHED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | static OperatorDispatchTarget wm_open_mainfile_dispatch_targets[] = { | 
					
						
							|  |  |  |     {OPEN_MAINFILE_STATE_DISCARD_CHANGES, wm_open_mainfile__discard_changes}, | 
					
						
							|  |  |  |     {OPEN_MAINFILE_STATE_SELECT_FILE_PATH, wm_open_mainfile__select_file_path}, | 
					
						
							|  |  |  |     {OPEN_MAINFILE_STATE_OPEN, wm_open_mainfile__open}, | 
					
						
							|  |  |  |     {0, NULL}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_open_mainfile_dispatch(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return operator_state_dispatch(C, op, wm_open_mainfile_dispatch_targets); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_open_mainfile_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return wm_open_mainfile_dispatch(C, op); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_open_mainfile_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return wm_open_mainfile__open(C, op); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 13:57:31 -05:00
										 |  |  | static char *wm_open_mainfile_description(struct bContext *UNUSED(C), | 
					
						
							|  |  |  |                                           struct wmOperatorType *UNUSED(op), | 
					
						
							|  |  |  |                                           struct PointerRNA *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!RNA_struct_property_is_set(params, "filepath")) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Filepath. */ | 
					
						
							|  |  |  |   char path[FILE_MAX]; | 
					
						
							|  |  |  |   RNA_string_get(params, "filepath", path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_stat_t stats; | 
					
						
							|  |  |  |   if (BLI_stat(path, &stats) == -1) { | 
					
						
							|  |  |  |     return BLI_sprintfN("%s\n\n%s", path, N_("File Not Found")); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Date. */ | 
					
						
							| 
									
										
										
										
											2020-10-15 10:54:06 +11:00
										 |  |  |   char date_st[FILELIST_DIRENTRY_DATE_LEN]; | 
					
						
							|  |  |  |   char time_st[FILELIST_DIRENTRY_TIME_LEN]; | 
					
						
							| 
									
										
										
										
											2020-10-14 13:57:31 -05:00
										 |  |  |   bool is_today, is_yesterday; | 
					
						
							|  |  |  |   BLI_filelist_entry_datetime_to_string( | 
					
						
							|  |  |  |       NULL, (int64_t)stats.st_mtime, false, time_st, date_st, &is_today, &is_yesterday); | 
					
						
							|  |  |  |   if (is_today || is_yesterday) { | 
					
						
							|  |  |  |     BLI_strncpy(date_st, is_today ? N_("Today") : N_("Yesterday"), sizeof(date_st)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Size. */ | 
					
						
							| 
									
										
										
										
											2020-10-15 10:54:06 +11:00
										 |  |  |   char size_str[FILELIST_DIRENTRY_SIZE_LEN]; | 
					
						
							| 
									
										
										
										
											2020-10-14 13:57:31 -05:00
										 |  |  |   BLI_filelist_entry_size_to_string(NULL, (uint64_t)stats.st_size, false, size_str); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return BLI_sprintfN( | 
					
						
							|  |  |  |       "%s\n\n%s: %s %s\n%s: %s", path, N_("Modified"), date_st, time_st, N_("Size"), size_str); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-15 10:54:06 +11:00
										 |  |  | /* currently fits in a pointer */ | 
					
						
							|  |  |  | struct FileRuntime { | 
					
						
							|  |  |  |   bool is_untrusted; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | BLI_STATIC_ASSERT(sizeof(struct FileRuntime) <= sizeof(void *), | 
					
						
							|  |  |  |                   "Struct must not exceed pointer size"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | static bool wm_open_mainfile_check(bContext *UNUSED(C), wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   struct FileRuntime *file_info = (struct FileRuntime *)&op->customdata; | 
					
						
							|  |  |  |   PropertyRNA *prop = RNA_struct_find_property(op->ptr, "use_scripts"); | 
					
						
							|  |  |  |   bool is_untrusted = false; | 
					
						
							|  |  |  |   char path[FILE_MAX]; | 
					
						
							|  |  |  |   char *lslash; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_string_get(op->ptr, "filepath", path); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* get the dir */ | 
					
						
							| 
									
										
										
										
											2020-04-07 12:02:21 +10:00
										 |  |  |   lslash = (char *)BLI_path_slash_rfind(path); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (lslash) { | 
					
						
							|  |  |  |     *(lslash + 1) = '\0'; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0) { | 
					
						
							|  |  |  |     if (BKE_autoexec_match(path) == true) { | 
					
						
							|  |  |  |       RNA_property_boolean_set(op->ptr, prop, false); | 
					
						
							|  |  |  |       is_untrusted = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (file_info) { | 
					
						
							|  |  |  |     file_info->is_untrusted = is_untrusted; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return is_untrusted; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_open_mainfile_ui(bContext *UNUSED(C), wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   struct FileRuntime *file_info = (struct FileRuntime *)&op->customdata; | 
					
						
							|  |  |  |   uiLayout *layout = op->layout; | 
					
						
							|  |  |  |   uiLayout *col = op->layout; | 
					
						
							|  |  |  |   const char *autoexec_text; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uiItemR(layout, op->ptr, "load_ui", 0, NULL, ICON_NONE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   col = uiLayoutColumn(layout, false); | 
					
						
							|  |  |  |   if (file_info->is_untrusted) { | 
					
						
							|  |  |  |     autoexec_text = IFACE_("Trusted Source [Untrusted Path]"); | 
					
						
							|  |  |  |     uiLayoutSetActive(col, false); | 
					
						
							|  |  |  |     uiLayoutSetEnabled(col, false); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     autoexec_text = IFACE_("Trusted Source"); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uiItemR(col, op->ptr, "use_scripts", 0, autoexec_text, ICON_NONE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | static void wm_open_mainfile_def_property_use_scripts(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   RNA_def_boolean(ot->srna, | 
					
						
							|  |  |  |                   "use_scripts", | 
					
						
							|  |  |  |                   true, | 
					
						
							|  |  |  |                   "Trusted Source", | 
					
						
							|  |  |  |                   "Allow .blend file to execute scripts automatically, default available from " | 
					
						
							|  |  |  |                   "system preferences"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | void WM_OT_open_mainfile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
											  
											
												UI: File Browser Design Overhaul
This is a general redesign of the File Browser GUI and interaction
methods. For screenshots, check patch D5601.
Main changes in short:
* File Browser as floating window
* New layout of regions
* Popovers for view and filter options
* Vertical list view with interactive column header
* New and updated icons
* Keymap consistency fixes
* Many tweaks and fixes to the drawing of views
----
General:
* The file browser now opens as temporary floating window. It closes on
  Esc. The header is hidden then.
* When the file browser is opened as regular editor, the header remains
  visible.
* All file browser regions are now defined in Python (the button
  layout).
* Adjusted related operator UI names.
Keymap:
Keymap is now consistent with other list-based views in Blender, such as
the Outliner.
* Left click to select, double-click to open
* Right-click context menus
* Shift-click to fill selection
* Ctrl-click to extend selection
Operator options:
These previously overlapped with the source list, which caused numerous
issues with resizing and presenting many settings in a small panel area.
It was also generally inconsistent with Blender.
* Moved to new sidebar, which can easily be shown or hidden using a
  prominent Options toggle.
* IO operators have new layouts to match this new sidebar, using
  sub-panels. This will have to be committed separately (Add-on
  repository).
* If operators want to show the options by default, they have the option
  to do so (see `WM_FILESEL_SHOW_PROPS`, `hide_props_region`), otherwise
  they are hidden by default.
General Layout:
The layout has been changed to be simpler, more standard, and fits
better in with Blender 2.8.
* More conventional layout (file path at top, file name at the bottom,
  execute/cancel buttons in bottom right).
* Use of popovers to group controls, and allow for more descriptive
  naming.
* Search box is always live now, just like Outliner.
Views:
* Date Modified column combines both date and time, also uses user
  friendly strings for recent dates (i.e. "Yesterday", "Today").
* Details columns (file size, modification date/time) are now toggleable
  for all display types, they are not hardcoded per display type.
* File sizes now show as B, KB, MB, ... rather than B, KiB, MiB, … They
  are now also calculated using base 10 of course.
* Option to sort in inverse order.
Vertical List View:
* This view now used a much simpler single vertical list with columns
  for information.
* Users can click on the headers of these columns to order by that
  category, and click again to reverse the ordering.
Icons:
* Updated icons by Jendrzych, with better centering.
* Files and folders have new icons in Icon view.
* Both files and folders have reworked superimposed icons that show
  users the file/folder type.
* 3D file documents correctly use the 3d file icon, which was unused
  previously.
* Workspaces now show their icon on Link/Append - also when listed in
  the Outliner.
Minor Python-API breakage:
* `bpy.types.FileSelectParams.display_type`: `LIST_SHORT` and
  `LIST_LONG` are replaced by `LIST_VERTICAL` and `LIST_HORIZONTAL`.
Removes the feature where directories would automatically be created if
they are entered into the file path text button, but don't exist. We
were not sure if users use it enough to keep it. We can definitely bring
it back.
----
//Combined effort by @billreynish, @harley, @jendrzych, my university
colleague Brian Meisenheimer and myself.//
Differential Revision: https://developer.blender.org/D5601
Reviewers: Brecht, Bastien
											
										 
											2019-09-03 15:43:38 +02:00
										 |  |  |   ot->name = "Open"; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->idname = "WM_OT_open_mainfile"; | 
					
						
							|  |  |  |   ot->description = "Open a Blender file"; | 
					
						
							| 
									
										
										
										
											2020-10-14 13:57:31 -05:00
										 |  |  |   ot->get_description = wm_open_mainfile_description; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ot->invoke = wm_open_mainfile_invoke; | 
					
						
							|  |  |  |   ot->exec = wm_open_mainfile_exec; | 
					
						
							|  |  |  |   ot->check = wm_open_mainfile_check; | 
					
						
							|  |  |  |   ot->ui = wm_open_mainfile_ui; | 
					
						
							|  |  |  |   /* omit window poll so this can work in background mode */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WM_operator_properties_filesel(ot, | 
					
						
							|  |  |  |                                  FILE_TYPE_FOLDER | FILE_TYPE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_OPENFILE, | 
					
						
							|  |  |  |                                  WM_FILESEL_FILEPATH, | 
					
						
							|  |  |  |                                  FILE_DEFAULTDISPLAY, | 
					
						
							| 
									
										
										
										
											2020-11-02 23:55:59 +01:00
										 |  |  |                                  FILE_SORT_DEFAULT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   RNA_def_boolean( | 
					
						
							|  |  |  |       ot->srna, "load_ui", true, "Load UI", "Load user interface setup in the .blend file"); | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   wm_open_mainfile_def_property_use_scripts(ot); | 
					
						
							| 
									
										
										
										
											2019-05-14 14:13:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   PropertyRNA *prop = RNA_def_boolean( | 
					
						
							|  |  |  |       ot->srna, "display_file_selector", true, "Display File Selector", ""); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_SKIP_SAVE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   create_operator_state(ot, OPEN_MAINFILE_STATE_DISCARD_CHANGES); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Reload (revert) Main .blend File Operator
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_revert_mainfile_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   bool success; | 
					
						
							|  |  |  |   char filepath[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_open_init_use_scripts(op, false); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   SET_FLAG_FROM_TEST(G.f, RNA_boolean_get(op->ptr, "use_scripts"), G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_strncpy(filepath, BKE_main_blendfile_path(bmain), sizeof(filepath)); | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   success = wm_file_read_opwrap(C, filepath, op->reports); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (success) { | 
					
						
							|  |  |  |     return OPERATOR_FINISHED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:47:00 +02:00
										 |  |  | static bool wm_revert_mainfile_poll(bContext *UNUSED(C)) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return G.relbase_valid; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_revert_mainfile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Revert"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_revert_mainfile"; | 
					
						
							|  |  |  |   ot->description = "Reload the saved file"; | 
					
						
							| 
									
										
										
										
											2019-09-07 21:08:20 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = WM_operator_confirm; | 
					
						
							| 
									
										
										
										
											2019-09-07 21:08:20 +10:00
										 |  |  |   ot->exec = wm_revert_mainfile_exec; | 
					
						
							|  |  |  |   ot->poll = wm_revert_mainfile_poll; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   wm_open_mainfile_def_property_use_scripts(ot); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \name Recover Last Session Operator
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 15:08:28 +11:00
										 |  |  | bool WM_recover_last_session(bContext *C, ReportList *reports) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filepath[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2020-03-07 13:23:26 +11:00
										 |  |  |   BLI_join_dirfile(filepath, sizeof(filepath), BKE_tempdir_base(), BLENDER_QUIT_FILE); | 
					
						
							| 
									
										
										
										
											2021-01-26 15:08:28 +11:00
										 |  |  |   G.fileflags |= G_FILE_RECOVER; | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   const bool success = wm_file_read_opwrap(C, filepath, reports); | 
					
						
							| 
									
										
										
										
											2021-01-26 15:08:28 +11:00
										 |  |  |   G.fileflags &= ~G_FILE_RECOVER; | 
					
						
							|  |  |  |   return success; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_recover_last_session_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  |   wm_open_init_use_scripts(op, true); | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(G.f, RNA_boolean_get(op->ptr, "use_scripts"), G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							| 
									
										
										
										
											2021-01-26 15:08:28 +11:00
										 |  |  |   if (WM_recover_last_session(C, op->reports)) { | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       wmOperatorType *ot = op->type; | 
					
						
							|  |  |  |       PointerRNA *props_ptr = MEM_callocN(sizeof(PointerRNA), __func__); | 
					
						
							|  |  |  |       WM_operator_properties_create_ptr(props_ptr, ot); | 
					
						
							|  |  |  |       RNA_boolean_set(props_ptr, "use_scripts", true); | 
					
						
							|  |  |  |       wm_test_autorun_revert_action_set(ot, props_ptr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-26 15:08:28 +11:00
										 |  |  |     return OPERATOR_FINISHED; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | static int wm_recover_last_session_invoke(bContext *C, wmOperator *op, const wmEvent *event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Keep the current setting instead of using the preferences since a file selector
 | 
					
						
							|  |  |  |    * doesn't give us the option to change the setting. */ | 
					
						
							|  |  |  |   wm_open_init_use_scripts(op, false); | 
					
						
							|  |  |  |   return WM_operator_confirm(C, op, event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | void WM_OT_recover_last_session(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Recover Last Session"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_recover_last_session"; | 
					
						
							|  |  |  |   ot->description = "Open the last closed file (\"" BLENDER_QUIT_FILE "\")"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   ot->invoke = wm_recover_last_session_invoke; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->exec = wm_recover_last_session_exec; | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   wm_open_mainfile_def_property_use_scripts(ot); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Auto-Save Main .blend File Operator
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | static int wm_recover_auto_save_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filepath[FILE_MAX]; | 
					
						
							|  |  |  |   bool success; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_string_get(op->ptr, "filepath", filepath); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   wm_open_init_use_scripts(op, true); | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(G.f, RNA_boolean_get(op->ptr, "use_scripts"), G_FLAG_SCRIPT_AUTOEXEC); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   G.fileflags |= G_FILE_RECOVER; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   success = wm_file_read_opwrap(C, filepath, op->reports); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   G.fileflags &= ~G_FILE_RECOVER; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (success) { | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  |     if (!G.background) { | 
					
						
							|  |  |  |       wmOperatorType *ot = op->type; | 
					
						
							|  |  |  |       PointerRNA *props_ptr = MEM_callocN(sizeof(PointerRNA), __func__); | 
					
						
							|  |  |  |       WM_operator_properties_create_ptr(props_ptr, ot); | 
					
						
							|  |  |  |       RNA_boolean_set(props_ptr, "use_scripts", true); | 
					
						
							|  |  |  |       wm_test_autorun_revert_action_set(ot, props_ptr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return OPERATOR_FINISHED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 11:23:02 +02:00
										 |  |  |   return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_recover_auto_save_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filename[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wm_autosave_location(filename); | 
					
						
							|  |  |  |   RNA_string_set(op->ptr, "filepath", filename); | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  |   wm_open_init_use_scripts(op, true); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_event_add_fileselect(C, op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_RUNNING_MODAL; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_recover_auto_save(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Recover Auto Save"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_recover_auto_save"; | 
					
						
							|  |  |  |   ot->description = "Open an automatically saved file to recover it"; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->invoke = wm_recover_auto_save_invoke; | 
					
						
							| 
									
										
										
										
											2019-09-07 21:08:20 +10:00
										 |  |  |   ot->exec = wm_recover_auto_save_exec; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_operator_properties_filesel(ot, | 
					
						
							|  |  |  |                                  FILE_TYPE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_OPENFILE, | 
					
						
							|  |  |  |                                  WM_FILESEL_FILEPATH, | 
					
						
							| 
									
										
										
										
											2019-09-20 16:16:09 +02:00
										 |  |  |                                  FILE_VERTICALDISPLAY, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                  FILE_SORT_TIME); | 
					
						
							| 
									
										
										
										
											2021-01-26 13:21:13 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   wm_open_mainfile_def_property_use_scripts(ot); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Save Main .blend File Operator
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Both #WM_OT_save_as_mainfile & #WM_OT_save_mainfile. | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_filepath_default(char *filepath) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G.save_over == false) { | 
					
						
							| 
									
										
										
										
											2020-04-07 12:02:21 +10:00
										 |  |  |     BLI_path_filename_ensure(filepath, FILE_MAX, "untitled.blend"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void save_set_compress(wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PropertyRNA *prop; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   prop = RNA_struct_find_property(op->ptr, "compress"); | 
					
						
							|  |  |  |   if (!RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |     if (G.save_over) { /* keep flag for existing file */ | 
					
						
							|  |  |  |       RNA_property_boolean_set(op->ptr, prop, (G.fileflags & G_FILE_COMPRESS) != 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { /* use userdef for new file */ | 
					
						
							|  |  |  |       RNA_property_boolean_set(op->ptr, prop, (U.flag & USER_FILECOMPRESS) != 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-05 15:10:33 +02:00
										 |  |  | static void save_set_filepath(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   PropertyRNA *prop; | 
					
						
							|  |  |  |   char name[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   prop = RNA_struct_find_property(op->ptr, "filepath"); | 
					
						
							|  |  |  |   if (!RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |     /* if not saved before, get the name of the most recently used .blend file */ | 
					
						
							|  |  |  |     if (BKE_main_blendfile_path(bmain)[0] == '\0' && G.recent_files.first) { | 
					
						
							|  |  |  |       struct RecentFile *recent = G.recent_files.first; | 
					
						
							|  |  |  |       BLI_strncpy(name, recent->filepath, FILE_MAX); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       BLI_strncpy(name, bmain->name, FILE_MAX); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     wm_filepath_default(name); | 
					
						
							|  |  |  |     RNA_property_string_set(op->ptr, prop, name); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_save_as_mainfile_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   save_set_compress(op); | 
					
						
							|  |  |  |   save_set_filepath(C, op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_event_add_fileselect(C, op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_RUNNING_MODAL; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* function used for WM_OT_save_mainfile too */ | 
					
						
							|  |  |  | static int wm_save_as_mainfile_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   char path[FILE_MAX]; | 
					
						
							|  |  |  |   const bool is_save_as = (op->type->invoke == wm_save_as_mainfile_invoke); | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   const bool use_save_as_copy = (RNA_struct_property_is_set(op->ptr, "copy") && | 
					
						
							|  |  |  |                                  RNA_boolean_get(op->ptr, "copy")); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-18 15:25:22 +10:00
										 |  |  |   /* We could expose all options to the users however in most cases remapping
 | 
					
						
							|  |  |  |    * existing relative paths is a good default. | 
					
						
							|  |  |  |    * Users can manually make their paths relative & absolute if they wish. */ | 
					
						
							|  |  |  |   const eBLO_WritePathRemap remap_mode = RNA_boolean_get(op->ptr, "relative_remap") ? | 
					
						
							|  |  |  |                                              BLO_WRITE_PATH_REMAP_RELATIVE : | 
					
						
							|  |  |  |                                              BLO_WRITE_PATH_REMAP_NONE; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   save_set_compress(op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (RNA_struct_property_is_set(op->ptr, "filepath")) { | 
					
						
							|  |  |  |     RNA_string_get(op->ptr, "filepath", path); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     BLI_strncpy(path, BKE_main_blendfile_path(bmain), FILE_MAX); | 
					
						
							|  |  |  |     wm_filepath_default(path); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int fileflags_orig = G.fileflags; | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   int fileflags = G.fileflags; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* set compression flag */ | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(fileflags, RNA_boolean_get(op->ptr, "compress"), G_FILE_COMPRESS); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 15:41:07 +10:00
										 |  |  |   const bool ok = wm_file_write(C, path, fileflags, remap_mode, use_save_as_copy, op->reports); | 
					
						
							| 
									
										
										
										
											2018-10-12 10:24:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((op->flag & OP_IS_INVOKE) == 0) { | 
					
						
							|  |  |  |     /* OP_IS_INVOKE is set when the operator is called from the GUI.
 | 
					
						
							|  |  |  |      * If it is not set, the operator is called from a script and | 
					
						
							|  |  |  |      * shouldn't influence G.fileflags. */ | 
					
						
							|  |  |  |     G.fileflags = fileflags_orig; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-10-12 10:24:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ok == false) { | 
					
						
							|  |  |  |     return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   WM_event_add_notifier(C, NC_WM | ND_FILESAVE, NULL); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!is_save_as && RNA_boolean_get(op->ptr, "exit")) { | 
					
						
							|  |  |  |     wm_exit_schedule_delayed(C); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-03-21 16:00:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* function used for WM_OT_save_mainfile too */ | 
					
						
							|  |  |  | static bool blend_save_check(bContext *UNUSED(C), wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   char filepath[FILE_MAX]; | 
					
						
							|  |  |  |   RNA_string_get(op->ptr, "filepath", filepath); | 
					
						
							|  |  |  |   if (!BLO_has_bfile_extension(filepath)) { | 
					
						
							|  |  |  |     /* some users would prefer BLI_path_extension_replace(),
 | 
					
						
							|  |  |  |      * we keep getting nitpicking bug reports about this - campbell */ | 
					
						
							|  |  |  |     BLI_path_extension_ensure(filepath, FILE_MAX, ".blend"); | 
					
						
							|  |  |  |     RNA_string_set(op->ptr, "filepath", filepath); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 19:21:30 +01:00
										 |  |  | static const char *wm_save_as_mainfile_get_name(wmOperatorType *ot, PointerRNA *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (RNA_boolean_get(ptr, "copy")) { | 
					
						
							|  |  |  |     return CTX_IFACE_(ot->translation_context, "Save Copy"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *wm_save_as_mainfile_get_description(bContext *UNUSED(C), | 
					
						
							|  |  |  |                                                  wmOperatorType *UNUSED(ot), | 
					
						
							|  |  |  |                                                  PointerRNA *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (RNA_boolean_get(ptr, "copy")) { | 
					
						
							|  |  |  |     return BLI_strdup( | 
					
						
							|  |  |  |         "Save the current file in the desired location but do not make the saved file active"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | void WM_OT_save_as_mainfile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   PropertyRNA *prop; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												UI: File Browser Design Overhaul
This is a general redesign of the File Browser GUI and interaction
methods. For screenshots, check patch D5601.
Main changes in short:
* File Browser as floating window
* New layout of regions
* Popovers for view and filter options
* Vertical list view with interactive column header
* New and updated icons
* Keymap consistency fixes
* Many tweaks and fixes to the drawing of views
----
General:
* The file browser now opens as temporary floating window. It closes on
  Esc. The header is hidden then.
* When the file browser is opened as regular editor, the header remains
  visible.
* All file browser regions are now defined in Python (the button
  layout).
* Adjusted related operator UI names.
Keymap:
Keymap is now consistent with other list-based views in Blender, such as
the Outliner.
* Left click to select, double-click to open
* Right-click context menus
* Shift-click to fill selection
* Ctrl-click to extend selection
Operator options:
These previously overlapped with the source list, which caused numerous
issues with resizing and presenting many settings in a small panel area.
It was also generally inconsistent with Blender.
* Moved to new sidebar, which can easily be shown or hidden using a
  prominent Options toggle.
* IO operators have new layouts to match this new sidebar, using
  sub-panels. This will have to be committed separately (Add-on
  repository).
* If operators want to show the options by default, they have the option
  to do so (see `WM_FILESEL_SHOW_PROPS`, `hide_props_region`), otherwise
  they are hidden by default.
General Layout:
The layout has been changed to be simpler, more standard, and fits
better in with Blender 2.8.
* More conventional layout (file path at top, file name at the bottom,
  execute/cancel buttons in bottom right).
* Use of popovers to group controls, and allow for more descriptive
  naming.
* Search box is always live now, just like Outliner.
Views:
* Date Modified column combines both date and time, also uses user
  friendly strings for recent dates (i.e. "Yesterday", "Today").
* Details columns (file size, modification date/time) are now toggleable
  for all display types, they are not hardcoded per display type.
* File sizes now show as B, KB, MB, ... rather than B, KiB, MiB, … They
  are now also calculated using base 10 of course.
* Option to sort in inverse order.
Vertical List View:
* This view now used a much simpler single vertical list with columns
  for information.
* Users can click on the headers of these columns to order by that
  category, and click again to reverse the ordering.
Icons:
* Updated icons by Jendrzych, with better centering.
* Files and folders have new icons in Icon view.
* Both files and folders have reworked superimposed icons that show
  users the file/folder type.
* 3D file documents correctly use the 3d file icon, which was unused
  previously.
* Workspaces now show their icon on Link/Append - also when listed in
  the Outliner.
Minor Python-API breakage:
* `bpy.types.FileSelectParams.display_type`: `LIST_SHORT` and
  `LIST_LONG` are replaced by `LIST_VERTICAL` and `LIST_HORIZONTAL`.
Removes the feature where directories would automatically be created if
they are entered into the file path text button, but don't exist. We
were not sure if users use it enough to keep it. We can definitely bring
it back.
----
//Combined effort by @billreynish, @harley, @jendrzych, my university
colleague Brian Meisenheimer and myself.//
Differential Revision: https://developer.blender.org/D5601
Reviewers: Brecht, Bastien
											
										 
											2019-09-03 15:43:38 +02:00
										 |  |  |   ot->name = "Save As"; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->idname = "WM_OT_save_as_mainfile"; | 
					
						
							|  |  |  |   ot->description = "Save the current file in the desired location"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ot->invoke = wm_save_as_mainfile_invoke; | 
					
						
							|  |  |  |   ot->exec = wm_save_as_mainfile_exec; | 
					
						
							| 
									
										
										
										
											2020-11-22 19:21:30 +01:00
										 |  |  |   ot->get_name = wm_save_as_mainfile_get_name; | 
					
						
							|  |  |  |   ot->get_description = wm_save_as_mainfile_get_description; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->check = blend_save_check; | 
					
						
							|  |  |  |   /* omit window poll so this can work in background mode */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WM_operator_properties_filesel(ot, | 
					
						
							|  |  |  |                                  FILE_TYPE_FOLDER | FILE_TYPE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_SAVE, | 
					
						
							|  |  |  |                                  WM_FILESEL_FILEPATH, | 
					
						
							|  |  |  |                                  FILE_DEFAULTDISPLAY, | 
					
						
							| 
									
										
										
										
											2020-11-02 23:55:59 +01:00
										 |  |  |                                  FILE_SORT_DEFAULT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_def_boolean(ot->srna, "compress", false, "Compress", "Write compressed .blend file"); | 
					
						
							|  |  |  |   RNA_def_boolean(ot->srna, | 
					
						
							|  |  |  |                   "relative_remap", | 
					
						
							|  |  |  |                   true, | 
					
						
							|  |  |  |                   "Remap Relative", | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |                   "Remap relative paths when saving to a different directory"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   prop = RNA_def_boolean( | 
					
						
							|  |  |  |       ot->srna, | 
					
						
							|  |  |  |       "copy", | 
					
						
							|  |  |  |       false, | 
					
						
							|  |  |  |       "Save Copy", | 
					
						
							|  |  |  |       "Save a copy of the actual working state but does not make saved file active"); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_SKIP_SAVE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wm_save_mainfile_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int ret; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* cancel if no active window */ | 
					
						
							|  |  |  |   if (CTX_wm_window(C) == NULL) { | 
					
						
							|  |  |  |     return OPERATOR_CANCELLED; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   save_set_compress(op); | 
					
						
							|  |  |  |   save_set_filepath(C, op); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 10:02:28 +02:00
										 |  |  |   /* if we're saving for the first time and prefer relative paths -
 | 
					
						
							|  |  |  |    * any existing paths will be absolute, | 
					
						
							|  |  |  |    * enable the option to remap paths to avoid confusion T37240. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((G.relbase_valid == false) && (U.flag & USER_RELPATHS)) { | 
					
						
							|  |  |  |     PropertyRNA *prop = RNA_struct_find_property(op->ptr, "relative_remap"); | 
					
						
							|  |  |  |     if (!RNA_property_is_set(op->ptr, prop)) { | 
					
						
							|  |  |  |       RNA_property_boolean_set(op->ptr, prop, true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (G.save_over) { | 
					
						
							|  |  |  |     char path[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     RNA_string_get(op->ptr, "filepath", path); | 
					
						
							|  |  |  |     ret = wm_save_as_mainfile_exec(C, op); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     WM_event_add_fileselect(C, op); | 
					
						
							|  |  |  |     ret = OPERATOR_RUNNING_MODAL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WM_OT_save_mainfile(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ot->name = "Save Blender File"; | 
					
						
							|  |  |  |   ot->idname = "WM_OT_save_mainfile"; | 
					
						
							|  |  |  |   ot->description = "Save the current Blender file"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ot->invoke = wm_save_mainfile_invoke; | 
					
						
							|  |  |  |   ot->exec = wm_save_as_mainfile_exec; | 
					
						
							|  |  |  |   ot->check = blend_save_check; | 
					
						
							|  |  |  |   /* omit window poll so this can work in background mode */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   PropertyRNA *prop; | 
					
						
							|  |  |  |   WM_operator_properties_filesel(ot, | 
					
						
							|  |  |  |                                  FILE_TYPE_FOLDER | FILE_TYPE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_BLENDER, | 
					
						
							|  |  |  |                                  FILE_SAVE, | 
					
						
							|  |  |  |                                  WM_FILESEL_FILEPATH, | 
					
						
							|  |  |  |                                  FILE_DEFAULTDISPLAY, | 
					
						
							| 
									
										
										
										
											2020-11-02 23:55:59 +01:00
										 |  |  |                                  FILE_SORT_DEFAULT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   RNA_def_boolean(ot->srna, "compress", false, "Compress", "Write compressed .blend file"); | 
					
						
							|  |  |  |   RNA_def_boolean(ot->srna, | 
					
						
							|  |  |  |                   "relative_remap", | 
					
						
							|  |  |  |                   false, | 
					
						
							|  |  |  |                   "Remap Relative", | 
					
						
							| 
									
										
										
										
											2020-02-18 22:03:07 +11:00
										 |  |  |                   "Remap relative paths when saving to a different directory"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   prop = RNA_def_boolean(ot->srna, "exit", false, "Exit", "Exit Blender after saving"); | 
					
						
							|  |  |  |   RNA_def_property_flag(prop, PROP_HIDDEN | PROP_SKIP_SAVE); | 
					
						
							| 
									
										
										
										
											2016-04-19 17:08:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-02 12:25:39 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \name Auto Script Execution Warning Dialog
 | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_block_autorun_warning_ignore(bContext *C, void *arg_block, void *UNUSED(arg)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Free the data as it's no longer needed. */ | 
					
						
							|  |  |  |   wm_test_autorun_revert_action_set(NULL, NULL); | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  | static void wm_block_autorun_warning_reload_with_scripts(bContext *C, | 
					
						
							|  |  |  |                                                          void *arg_block, | 
					
						
							|  |  |  |                                                          void *UNUSED(arg)) | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Save user preferences for permanent execution. */ | 
					
						
							|  |  |  |   if ((U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0) { | 
					
						
							|  |  |  |     WM_operator_name_call(C, "WM_OT_save_userpref", WM_OP_EXEC_DEFAULT, NULL); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-02 17:24:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  |   /* Load file again with scripts enabled.
 | 
					
						
							|  |  |  |    * The reload is necessary to allow scripts to run when the files loads. */ | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  |   wm_test_autorun_revert_action_exec(C); | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  | static void wm_block_autorun_warning_enable_scripts(bContext *C, | 
					
						
							|  |  |  |                                                     void *arg_block, | 
					
						
							|  |  |  |                                                     void *UNUSED(arg)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Save user preferences for permanent execution. */ | 
					
						
							|  |  |  |   if ((U.flag & USER_SCRIPT_AUTOEXEC_DISABLE) == 0) { | 
					
						
							|  |  |  |     WM_operator_name_call(C, "WM_OT_save_userpref", WM_OP_EXEC_DEFAULT, NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Force a full refresh, but without reloading the file. */ | 
					
						
							|  |  |  |   LISTBASE_FOREACH (Scene *, scene, &bmain->scenes) { | 
					
						
							|  |  |  |     BKE_scene_free_depsgraph_hash(scene); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | /* Build the autorun warning dialog UI */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static uiBlock *block_create_autorun_warning(struct bContext *C, | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |                                              struct ARegion *region, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                              void *UNUSED(arg1)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							| 
									
										
										
										
											2020-10-18 10:31:55 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |   uiBlock *block = UI_block_begin(C, region, "autorun_warning_popup", UI_EMBOSS); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   UI_block_flag_enable( | 
					
						
							|  |  |  |       block, UI_BLOCK_KEEP_OPEN | UI_BLOCK_LOOP | UI_BLOCK_NO_WIN_CLIP | UI_BLOCK_NUMSELECT); | 
					
						
							|  |  |  |   UI_block_theme_style_set(block, UI_BLOCK_THEME_STYLE_POPUP); | 
					
						
							|  |  |  |   UI_block_emboss_set(block, UI_EMBOSS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-13 12:39:28 -08:00
										 |  |  |   uiLayout *layout = uiItemsAlertBox(block, 44, ALERT_ICON_ERROR); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-13 12:39:28 -08:00
										 |  |  |   /* Title and explanation text. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uiLayout *col = uiLayoutColumn(layout, true); | 
					
						
							| 
									
										
										
										
											2020-02-28 13:51:40 +01:00
										 |  |  |   uiItemL_ex(col, | 
					
						
							|  |  |  |              TIP_("For security reasons, automatic execution of Python scripts " | 
					
						
							|  |  |  |                   "in this file was disabled:"), | 
					
						
							| 
									
										
										
										
											2020-12-13 12:39:28 -08:00
										 |  |  |              ICON_NONE, | 
					
						
							| 
									
										
										
										
											2020-02-28 13:51:40 +01:00
										 |  |  |              true, | 
					
						
							|  |  |  |              false); | 
					
						
							| 
									
										
										
										
											2020-12-13 12:39:28 -08:00
										 |  |  |   uiItemL_ex(col, G.autoexec_fail, ICON_NONE, false, true); | 
					
						
							|  |  |  |   uiItemL(col, TIP_("This may lead to unexpected behavior"), ICON_NONE); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uiItemS(layout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   PointerRNA pref_ptr; | 
					
						
							|  |  |  |   RNA_pointer_create(NULL, &RNA_PreferencesFilePaths, &U, &pref_ptr); | 
					
						
							|  |  |  |   uiItemR(layout, | 
					
						
							|  |  |  |           &pref_ptr, | 
					
						
							|  |  |  |           "use_scripts_auto_execute", | 
					
						
							|  |  |  |           0, | 
					
						
							| 
									
										
										
										
											2019-06-11 22:25:01 +02:00
										 |  |  |           TIP_("Permanently allow execution of scripts"), | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           ICON_NONE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-13 12:39:28 -08:00
										 |  |  |   uiItemS_ex(layout, 3.0f); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Buttons */ | 
					
						
							|  |  |  |   uiBut *but; | 
					
						
							|  |  |  |   uiLayout *split = uiLayoutSplit(layout, 0.0f, true); | 
					
						
							| 
									
										
										
										
											2019-08-23 18:41:25 +02:00
										 |  |  |   uiLayoutSetScaleY(split, 1.2f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* empty space */ | 
					
						
							|  |  |  |   col = uiLayoutColumn(split, false); | 
					
						
							|  |  |  |   uiItemS(col); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   col = uiLayoutColumn(split, false); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  |   /* Allow reload if we have a saved file.
 | 
					
						
							|  |  |  |    * Otherwise just enable scripts and reset the depsgraphs. */ | 
					
						
							|  |  |  |   if (G.relbase_valid && wm->file_saved) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     but = uiDefIconTextBut(block, | 
					
						
							|  |  |  |                            UI_BTYPE_BUT, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            ICON_NONE, | 
					
						
							|  |  |  |                            IFACE_("Allow Execution"), | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            50, | 
					
						
							|  |  |  |                            UI_UNIT_Y, | 
					
						
							|  |  |  |                            NULL, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            TIP_("Reload file with execution of Python scripts enabled")); | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  |     UI_but_func_set(but, wm_block_autorun_warning_reload_with_scripts, block, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2019-06-04 13:37:45 +02:00
										 |  |  |     but = uiDefIconTextBut(block, | 
					
						
							|  |  |  |                            UI_BTYPE_BUT, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            ICON_NONE, | 
					
						
							|  |  |  |                            IFACE_("Allow Execution"), | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            50, | 
					
						
							|  |  |  |                            UI_UNIT_Y, | 
					
						
							|  |  |  |                            NULL, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            0, | 
					
						
							|  |  |  |                            TIP_("Enable scripts")); | 
					
						
							|  |  |  |     UI_but_func_set(but, wm_block_autorun_warning_enable_scripts, block, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-08-23 18:41:25 +02:00
										 |  |  |   UI_but_drawflag_disable(but, UI_BUT_TEXT_LEFT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-23 18:41:25 +02:00
										 |  |  |   col = uiLayoutColumn(split, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   but = uiDefIconTextBut(block, | 
					
						
							|  |  |  |                          UI_BTYPE_BUT, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          ICON_NONE, | 
					
						
							|  |  |  |                          IFACE_("Ignore"), | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          50, | 
					
						
							|  |  |  |                          UI_UNIT_Y, | 
					
						
							|  |  |  |                          NULL, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          0, | 
					
						
							|  |  |  |                          TIP_("Continue using file without Python scripts")); | 
					
						
							|  |  |  |   UI_but_func_set(but, wm_block_autorun_warning_ignore, block, NULL); | 
					
						
							| 
									
										
										
										
											2019-08-23 18:41:25 +02:00
										 |  |  |   UI_but_drawflag_disable(but, UI_BUT_TEXT_LEFT); | 
					
						
							|  |  |  |   UI_but_flag_enable(but, UI_BUT_ACTIVE_DEFAULT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-23 18:41:25 +02:00
										 |  |  |   UI_block_bounds_set_centered(block, 14 * U.dpi_fac); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return block; | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 21:56:49 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Store the action needed if the user needs to reload the file with Python scripts enabled. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When left to NULL, this is simply revert. | 
					
						
							|  |  |  |  * When loading files through the recover auto-save or session, | 
					
						
							|  |  |  |  * we need to revert using other operators. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct { | 
					
						
							|  |  |  |   wmOperatorType *ot; | 
					
						
							|  |  |  |   PointerRNA *ptr; | 
					
						
							|  |  |  | } wm_test_autorun_revert_action_data = { | 
					
						
							|  |  |  |     .ot = NULL, | 
					
						
							|  |  |  |     .ptr = NULL, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_test_autorun_revert_action_set(wmOperatorType *ot, PointerRNA *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(!G.background); | 
					
						
							|  |  |  |   wm_test_autorun_revert_action_data.ot = NULL; | 
					
						
							|  |  |  |   if (wm_test_autorun_revert_action_data.ptr != NULL) { | 
					
						
							|  |  |  |     WM_operator_properties_free(wm_test_autorun_revert_action_data.ptr); | 
					
						
							|  |  |  |     MEM_freeN(wm_test_autorun_revert_action_data.ptr); | 
					
						
							|  |  |  |     wm_test_autorun_revert_action_data.ptr = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   wm_test_autorun_revert_action_data.ot = ot; | 
					
						
							|  |  |  |   wm_test_autorun_revert_action_data.ptr = ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void wm_test_autorun_revert_action_exec(bContext *C) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   wmOperatorType *ot = wm_test_autorun_revert_action_data.ot; | 
					
						
							|  |  |  |   PointerRNA *ptr = wm_test_autorun_revert_action_data.ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Use regular revert. */ | 
					
						
							|  |  |  |   if (ot == NULL) { | 
					
						
							|  |  |  |     ot = WM_operatortype_find("WM_OT_revert_mainfile", false); | 
					
						
							|  |  |  |     ptr = MEM_callocN(sizeof(PointerRNA), __func__); | 
					
						
							|  |  |  |     WM_operator_properties_create_ptr(ptr, ot); | 
					
						
							|  |  |  |     RNA_boolean_set(ptr, "use_scripts", true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Set state, so it's freed correctly */ | 
					
						
							|  |  |  |     wm_test_autorun_revert_action_set(ot, ptr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WM_operator_name_call_ptr(C, ot, WM_OP_EXEC_DEFAULT, ptr); | 
					
						
							|  |  |  |   wm_test_autorun_revert_action_set(NULL, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | void wm_test_autorun_warning(bContext *C) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Test if any auto-execution of scripts failed. */ | 
					
						
							|  |  |  |   if ((G.f & G_FLAG_SCRIPT_AUTOEXEC_FAIL) == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Only show the warning once. */ | 
					
						
							|  |  |  |   if (G.f & G_FLAG_SCRIPT_AUTOEXEC_FAIL_QUIET) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   G.f |= G_FLAG_SCRIPT_AUTOEXEC_FAIL_QUIET; | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   wmWindowManager *wm = CTX_wm_manager(C); | 
					
						
							|  |  |  |   wmWindow *win = (wm->winactive) ? wm->winactive : wm->windows.first; | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (win) { | 
					
						
							|  |  |  |     wmWindow *prevwin = CTX_wm_window(C); | 
					
						
							|  |  |  |     CTX_wm_window_set(C, win); | 
					
						
							| 
									
										
										
										
											2019-05-14 15:38:51 +02:00
										 |  |  |     UI_popup_block_invoke(C, block_create_autorun_warning, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     CTX_wm_window_set(C, prevwin); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 15:54:16 +10:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Close File Dialog
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char save_images_when_file_is_closed = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_block_file_close_cancel(bContext *C, void *arg_block, void *UNUSED(arg_data)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_block_file_close_discard(bContext *C, void *arg_block, void *arg_data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-20 16:52:49 +10:00
										 |  |  |   wmGenericCallback *callback = WM_generic_callback_steal((wmGenericCallback *)arg_data); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Close the popup before executing the callback. Otherwise
 | 
					
						
							|  |  |  |    * the popup might be closed by the callback, which will lead | 
					
						
							|  |  |  |    * to a crash. */ | 
					
						
							|  |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   callback->exec(C, callback->user_data); | 
					
						
							| 
									
										
										
										
											2019-05-20 16:52:49 +10:00
										 |  |  |   WM_generic_callback_free(callback); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void wm_block_file_close_save(bContext *C, void *arg_block, void *arg_data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   const Main *bmain = CTX_data_main(C); | 
					
						
							| 
									
										
										
										
											2019-05-20 16:52:49 +10:00
										 |  |  |   wmGenericCallback *callback = WM_generic_callback_steal((wmGenericCallback *)arg_data); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   bool execute_callback = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  |   UI_popup_block_close(C, win, arg_block); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   int modified_images_count = ED_image_save_all_modified_info(CTX_data_main(C), NULL); | 
					
						
							| 
									
										
										
										
											2019-07-23 14:32:32 +02:00
										 |  |  |   if (modified_images_count > 0 && save_images_when_file_is_closed) { | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |     if (ED_image_should_save_modified(bmain)) { | 
					
						
							| 
									
										
										
										
											2019-07-11 12:06:21 +02:00
										 |  |  |       ReportList *reports = CTX_wm_reports(C); | 
					
						
							|  |  |  |       ED_image_save_all_modified(C, reports); | 
					
						
							|  |  |  |       WM_report_banner_show(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |       execute_callback = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool file_has_been_saved_before = BKE_main_blendfile_path(bmain)[0] != '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (file_has_been_saved_before) { | 
					
						
							|  |  |  |     WM_operator_name_call(C, "WM_OT_save_mainfile", WM_OP_EXEC_DEFAULT, NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     WM_operator_name_call(C, "WM_OT_save_mainfile", WM_OP_INVOKE_DEFAULT, NULL); | 
					
						
							|  |  |  |     execute_callback = false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (execute_callback) { | 
					
						
							|  |  |  |     callback->exec(C, callback->user_data); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-20 16:52:49 +10:00
										 |  |  |   WM_generic_callback_free(callback); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  | static void wm_block_file_close_cancel_button(uiBlock *block, wmGenericCallback *post_action) | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   uiBut *but = uiDefIconTextBut( | 
					
						
							|  |  |  |       block, UI_BTYPE_BUT, 0, 0, IFACE_("Cancel"), 0, 0, 0, UI_UNIT_Y, 0, 0, 0, 0, 0, ""); | 
					
						
							|  |  |  |   UI_but_func_set(but, wm_block_file_close_cancel, block, post_action); | 
					
						
							|  |  |  |   UI_but_drawflag_disable(but, UI_BUT_TEXT_LEFT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  | static void wm_block_file_close_discard_button(uiBlock *block, wmGenericCallback *post_action) | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   uiBut *but = uiDefIconTextBut( | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  |       block, UI_BTYPE_BUT, 0, 0, IFACE_("Don't Save"), 0, 0, 0, UI_UNIT_Y, 0, 0, 0, 0, 0, ""); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  |   UI_but_func_set(but, wm_block_file_close_discard, block, post_action); | 
					
						
							|  |  |  |   UI_but_drawflag_disable(but, UI_BUT_TEXT_LEFT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  | static void wm_block_file_close_save_button(uiBlock *block, wmGenericCallback *post_action) | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   uiBut *but = uiDefIconTextBut( | 
					
						
							|  |  |  |       block, UI_BTYPE_BUT, 0, 0, IFACE_("Save"), 0, 0, 0, UI_UNIT_Y, 0, 0, 0, 0, 0, ""); | 
					
						
							|  |  |  |   UI_but_func_set(but, wm_block_file_close_save, block, post_action); | 
					
						
							|  |  |  |   UI_but_drawflag_disable(but, UI_BUT_TEXT_LEFT); | 
					
						
							|  |  |  |   UI_but_flag_enable(but, UI_BUT_ACTIVE_DEFAULT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 11:38:23 +02:00
										 |  |  | static const char *close_file_dialog_name = "file_close_popup"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  | static uiBlock *block_create__close_file_dialog(struct bContext *C, | 
					
						
							|  |  |  |                                                 struct ARegion *region, | 
					
						
							|  |  |  |                                                 void *arg1) | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  |   wmGenericCallback *post_action = (wmGenericCallback *)arg1; | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:56:42 +01:00
										 |  |  |   uiBlock *block = UI_block_begin(C, region, close_file_dialog_name, UI_EMBOSS); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   UI_block_flag_enable( | 
					
						
							|  |  |  |       block, UI_BLOCK_KEEP_OPEN | UI_BLOCK_LOOP | UI_BLOCK_NO_WIN_CLIP | UI_BLOCK_NUMSELECT); | 
					
						
							|  |  |  |   UI_block_theme_style_set(block, UI_BLOCK_THEME_STYLE_POPUP); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-06 11:17:51 -08:00
										 |  |  |   uiLayout *layout = uiItemsAlertBox(block, 34, ALERT_ICON_QUESTION); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |   /* Title. */ | 
					
						
							|  |  |  |   uiItemL_ex(layout, TIP_("Save changes before closing?"), ICON_NONE, true, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Filename. */ | 
					
						
							|  |  |  |   const char *blendfile_pathpath = BKE_main_blendfile_path(CTX_data_main(C)); | 
					
						
							|  |  |  |   char filename[FILE_MAX]; | 
					
						
							|  |  |  |   if (blendfile_pathpath[0] != '\0') { | 
					
						
							|  |  |  |     BLI_split_file_part(blendfile_pathpath, filename, sizeof(filename)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2020-10-28 10:09:31 -07:00
										 |  |  |     STRNCPY(filename, IFACE_("untitled.blend")); | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |   } | 
					
						
							|  |  |  |   uiItemL(layout, filename, ICON_NONE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Image Saving Warnings. */ | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   ReportList reports; | 
					
						
							|  |  |  |   BKE_reports_init(&reports, RPT_STORE); | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   uint modified_images_count = ED_image_save_all_modified_info(bmain, &reports); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-27 12:29:06 +02:00
										 |  |  |   LISTBASE_FOREACH (Report *, report, &reports.list) { | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |     uiLayout *row = uiLayoutColumn(layout, false); | 
					
						
							|  |  |  |     uiLayoutSetScaleY(row, 0.6f); | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiItemS(row); | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Error messages created in ED_image_save_all_modified_info() can be long,
 | 
					
						
							|  |  |  |      * but are made to separate into two parts at first colon between text and paths. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     char *message = BLI_strdupn(report->message, report->len); | 
					
						
							|  |  |  |     char *path_info = strstr(message, ": "); | 
					
						
							|  |  |  |     if (path_info) { | 
					
						
							|  |  |  |       /* Terminate message string at colon. */ | 
					
						
							|  |  |  |       path_info[1] = '\0'; | 
					
						
							|  |  |  |       /* Skip over the ": " */ | 
					
						
							|  |  |  |       path_info += 2; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     uiItemL_ex(row, message, ICON_NONE, false, true); | 
					
						
							|  |  |  |     if (path_info) { | 
					
						
							|  |  |  |       uiItemL_ex(row, path_info, ICON_NONE, false, true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     MEM_freeN(message); | 
					
						
							| 
									
										
										
										
											2019-08-27 12:29:06 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |   /* Modified Images Checkbox. */ | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   if (modified_images_count > 0) { | 
					
						
							|  |  |  |     char message[64]; | 
					
						
							| 
									
										
										
										
											2020-10-31 01:55:44 +01:00
										 |  |  |     BLI_snprintf(message, sizeof(message), "Save %u modified image(s)", modified_images_count); | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiItemS(layout); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |     uiDefButBitC(block, | 
					
						
							|  |  |  |                  UI_BTYPE_CHECKBOX, | 
					
						
							|  |  |  |                  1, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  message, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  UI_UNIT_Y, | 
					
						
							|  |  |  |                  &save_images_when_file_is_closed, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  0, | 
					
						
							|  |  |  |                  ""); | 
					
						
							| 
									
										
										
										
											2019-06-19 14:28:54 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   BKE_reports_clear(&reports); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |   uiItemS_ex(layout, modified_images_count > 0 ? 2.0f : 4.0f); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |   /* Buttons. */ | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |   const bool windows_layout = true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   const bool windows_layout = false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (windows_layout) { | 
					
						
							|  |  |  |     /* Windows standard layout. */ | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiLayout *split = uiLayoutSplit(layout, 0.0f, true); | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  |     uiLayoutSetScaleY(split, 1.2f); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |     uiLayoutColumn(split, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  |     wm_block_file_close_save_button(block, post_action); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |     uiLayoutColumn(split, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  |     wm_block_file_close_discard_button(block, post_action); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 11:05:09 -07:00
										 |  |  |     uiLayoutColumn(split, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  |     wm_block_file_close_cancel_button(block, post_action); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2020-03-19 12:28:23 +11:00
										 |  |  |     /* Non-Windows layout (macOS and Linux). */ | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiLayout *split = uiLayoutSplit(layout, 0.3f, true); | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  |     uiLayoutSetScaleY(split, 1.2f); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiLayoutColumn(split, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 14:42:34 +02:00
										 |  |  |     wm_block_file_close_discard_button(block, post_action); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 16:22:37 -07:00
										 |  |  |     uiLayout *split_right = uiLayoutSplit(split, 0.1f, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uiLayoutColumn(split_right, false); | 
					
						
							|  |  |  |     /* Empty space. */ | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 12:28:23 +11:00
										 |  |  |     uiLayoutColumn(split_right, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 14:42:34 +02:00
										 |  |  |     wm_block_file_close_cancel_button(block, post_action); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 12:28:23 +11:00
										 |  |  |     uiLayoutColumn(split_right, false); | 
					
						
							| 
									
										
										
										
											2019-05-18 12:15:59 +02:00
										 |  |  |     wm_block_file_close_save_button(block, post_action); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-23 17:53:36 +02:00
										 |  |  |   UI_block_bounds_set_centered(block, 14 * U.dpi_fac); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  |   return block; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void free_post_file_close_action(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  |   wmGenericCallback *action = (wmGenericCallback *)arg; | 
					
						
							| 
									
										
										
										
											2019-05-20 16:52:49 +10:00
										 |  |  |   WM_generic_callback_free(action); | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 14:56:49 +10:00
										 |  |  | void wm_close_file_dialog(bContext *C, wmGenericCallback *post_action) | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-06 16:22:28 +01:00
										 |  |  |   if (!UI_popup_block_name_exists(CTX_wm_screen(C), close_file_dialog_name)) { | 
					
						
							| 
									
										
										
										
											2019-05-20 11:38:23 +02:00
										 |  |  |     UI_popup_block_invoke( | 
					
						
							|  |  |  |         C, block_create__close_file_dialog, post_action, free_post_file_close_action); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     WM_generic_callback_free(post_action); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-17 17:31:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-01 18:32:37 +01:00
										 |  |  | /** \} */ |