This repository has been archived on 2023-10-09. You can view files and clone it, but cannot push or open issues or pull requests.
Files
blender-archive/source/blender/editors/sound/sound_ops.c

333 lines
8.6 KiB
C
Raw Normal View History

/**
2009-08-28 21:31:13 +00:00
* $Id$
*
* ***** BEGIN GPL LICENSE BLOCK *****
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
2010-02-12 13:34:04 +00:00
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* The Original Code is Copyright (C) 2007 Blender Foundation.
* All rights reserved.
*
*
* Contributor(s): Blender Foundation
*
* ***** END GPL LICENSE BLOCK *****
*/
2009-08-28 21:47:05 +00:00
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include "MEM_guardedalloc.h"
2009-08-28 21:47:05 +00:00
#include "DNA_packedFile_types.h"
#include "DNA_scene_types.h"
#include "DNA_space_types.h"
2009-08-28 21:47:05 +00:00
#include "DNA_sequence_types.h"
#include "DNA_userdef_types.h"
#include "BKE_context.h"
2009-08-28 21:47:05 +00:00
#include "BKE_global.h"
#include "BKE_main.h"
#include "BKE_report.h"
2009-08-28 21:47:05 +00:00
#include "BKE_packedFile.h"
#include "BKE_sound.h"
#include "BKE_utildefines.h"
2009-08-28 21:47:05 +00:00
#include "BLI_blenlib.h"
#include "RNA_access.h"
#include "RNA_define.h"
2009-08-28 21:47:05 +00:00
#include "RNA_enum_types.h"
#include "UI_interface.h"
#include "WM_api.h"
#include "WM_types.h"
#include "AUD_C-API.h"
#include "sound_intern.h"
/******************** open sound operator ********************/
static void open_init(bContext *C, wmOperator *op)
{
PropertyPointerRNA *pprop;
op->customdata= pprop= MEM_callocN(sizeof(PropertyPointerRNA), "OpenPropertyPointerRNA");
uiIDContextProperty(C, &pprop->ptr, &pprop->prop);
}
static int open_exec(bContext *C, wmOperator *op)
{
char path[FILE_MAX];
bSound *sound;
PropertyPointerRNA *pprop;
PointerRNA idptr;
AUD_SoundInfo info;
RNA_string_get(op->ptr, "path", path);
sound = sound_new_file(CTX_data_main(C), path);
if(!op->customdata)
open_init(C, op);
if (sound==NULL || sound->playback_handle == NULL) {
if(op->customdata) MEM_freeN(op->customdata);
BKE_report(op->reports, RPT_ERROR, "Unsupported audio format");
return OPERATOR_CANCELLED;
}
info = AUD_getInfo(sound->playback_handle);
if (info.specs.channels == AUD_CHANNELS_INVALID) {
sound_delete(C, sound);
if(op->customdata) MEM_freeN(op->customdata);
BKE_report(op->reports, RPT_ERROR, "Unsupported audio format");
return OPERATOR_CANCELLED;
}
if (RNA_boolean_get(op->ptr, "cache")) {
sound_cache(sound, 0);
}
/* hook into UI */
pprop= op->customdata;
if(pprop->prop) {
/* when creating new ID blocks, use is already 1, but RNA
* pointer se also increases user, so this compensates it */
sound->id.us--;
RNA_id_pointer_create(&sound->id, &idptr);
RNA_property_pointer_set(&pprop->ptr, pprop->prop, idptr);
RNA_property_update(C, &pprop->ptr, pprop->prop);
}
MEM_freeN(op->customdata);
return OPERATOR_FINISHED;
}
static int open_invoke(bContext *C, wmOperator *op, wmEvent *event)
{
if(!RNA_property_is_set(op->ptr, "relative_path"))
RNA_boolean_set(op->ptr, "relative_path", U.flag & USER_RELPATHS);
if(RNA_property_is_set(op->ptr, "path"))
return open_exec(C, op);
open_init(C, op);
return WM_operator_filesel(C, op, event);
}
void SOUND_OT_open(wmOperatorType *ot)
{
/* identifiers */
ot->name= "Open Sound";
ot->description= "Load a sound file";
ot->idname= "SOUND_OT_open";
/* api callbacks */
ot->exec= open_exec;
ot->invoke= open_invoke;
/* flags */
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
/* properties */
WM_operator_properties_filesel(ot, FOLDERFILE|SOUNDFILE|MOVIEFILE, FILE_SPECIAL, FILE_OPENFILE);
RNA_def_boolean(ot->srna, "cache", FALSE, "Cache", "Cache the sound in memory.");
RNA_def_boolean(ot->srna, "relative_path", FALSE, "Relative Path", "Load image with relative path to current .blend file");
}
/* ******************************************************* */
2009-08-28 21:47:05 +00:00
static int sound_poll(bContext *C)
{
Editing* ed = CTX_data_scene(C)->ed;
2009-08-28 22:04:37 +00:00
if(!ed || !ed->act_seq || ed->act_seq->type != SEQ_SOUND)
2009-08-28 21:47:05 +00:00
return 0;
return 1;
}
/********************* pack operator *********************/
static int pack_exec(bContext *C, wmOperator *op)
{
Editing* ed = CTX_data_scene(C)->ed;
bSound* sound;
if(!ed || !ed->act_seq || ed->act_seq->type != SEQ_SOUND)
return OPERATOR_CANCELLED;
sound = ed->act_seq->sound;
if(!sound || sound->packedfile)
return OPERATOR_CANCELLED;
sound->packedfile= newPackedFile(op->reports, sound->name);
sound_load(CTX_data_main(C), sound);
return OPERATOR_FINISHED;
}
void SOUND_OT_pack(wmOperatorType *ot)
{
/* identifiers */
ot->name= "Pack Sound";
ot->description= "Pack the sound into the current blend file";
2009-08-28 21:47:05 +00:00
ot->idname= "SOUND_OT_pack";
/* api callbacks */
ot->exec= pack_exec;
ot->poll= sound_poll;
/* flags */
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
}
/********************* unpack operator *********************/
// XXX this function is in image_ops.c too, exactly the same, should be moved to a generally accessible position
2009-08-28 21:54:41 +00:00
static void unpack_menu(bContext *C, char *opname, char *abs_name, char *folder, PackedFile *pf)
2009-08-28 21:47:05 +00:00
{
uiPopupMenu *pup;
uiLayout *layout;
char line[FILE_MAX + 100];
char local_name[FILE_MAXDIR + FILE_MAX], fi[FILE_MAX];
strcpy(local_name, abs_name);
BLI_splitdirstring(local_name, fi);
sprintf(local_name, "//%s/%s", folder, fi);
pup= uiPupMenuBegin(C, "Unpack file", 0);
layout= uiPupMenuLayout(pup);
uiItemEnumO(layout, opname, "Remove Pack", 0, "method", PF_REMOVE);
2009-08-28 21:47:05 +00:00
if(strcmp(abs_name, local_name)) {
switch(checkPackedFile(local_name, pf)) {
case PF_NOFILE:
sprintf(line, "Create %s", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_WRITE_LOCAL);
2009-08-28 21:47:05 +00:00
break;
case PF_EQUAL:
sprintf(line, "Use %s (identical)", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_USE_LOCAL);
2009-08-28 21:47:05 +00:00
break;
case PF_DIFFERS:
sprintf(line, "Use %s (differs)", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_USE_LOCAL);
2009-08-28 21:47:05 +00:00
sprintf(line, "Overwrite %s", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_WRITE_LOCAL);
2009-08-28 21:47:05 +00:00
break;
}
}
switch(checkPackedFile(abs_name, pf)) {
case PF_NOFILE:
sprintf(line, "Create %s", abs_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_WRITE_ORIGINAL);
2009-08-28 21:47:05 +00:00
break;
case PF_EQUAL:
sprintf(line, "Use %s (identical)", abs_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_USE_ORIGINAL);
2009-08-28 21:47:05 +00:00
break;
case PF_DIFFERS:
sprintf(line, "Use %s (differs)", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_USE_ORIGINAL);
2009-08-28 21:47:05 +00:00
sprintf(line, "Overwrite %s", local_name);
uiItemEnumO(layout, opname, line, 0, "method", PF_WRITE_ORIGINAL);
2009-08-28 21:47:05 +00:00
break;
}
uiPupMenuEnd(C, pup);
}
static int unpack_exec(bContext *C, wmOperator *op)
{
int method= RNA_enum_get(op->ptr, "method");
Editing* ed = CTX_data_scene(C)->ed;
bSound* sound;
if(!ed || !ed->act_seq || ed->act_seq->type != SEQ_SOUND)
return OPERATOR_CANCELLED;
sound = ed->act_seq->sound;
if(!sound || !sound->packedfile)
return OPERATOR_CANCELLED;
if(G.fileflags & G_AUTOPACK)
BKE_report(op->reports, RPT_WARNING, "AutoPack is enabled, so image will be packed again on file save.");
unpackSound(op->reports, sound, method);
return OPERATOR_FINISHED;
}
static int unpack_invoke(bContext *C, wmOperator *op, wmEvent *event)
{
Editing* ed = CTX_data_scene(C)->ed;
bSound* sound;
if(!ed || !ed->act_seq || ed->act_seq->type != SEQ_SOUND)
return OPERATOR_CANCELLED;
sound = ed->act_seq->sound;
if(!sound || !sound->packedfile)
return OPERATOR_CANCELLED;
if(G.fileflags & G_AUTOPACK)
BKE_report(op->reports, RPT_WARNING, "AutoPack is enabled, so image will be packed again on file save.");
unpack_menu(C, "SOUND_OT_unpack", sound->name, "audio", sound->packedfile);
return OPERATOR_FINISHED;
}
void SOUND_OT_unpack(wmOperatorType *ot)
{
/* identifiers */
ot->name= "Unpack Sound";
ot->description= "Unpack the sound to the samples filename";
2009-08-28 21:47:05 +00:00
ot->idname= "SOUND_OT_unpack";
/* api callbacks */
ot->exec= unpack_exec;
ot->invoke= unpack_invoke;
ot->poll= sound_poll;
/* flags */
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
/* properties */
RNA_def_enum(ot->srna, "method", unpack_method_items, PF_USE_LOCAL, "Method", "How to unpack.");
}
/* ******************************************************* */
void ED_operatortypes_sound(void)
{
WM_operatortype_append(SOUND_OT_open);
2009-08-28 21:47:05 +00:00
WM_operatortype_append(SOUND_OT_pack);
WM_operatortype_append(SOUND_OT_unpack);
}