to show properties. * One problem is that we currently have 3 different kinds of property layouts, single column, two column, and single column with text inside button, probably best to reduce this.. * Last operator panel now shows operator name in the header. * Fix extrude operator to not include transform properties anymore, since they are already there now due to macro system.
		
			
				
	
	
		
			1146 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1146 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**
 | |
|  * ***** 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,
 | |
|  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 | |
|  *
 | |
|  * The Original Code is Copyright (C) 2009 Blender Foundation.
 | |
|  * All rights reserved.
 | |
|  * 
 | |
|  * Contributor(s): Blender Foundation
 | |
|  *
 | |
|  * ***** END GPL LICENSE BLOCK *****
 | |
|  */
 | |
| 
 | |
| #include <math.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include "MEM_guardedalloc.h"
 | |
| 
 | |
| #include "DNA_action_types.h"
 | |
| #include "DNA_color_types.h"
 | |
| #include "DNA_listBase.h"
 | |
| #include "DNA_material_types.h"
 | |
| #include "DNA_object_types.h"
 | |
| #include "DNA_screen_types.h"
 | |
| #include "DNA_texture_types.h"
 | |
| #include "DNA_windowmanager_types.h"
 | |
| 
 | |
| #include "BLI_blenlib.h"
 | |
| 
 | |
| #include "BKE_colortools.h"
 | |
| #include "BKE_context.h"
 | |
| #include "BKE_idprop.h"
 | |
| #include "BKE_icons.h"
 | |
| #include "BKE_library.h"
 | |
| #include "BKE_main.h"
 | |
| #include "BKE_texture.h"
 | |
| #include "BKE_utildefines.h"
 | |
| 
 | |
| #include "RNA_access.h"
 | |
| 
 | |
| #include "UI_interface.h"
 | |
| #include "UI_resources.h"
 | |
| 
 | |
| #include "ED_screen.h"
 | |
| #include "ED_util.h"
 | |
| 
 | |
| #include "WM_api.h"
 | |
| #include "WM_types.h"
 | |
| 
 | |
| #include "interface_intern.h"
 | |
| 
 | |
| #define DEF_BUT_WIDTH 		150
 | |
| #define DEF_ICON_BUT_WIDTH	20
 | |
| #define DEF_BUT_HEIGHT		20
 | |
| 
 | |
| /*************************** RNA Utilities ******************************/
 | |
| 
 | |
| uiBut *uiDefAutoButR(uiBlock *block, PointerRNA *ptr, PropertyRNA *prop, int index, char *name, int icon, int x1, int y1, int x2, int y2)
 | |
| {
 | |
| 	uiBut *but=NULL;
 | |
| 	const char *propname= RNA_property_identifier(prop);
 | |
| 	int arraylen= RNA_property_array_length(prop);
 | |
| 
 | |
| 	switch(RNA_property_type(prop)) {
 | |
| 		case PROP_BOOLEAN: {
 | |
| 			int value, length;
 | |
| 
 | |
| 			if(arraylen && index == -1)
 | |
| 				return NULL;
 | |
| 
 | |
| 			length= RNA_property_array_length(prop);
 | |
| 
 | |
| 			if(length)
 | |
| 				value= RNA_property_boolean_get_index(ptr, prop, index);
 | |
| 			else
 | |
| 				value= RNA_property_boolean_get(ptr, prop);
 | |
| 
 | |
| 			if(icon && name && strcmp(name, "") == 0)
 | |
| 				but= uiDefIconButR(block, ICONTOG, 0, icon, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			else if(icon)
 | |
| 				but= uiDefIconTextButR(block, ICONTOG, 0, icon, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			else
 | |
| 				but= uiDefButR(block, OPTION, 0, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			break;
 | |
| 		}
 | |
| 		case PROP_INT:
 | |
| 		case PROP_FLOAT:
 | |
| 			if(arraylen && index == -1) {
 | |
| 				if(RNA_property_subtype(prop) == PROP_COLOR)
 | |
| 					but= uiDefButR(block, COL, 0, name, x1, y1, x2, y2, ptr, propname, 0, 0, 0, -1, -1, NULL);
 | |
| 			}
 | |
| 			else if(RNA_property_subtype(prop) == PROP_PERCENTAGE)
 | |
| 				but= uiDefButR(block, NUMSLI, 0, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			else
 | |
| 				but= uiDefButR(block, NUM, 0, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			break;
 | |
| 		case PROP_ENUM:
 | |
| 			but= uiDefButR(block, MENU, 0, NULL, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			break;
 | |
| 		case PROP_STRING:
 | |
| 			if(icon && name && strcmp(name, "") == 0)
 | |
| 				but= uiDefIconButR(block, TEX, 0, icon, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			else if(icon)
 | |
| 				but= uiDefIconTextButR(block, TEX, 0, icon, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			else
 | |
| 				but= uiDefButR(block, TEX, 0, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			break;
 | |
| 		case PROP_POINTER: {
 | |
| 			PointerRNA pptr;
 | |
| 			int icon;
 | |
| 
 | |
| 			pptr= RNA_property_pointer_get(ptr, prop);
 | |
| 			if(!pptr.type)
 | |
| 				pptr.type= RNA_property_pointer_type(ptr, prop);
 | |
| 			icon= RNA_struct_ui_icon(pptr.type);
 | |
| 			if(icon == ICON_DOT)
 | |
| 				icon= 0;
 | |
| 
 | |
| 			but= uiDefIconTextButR(block, IDPOIN, 0, icon, name, x1, y1, x2, y2, ptr, propname, index, 0, 0, -1, -1, NULL);
 | |
| 			break;
 | |
| 		}
 | |
| 		case PROP_COLLECTION: {
 | |
| 			char text[256];
 | |
| 			sprintf(text, "%d items", RNA_property_collection_length(ptr, prop));
 | |
| 			but= uiDefBut(block, LABEL, 0, text, x1, y1, x2, y2, NULL, 0, 0, 0, 0, NULL);
 | |
| 			uiButSetFlag(but, UI_BUT_DISABLED);
 | |
| 			break;
 | |
| 		}
 | |
| 		default:
 | |
| 			but= NULL;
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	return but;
 | |
| }
 | |
| 
 | |
| void uiDefAutoButsRNA(const bContext *C, uiLayout *layout, PointerRNA *ptr, int columns)
 | |
| {
 | |
| 	uiLayout *split, *col;
 | |
| 	char *name;
 | |
| 
 | |
| 	RNA_STRUCT_BEGIN(ptr, prop) {
 | |
| 		if(strcmp(RNA_property_identifier(prop), "rna_type") == 0)
 | |
| 			continue;
 | |
| 
 | |
| 		name= (char*)RNA_property_ui_name(prop);
 | |
| 
 | |
| 		if(columns == 1) {
 | |
| 			col= uiLayoutColumn(layout, 1);
 | |
| 			uiItemL(col, name, 0);
 | |
| 		}
 | |
| 		else if(columns == 2) {
 | |
| 			split = uiLayoutSplit(layout, 0.5f);
 | |
| 
 | |
| 			uiItemL(uiLayoutColumn(split, 0), name, 0);
 | |
| 			col= uiLayoutColumn(split, 0);
 | |
| 		}
 | |
| 		else
 | |
| 			col= NULL;
 | |
| 
 | |
| 		/* temp hack to show normal button for spin/screw */
 | |
| 		if(strcmp(name, "Axis")==0) {
 | |
| 			uiDefButR(uiLayoutGetBlock(col), BUT_NORMAL, 0, name, 0, 0, 100, 100, ptr, "axis", -1, 0, 0, -1, -1, NULL);
 | |
| 		}
 | |
| 		else uiItemFullR(col, "", 0, ptr, prop, -1, 0, 0, 0, 0);
 | |
| 	}
 | |
| 	RNA_STRUCT_END;
 | |
| }
 | |
| 
 | |
| /***************************** ID Utilities *******************************/
 | |
| /* note, C code version, will be replaced with version in interface_templates.c */
 | |
| 
 | |
| typedef struct uiIDPoinParams {
 | |
| 	uiIDPoinFunc func;
 | |
| 	ListBase *lb;
 | |
| 	ID *id;
 | |
| 	short id_code;
 | |
| 	short browsenr;
 | |
| } uiIDPoinParams;
 | |
| 
 | |
| static void idpoin_cb(bContext *C, void *arg_params, void *arg_event)
 | |
| {
 | |
| 	uiIDPoinParams *params= (uiIDPoinParams*)arg_params;
 | |
| 	ListBase *lb= params->lb;
 | |
| 	uiIDPoinFunc func= params->func;
 | |
| 	ID *id= params->id, *idtest;
 | |
| 	int nr, event= GET_INT_FROM_POINTER(arg_event);
 | |
| 
 | |
| 	if(event == UI_ID_BROWSE && params->browsenr == 32767)
 | |
| 		event= UI_ID_ADD_NEW;
 | |
| 	else if(event == UI_ID_BROWSE && params->browsenr == 32766)
 | |
| 		event= UI_ID_OPEN;
 | |
| 
 | |
| 	switch(event) {
 | |
| 		case UI_ID_RENAME:
 | |
| 			if(id) test_idbutton(id->name+2);
 | |
| 			else return;
 | |
| 			break;
 | |
| 		case UI_ID_BROWSE: {
 | |
| 			/* ID can be NULL, if nothing was assigned yet */
 | |
| 			if(lb->first==NULL) return;
 | |
| 
 | |
| 			if(params->browsenr== -2) {
 | |
| 				/* XXX implement or find a replacement (ID can be NULL!)
 | |
| 				 * activate_databrowse((ID *)G.buts->lockpoin, GS(id->name), 0, B_MESHBROWSE, ¶ms->browsenr, do_global_buttons); */
 | |
| 				return;
 | |
| 			}
 | |
| 			if(params->browsenr < 0)
 | |
| 				return;
 | |
| 
 | |
| 			for(idtest=lb->first, nr=1; idtest; idtest=idtest->next, nr++) {
 | |
| 				if(nr==params->browsenr) {
 | |
| 					if(id == idtest)
 | |
| 						return;
 | |
| 
 | |
| 					id= idtest;
 | |
| 
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 		case UI_ID_DELETE:
 | |
| 			id= NULL;
 | |
| 			break;
 | |
| 		case UI_ID_FAKE_USER:
 | |
| 			if(id) {
 | |
| 				if(id->flag & LIB_FAKEUSER) id->us++;
 | |
| 				else id->us--;
 | |
| 			}
 | |
| 			else return;
 | |
| 			break;
 | |
| 		case UI_ID_PIN:
 | |
| 			break;
 | |
| 		case UI_ID_ADD_NEW:
 | |
| 			break;
 | |
| 		case UI_ID_OPEN:
 | |
| 			break;
 | |
| 		case UI_ID_ALONE:
 | |
| 			if(!id || id->us < 1)
 | |
| 				return;
 | |
| 			break;
 | |
| 		case UI_ID_LOCAL:
 | |
| 			if(!id || id->us < 1)
 | |
| 				return;
 | |
| 			break;
 | |
| 		case UI_ID_AUTO_NAME:
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	if(func)
 | |
| 		func(C, id, event);
 | |
| }
 | |
| 
 | |
| /* ***************************** ID Search browse menu ********************** */
 | |
| 
 | |
| static void id_search_call_cb(struct bContext *C, void *arg_params, void *item)
 | |
| {
 | |
| 	uiIDPoinParams *params= (uiIDPoinParams*)arg_params;
 | |
| 
 | |
| 	if(item && params->func)
 | |
| 		params->func(C, item, UI_ID_BROWSE);
 | |
| 
 | |
| }
 | |
| 
 | |
| static void id_search_cb(const struct bContext *C, void *arg_params, char *str, uiSearchItems *items)
 | |
| {
 | |
| 	uiIDPoinParams *params= (uiIDPoinParams*)arg_params;
 | |
| 	ID *id;
 | |
| 	
 | |
| 	for(id= params->lb->first; id; id= id->next) {
 | |
| 		int iconid= 0;
 | |
| 		
 | |
| 		
 | |
| 		/* icon */
 | |
| 		switch(GS(id->name))
 | |
| 		{
 | |
| 			case ID_MA: /* fall through */
 | |
| 			case ID_TE: /* fall through */
 | |
| 			case ID_IM: /* fall through */
 | |
| 			case ID_WO: /* fall through */
 | |
| 			case ID_LA: /* fall through */
 | |
| 				iconid= BKE_icon_getid(id);
 | |
| 				break;
 | |
| 			default:
 | |
| 				break;
 | |
| 		}
 | |
| 		
 | |
| 		if(BLI_strcasestr(id->name+2, str)) {
 | |
| 			if(0==uiSearchItemAdd(items, id->name+2, id, iconid))
 | |
| 				break;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static uiBlock *id_search_menu(bContext *C, ARegion *ar, void *arg_params)
 | |
| {
 | |
| 	static char search[256];
 | |
| 	static uiIDPoinParams params;
 | |
| 	wmEvent event;
 | |
| 	wmWindow *win= CTX_wm_window(C);
 | |
| 	uiBlock *block;
 | |
| 	uiBut *but;
 | |
| 	
 | |
| 	/* clear initial search string, then all items show */
 | |
| 	search[0]= 0;
 | |
| 	/* params is malloced, can be freed by parent button */
 | |
| 	params= *((uiIDPoinParams*)arg_params);
 | |
| 	
 | |
| 	block= uiBeginBlock(C, ar, "_popup", UI_EMBOSS);
 | |
| 	uiBlockSetFlag(block, UI_BLOCK_LOOP|UI_BLOCK_REDRAW|UI_BLOCK_RET_1);
 | |
| 	
 | |
| 	/* fake button, it holds space for search items */
 | |
| 	uiDefBut(block, LABEL, 0, "", 10, 15, 150, uiSearchBoxhHeight(), NULL, 0, 0, 0, 0, NULL);
 | |
| 	
 | |
| 	but= uiDefSearchBut(block, search, 0, ICON_VIEWZOOM, 256, 10, 0, 150, 19, "");
 | |
| 	uiButSetSearchFunc(but, id_search_cb, ¶ms, id_search_call_cb, NULL);
 | |
| 	
 | |
| 	uiBoundsBlock(block, 6);
 | |
| 	uiBlockSetDirection(block, UI_DOWN);	
 | |
| 	uiEndBlock(C, block);
 | |
| 	
 | |
| 	event= *(win->eventstate);	/* XXX huh huh? make api call */
 | |
| 	event.type= EVT_BUT_OPEN;
 | |
| 	event.val= KM_PRESS;
 | |
| 	event.customdata= but;
 | |
| 	event.customdatafree= FALSE;
 | |
| 	wm_event_add(win, &event);
 | |
| 	
 | |
| 	return block;
 | |
| }
 | |
| 
 | |
| /* ****************** */
 | |
| 
 | |
| int uiDefIDPoinButs(uiBlock *block, Main *bmain, ID *parid, ID *id, int id_code, short *pin_p, int x, int y, uiIDPoinFunc func, int events)
 | |
| {
 | |
| 	uiBut *but;
 | |
| 	uiIDPoinParams *params, *dup_params;
 | |
| 	char str1[10];
 | |
| 	int len, add_addbutton=0;
 | |
| 
 | |
| 	/* setup struct that we will pass on with the buttons */
 | |
| 	params= MEM_callocN(sizeof(uiIDPoinParams), "uiIDPoinParams");
 | |
| 	params->lb= wich_libbase(bmain, id_code);
 | |
| 	params->id= id;
 | |
| 	params->id_code= id_code;
 | |
| 	params->func= func;
 | |
| 
 | |
| 	/* create buttons */
 | |
| 	uiBlockBeginAlign(block);
 | |
| 
 | |
| 	/* XXX solve?
 | |
| 	if(id && id->us>1)
 | |
| 		uiBlockSetCol(block, TH_BUT_SETTING1);
 | |
| 
 | |
| 	if((events & UI_ID_PIN) && *pin_p)
 | |
| 		uiBlockSetCol(block, TH_BUT_SETTING2);
 | |
| 	*/
 | |
| 	
 | |
| 	/* pin button */
 | |
| 	if(id && (events & UI_ID_PIN)) {
 | |
| 		but= uiDefIconButS(block, ICONTOG, (events & UI_ID_PIN), ICON_KEY_DEHLT, x, y ,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, pin_p, 0, 0, 0, 0, "Keeps this view displaying the current data regardless of what object is selected");
 | |
| 		uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_PIN));
 | |
| 		x+= DEF_ICON_BUT_WIDTH;
 | |
| 	}
 | |
| 
 | |
| 	/* browse menu */
 | |
| 	if(events & UI_ID_BROWSE) {
 | |
| 		uiDefBlockButN(block, id_search_menu, MEM_dupallocN(params), "", x, y, DEF_ICON_BUT_WIDTH, DEF_BUT_HEIGHT, "Browse ID data");
 | |
| 		x+= DEF_ICON_BUT_WIDTH;
 | |
| 	}
 | |
| 	
 | |
| 	
 | |
| 
 | |
| 	/* text button with name */
 | |
| 	if(id) {
 | |
| 		/* XXX solve?
 | |
| 		if(id->us > 1)
 | |
| 			uiBlockSetCol(block, TH_BUT_SETTING1);
 | |
| 		*/
 | |
| 		/* pinned data? 
 | |
| 		if((events & UI_ID_PIN) && *pin_p)
 | |
| 			uiBlockSetCol(block, TH_BUT_SETTING2);
 | |
| 		*/
 | |
| 		/* redalert overrides pin color 
 | |
| 		if(id->us<=0)
 | |
| 			uiBlockSetCol(block, TH_REDALERT);
 | |
| 		*/
 | |
| 		uiBlockSetButLock(block, id->lib!=0, "Can't edit external libdata");
 | |
| 		
 | |
| 		/* name button */
 | |
| 		text_idbutton(id, str1);
 | |
| 		
 | |
| 		if(GS(id->name)==ID_IP) len= 110;
 | |
| 		else if((y) && (GS(id->name)==ID_AC)) len= 100; // comes from button panel (poselib)
 | |
| 		else if(y) len= 140;	// comes from button panel
 | |
| 		else len= 120;
 | |
| 		
 | |
| 		but= uiDefBut(block, TEX, 0, str1,x, y, (short)len, DEF_BUT_HEIGHT, id->name+2, 0.0, 21.0, 0, 0, "Displays current Datablock name. Click to change.");
 | |
| 		uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_RENAME));
 | |
| 
 | |
| 		x+= len;
 | |
| 
 | |
| 		uiBlockClearButLock(block);
 | |
| 		
 | |
| 		/* lib make local button */
 | |
| 		if(id->lib) {
 | |
| 			if(id->flag & LIB_INDIRECT) uiDefIconBut(block, BUT, 0, 0 /* XXX ICON_DATALIB */,x,y,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, 0, 0, 0, 0, 0, "Indirect Library Datablock. Cannot change.");
 | |
| 			else {
 | |
| 				but= uiDefIconBut(block, BUT, 0, 0 /* XXX ICON_PARLIB */, x,y,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, 0, 0, 0, 0, 0, 
 | |
| 							  (events & UI_ID_LOCAL)? "Direct linked Library Datablock. Click to make local.": "Direct linked Library Datablock, cannot make local.");
 | |
| 				uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_ALONE));
 | |
| 			}
 | |
| 			
 | |
| 			x+= DEF_ICON_BUT_WIDTH;
 | |
| 		}
 | |
| 		
 | |
| 		/* number of users / make local button */
 | |
| 		if((events & UI_ID_ALONE) && id->us>1) {
 | |
| 			int butwidth;
 | |
| 
 | |
| 			uiBlockSetButLock(block, (events & UI_ID_PIN) && *pin_p, "Can't make pinned data single-user");
 | |
| 			
 | |
| 			sprintf(str1, "%d", id->us);
 | |
| 			butwidth= (id->us<10)? DEF_ICON_BUT_WIDTH: DEF_ICON_BUT_WIDTH+10;
 | |
| 
 | |
| 			but= uiDefBut(block, BUT, 0, str1, x, y, butwidth, DEF_BUT_HEIGHT, 0, 0, 0, 0, 0, "Displays number of users of this data. Click to make a single-user copy.");
 | |
| 			uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_ALONE));
 | |
| 			x+= butwidth;
 | |
| 			
 | |
| 			uiBlockClearButLock(block);
 | |
| 		}
 | |
| 		
 | |
| 		/* delete button */
 | |
| 		if(events & UI_ID_DELETE) {
 | |
| 			uiBlockSetButLock(block, (events & UI_ID_PIN) && *pin_p, "Can't unlink pinned data");
 | |
| 			if(parid && parid->lib);
 | |
| 			else {
 | |
| 				but= uiDefIconBut(block, BUT, 0, ICON_X, x,y,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, 0, 0, 0, 0, 0, "Deletes link to this Datablock");
 | |
| 				uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_DELETE));
 | |
| 				x+= DEF_ICON_BUT_WIDTH;
 | |
| 			}
 | |
| 			
 | |
| 			uiBlockClearButLock(block);
 | |
| 		}
 | |
| 
 | |
| 		/* auto name button */
 | |
| 		if(events & UI_ID_AUTO_NAME) {
 | |
| 			if(parid && parid->lib);
 | |
| 			else {
 | |
| 				but= uiDefIconBut(block, BUT, 0, ICON_AUTO,x,y,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, 0, 0, 0, 0, 0, "Generates an automatic name");
 | |
| 				uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_AUTO_NAME));
 | |
| 				x+= DEF_ICON_BUT_WIDTH;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/* fake user button */
 | |
| 		if(events & UI_ID_FAKE_USER) {
 | |
| 			but= uiDefButBitS(block, TOG, LIB_FAKEUSER, 0, "F", x,y,DEF_ICON_BUT_WIDTH,DEF_BUT_HEIGHT, &id->flag, 0, 0, 0, 0, "Saves this datablock even if it has no users");
 | |
| 			uiButSetNFunc(but, idpoin_cb, MEM_dupallocN(params), SET_INT_IN_POINTER(UI_ID_FAKE_USER));
 | |
| 			x+= DEF_ICON_BUT_WIDTH;
 | |
| 		}
 | |
| 	}
 | |
| 	/* add new button */
 | |
| 	else if(add_addbutton) {
 | |
| 		if(parid) uiBlockSetButLock(block, parid->lib!=0, "Can't edit external libdata");
 | |
| 		dup_params= MEM_dupallocN(params);
 | |
| 		but= uiDefButS(block, TOG, 0, "Add New", x, y, 110, DEF_BUT_HEIGHT, &dup_params->browsenr, params->browsenr, 32767.0, 0, 0, "Add new data block");
 | |
| 		uiButSetNFunc(but, idpoin_cb, dup_params, SET_INT_IN_POINTER(UI_ID_ADD_NEW));
 | |
| 		x+= 110;
 | |
| 	}
 | |
| 	
 | |
| 	uiBlockEndAlign(block);
 | |
| 
 | |
| 	MEM_freeN(params);
 | |
| 
 | |
| 	return x;
 | |
| }
 | |
| 
 | |
| /* ****************************** default button callbacks ******************* */
 | |
| /* ************ LEGACY WARNING, only to get things work with 2.48 code! ****** */
 | |
| 
 | |
| void test_idbutton_cb(struct bContext *C, void *namev, void *arg2)
 | |
| {
 | |
| 	char *name= namev;
 | |
| 	
 | |
| 	test_idbutton(name+2);
 | |
| }
 | |
| 
 | |
| 
 | |
| void test_scriptpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->text.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_actionpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->action.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			id_us_plus(id);
 | |
| 			*idpp= id;
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| void test_obpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| // XXX	if(idpp == (ID **)&(emptytex.object)) {
 | |
| //		error("You must add a texture first");
 | |
| //		*idpp= 0;
 | |
| //		return;
 | |
| //	}
 | |
| 	
 | |
| 	id= CTX_data_main(C)->object.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_lib_extern(id);	/* checks lib data, sets correct flag for saving then */
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| /* tests for an object of type OB_MESH */
 | |
| void test_meshobpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 
 | |
| 	id = CTX_data_main(C)->object.first;
 | |
| 	while(id) {
 | |
| 		Object *ob = (Object *)id;
 | |
| 		if(ob->type == OB_MESH && strcmp(name, id->name + 2) == 0) {
 | |
| 			*idpp = id;
 | |
| 			/* checks lib data, sets correct flag for saving then */
 | |
| 			id_lib_extern(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id = id->next;
 | |
| 	}
 | |
| 	*idpp = NULL;
 | |
| }
 | |
| 
 | |
| void test_meshpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->mesh.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_matpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->mat.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_scenepoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->scene.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_grouppoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->group.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_texpoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->tex.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| void test_imapoin_but(struct bContext *C, char *name, ID **idpp)
 | |
| {
 | |
| 	ID *id;
 | |
| 	
 | |
| 	if( *idpp ) (*idpp)->us--;
 | |
| 	
 | |
| 	id= CTX_data_main(C)->image.first;
 | |
| 	while(id) {
 | |
| 		if( strcmp(name, id->name+2)==0 ) {
 | |
| 			*idpp= id;
 | |
| 			id_us_plus(id);
 | |
| 			return;
 | |
| 		}
 | |
| 		id= id->next;
 | |
| 	}
 | |
| 	*idpp= NULL;
 | |
| }
 | |
| 
 | |
| /* autocomplete callback for  buttons */
 | |
| void autocomplete_bone(struct bContext *C, char *str, void *arg_v)
 | |
| {
 | |
| 	Object *ob= (Object *)arg_v;
 | |
| 	
 | |
| 	if(ob==NULL || ob->pose==NULL) return;
 | |
| 	
 | |
| 	/* search if str matches the beginning of name */
 | |
| 	if(str[0]) {
 | |
| 		AutoComplete *autocpl= autocomplete_begin(str, 32);
 | |
| 		bPoseChannel *pchan;
 | |
| 		
 | |
| 		for(pchan= ob->pose->chanbase.first; pchan; pchan= pchan->next)
 | |
| 			autocomplete_do_name(autocpl, pchan->name);
 | |
| 		
 | |
| 		autocomplete_end(autocpl, str);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* autocomplete callback for buttons */
 | |
| void autocomplete_vgroup(struct bContext *C, char *str, void *arg_v)
 | |
| {
 | |
| 	Object *ob= (Object *)arg_v;
 | |
| 	
 | |
| 	if(ob==NULL) return;
 | |
| 	
 | |
| 	/* search if str matches the beginning of a name */
 | |
| 	if(str[0]) {
 | |
| 		AutoComplete *autocpl= autocomplete_begin(str, 32);
 | |
| 		bDeformGroup *dg;
 | |
| 		
 | |
| 		for(dg= ob->defbase.first; dg; dg= dg->next)
 | |
| 			if(dg->name!=str)
 | |
| 				autocomplete_do_name(autocpl, dg->name);
 | |
| 		
 | |
| 		autocomplete_end(autocpl, str);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| /* ----------- custom button group ---------------------- */
 | |
| 
 | |
| static void curvemap_buttons_zoom_in(bContext *C, void *cumap_v, void *unused)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	float d;
 | |
| 	
 | |
| 	/* we allow 20 times zoom */
 | |
| 	if( (cumap->curr.xmax - cumap->curr.xmin) > 0.04f*(cumap->clipr.xmax - cumap->clipr.xmin) ) {
 | |
| 		d= 0.1154f*(cumap->curr.xmax - cumap->curr.xmin);
 | |
| 		cumap->curr.xmin+= d;
 | |
| 		cumap->curr.xmax-= d;
 | |
| 		d= 0.1154f*(cumap->curr.ymax - cumap->curr.ymin);
 | |
| 		cumap->curr.ymin+= d;
 | |
| 		cumap->curr.ymax-= d;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void curvemap_buttons_zoom_out(bContext *C, void *cumap_v, void *unused)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	float d, d1;
 | |
| 	
 | |
| 	/* we allow 20 times zoom, but dont view outside clip */
 | |
| 	if( (cumap->curr.xmax - cumap->curr.xmin) < 20.0f*(cumap->clipr.xmax - cumap->clipr.xmin) ) {
 | |
| 		d= d1= 0.15f*(cumap->curr.xmax - cumap->curr.xmin);
 | |
| 		
 | |
| 		if(cumap->flag & CUMA_DO_CLIP) 
 | |
| 			if(cumap->curr.xmin-d < cumap->clipr.xmin)
 | |
| 				d1= cumap->curr.xmin - cumap->clipr.xmin;
 | |
| 		cumap->curr.xmin-= d1;
 | |
| 		
 | |
| 		d1= d;
 | |
| 		if(cumap->flag & CUMA_DO_CLIP) 
 | |
| 			if(cumap->curr.xmax+d > cumap->clipr.xmax)
 | |
| 				d1= -cumap->curr.xmax + cumap->clipr.xmax;
 | |
| 		cumap->curr.xmax+= d1;
 | |
| 		
 | |
| 		d= d1= 0.15f*(cumap->curr.ymax - cumap->curr.ymin);
 | |
| 		
 | |
| 		if(cumap->flag & CUMA_DO_CLIP) 
 | |
| 			if(cumap->curr.ymin-d < cumap->clipr.ymin)
 | |
| 				d1= cumap->curr.ymin - cumap->clipr.ymin;
 | |
| 		cumap->curr.ymin-= d1;
 | |
| 		
 | |
| 		d1= d;
 | |
| 		if(cumap->flag & CUMA_DO_CLIP) 
 | |
| 			if(cumap->curr.ymax+d > cumap->clipr.ymax)
 | |
| 				d1= -cumap->curr.ymax + cumap->clipr.ymax;
 | |
| 		cumap->curr.ymax+= d1;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void curvemap_buttons_setclip(bContext *C, void *cumap_v, void *unused)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	
 | |
| 	curvemapping_changed(cumap, 0);
 | |
| }	
 | |
| 
 | |
| static void curvemap_buttons_delete(bContext *C, void *cumap_v, void *unused)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	
 | |
| 	curvemap_remove(cumap->cm+cumap->cur, SELECT);
 | |
| 	curvemapping_changed(cumap, 0);
 | |
| }
 | |
| 
 | |
| /* NOTE: this is a block-menu, needs 0 events, otherwise the menu closes */
 | |
| static uiBlock *curvemap_clipping_func(struct bContext *C, struct ARegion *ar, void *cumap_v)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	uiBlock *block;
 | |
| 	uiBut *bt;
 | |
| 	
 | |
| 	block= uiBeginBlock(C, ar, "curvemap_clipping_func", UI_EMBOSS);
 | |
| 	
 | |
| 	/* use this for a fake extra empy space around the buttons */
 | |
| 	uiDefBut(block, LABEL, 0, "",			-4, 16, 128, 106, NULL, 0, 0, 0, 0, "");
 | |
| 	
 | |
| 	bt= uiDefButBitI(block, TOG, CUMA_DO_CLIP, 1, "Use Clipping",	 
 | |
| 										0,100,120,18, &cumap->flag, 0.0, 0.0, 10, 0, "");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_setclip, cumap, NULL);
 | |
| 
 | |
| 	uiBlockBeginAlign(block);
 | |
| 	uiDefButF(block, NUM, 0, "Min X ",	 0,74,120,18, &cumap->clipr.xmin, -100.0, cumap->clipr.xmax, 10, 0, "");
 | |
| 	uiDefButF(block, NUM, 0, "Min Y ",	 0,56,120,18, &cumap->clipr.ymin, -100.0, cumap->clipr.ymax, 10, 0, "");
 | |
| 	uiDefButF(block, NUM, 0, "Max X ",	 0,38,120,18, &cumap->clipr.xmax, cumap->clipr.xmin, 100.0, 10, 0, "");
 | |
| 	uiDefButF(block, NUM, 0, "Max Y ",	 0,20,120,18, &cumap->clipr.ymax, cumap->clipr.ymin, 100.0, 10, 0, "");
 | |
| 	
 | |
| 	uiBlockSetDirection(block, UI_RIGHT);
 | |
| 	
 | |
| 	uiEndBlock(C, block);
 | |
| 	return block;
 | |
| }
 | |
| 
 | |
| 
 | |
| static void curvemap_tools_dofunc(bContext *C, void *cumap_v, int event)
 | |
| {
 | |
| 	CurveMapping *cumap = cumap_v;
 | |
| 	CurveMap *cuma= cumap->cm+cumap->cur;
 | |
| 	
 | |
| 	switch(event) {
 | |
| 		case 0:
 | |
| 			curvemap_reset(cuma, &cumap->clipr);
 | |
| 			curvemapping_changed(cumap, 0);
 | |
| 			break;
 | |
| 		case 1:
 | |
| 			cumap->curr= cumap->clipr;
 | |
| 			break;
 | |
| 		case 2:	/* set vector */
 | |
| 			curvemap_sethandle(cuma, 1);
 | |
| 			curvemapping_changed(cumap, 0);
 | |
| 			break;
 | |
| 		case 3: /* set auto */
 | |
| 			curvemap_sethandle(cuma, 0);
 | |
| 			curvemapping_changed(cumap, 0);
 | |
| 			break;
 | |
| 		case 4: /* extend horiz */
 | |
| 			cuma->flag &= ~CUMA_EXTEND_EXTRAPOLATE;
 | |
| 			curvemapping_changed(cumap, 0);
 | |
| 			break;
 | |
| 		case 5: /* extend extrapolate */
 | |
| 			cuma->flag |= CUMA_EXTEND_EXTRAPOLATE;
 | |
| 			curvemapping_changed(cumap, 0);
 | |
| 			break;
 | |
| 	}
 | |
| 	ED_region_tag_redraw(CTX_wm_region(C));
 | |
| }
 | |
| 
 | |
| static uiBlock *curvemap_tools_func(struct bContext *C, struct ARegion *ar, void *cumap_v)
 | |
| {
 | |
| 	uiBlock *block;
 | |
| 	short yco= 0, menuwidth=120;
 | |
| 	
 | |
| 	block= uiBeginBlock(C, ar, "curvemap_tools_func", UI_EMBOSS);
 | |
| 	uiBlockSetButmFunc(block, curvemap_tools_dofunc, cumap_v);
 | |
| 	
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Reset View",				0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 1, "");
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Vector Handle",			0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 2, "");
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Auto Handle",			0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 3, "");
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Extend Horizontal",		0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 4, "");
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Extend Extrapolated",	0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 5, "");
 | |
| 	uiDefIconTextBut(block, BUTM, 1, ICON_BLANK1, "Reset Curve",			0, yco-=20, menuwidth, 19, NULL, 0.0, 0.0, 0, 0, "");
 | |
| 	
 | |
| 	uiBlockSetDirection(block, UI_RIGHT);
 | |
| 	uiTextBoundsBlock(block, 50);
 | |
| 	
 | |
| 	uiEndBlock(C, block);
 | |
| 	return block;
 | |
| }
 | |
| 
 | |
| /* still unsure how this call evolves... we use labeltype for defining what curve-channels to show */
 | |
| void curvemap_buttons(uiBlock *block, CurveMapping *cumap, char labeltype, short event, short redraw, rctf *rect)
 | |
| {
 | |
| 	uiBut *bt;
 | |
| 	float dx, fy= rect->ymax-18.0f;
 | |
| 	int icon;
 | |
| 	short xco, yco;
 | |
| 	
 | |
| 	yco= (short)(rect->ymax-18.0f);
 | |
| 	
 | |
| 	/* curve choice options + tools/settings, 8 icons + spacer */
 | |
| 	dx= (rect->xmax-rect->xmin)/(9.0f);
 | |
| 	
 | |
| 	uiBlockBeginAlign(block);
 | |
| 	if(labeltype=='v') {	/* vector */
 | |
| 		xco= (short)rect->xmin;
 | |
| 		if(cumap->cm[0].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "X", xco, yco+2, dx, 16, &cumap->cur, 0.0, 0.0, 0.0, 0.0, "");
 | |
| 		xco= (short)(rect->xmin+1.0f*dx);
 | |
| 		if(cumap->cm[1].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "Y", xco, yco+2, dx, 16, &cumap->cur, 0.0, 1.0, 0.0, 0.0, "");
 | |
| 		xco= (short)(rect->xmin+2.0f*dx);
 | |
| 		if(cumap->cm[2].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "Z", xco, yco+2, dx, 16, &cumap->cur, 0.0, 2.0, 0.0, 0.0, "");
 | |
| 	}
 | |
| 	else if(labeltype=='c') { /* color */
 | |
| 		xco= (short)rect->xmin;
 | |
| 		if(cumap->cm[3].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "C", xco, yco+2, dx, 16, &cumap->cur, 0.0, 3.0, 0.0, 0.0, "");
 | |
| 		xco= (short)(rect->xmin+1.0f*dx);
 | |
| 		if(cumap->cm[0].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "R", xco, yco+2, dx, 16, &cumap->cur, 0.0, 0.0, 0.0, 0.0, "");
 | |
| 		xco= (short)(rect->xmin+2.0f*dx);
 | |
| 		if(cumap->cm[1].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "G", xco, yco+2, dx, 16, &cumap->cur, 0.0, 1.0, 0.0, 0.0, "");
 | |
| 		xco= (short)(rect->xmin+3.0f*dx);
 | |
| 		if(cumap->cm[2].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "B", xco, yco+2, dx, 16, &cumap->cur, 0.0, 2.0, 0.0, 0.0, "");
 | |
| 	}
 | |
| 	/* else no channels ! */
 | |
| 	uiBlockEndAlign(block);
 | |
| 
 | |
| 	xco= (short)(rect->xmin+4.5f*dx);
 | |
| 	uiBlockSetEmboss(block, UI_EMBOSSN);
 | |
| 	bt= uiDefIconBut(block, BUT, redraw, ICON_ZOOMIN, xco, yco, dx, 14, NULL, 0.0, 0.0, 0.0, 0.0, "Zoom in");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_zoom_in, cumap, NULL);
 | |
| 	
 | |
| 	xco= (short)(rect->xmin+5.25f*dx);
 | |
| 	bt= uiDefIconBut(block, BUT, redraw, ICON_ZOOMOUT, xco, yco, dx, 14, NULL, 0.0, 0.0, 0.0, 0.0, "Zoom out");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_zoom_out, cumap, NULL);
 | |
| 	
 | |
| 	xco= (short)(rect->xmin+6.0f*dx);
 | |
| 	bt= uiDefIconBlockBut(block, curvemap_tools_func, cumap, event, ICON_MODIFIER, xco, yco, dx, 18, "Tools");
 | |
| 	
 | |
| 	xco= (short)(rect->xmin+7.0f*dx);
 | |
| 	if(cumap->flag & CUMA_DO_CLIP) icon= ICON_CLIPUV_HLT; else icon= ICON_CLIPUV_DEHLT;
 | |
| 	bt= uiDefIconBlockBut(block, curvemap_clipping_func, cumap, event, icon, xco, yco, dx, 18, "Clipping Options");
 | |
| 	
 | |
| 	xco= (short)(rect->xmin+8.0f*dx);
 | |
| 	bt= uiDefIconBut(block, BUT, event, ICON_X, xco, yco, dx, 18, NULL, 0.0, 0.0, 0.0, 0.0, "Delete points");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_delete, cumap, NULL);
 | |
| 	
 | |
| 	uiBlockSetEmboss(block, UI_EMBOSS);
 | |
| 	
 | |
| 	uiDefBut(block, BUT_CURVE, event, "", 
 | |
| 			  rect->xmin, rect->ymin, rect->xmax-rect->xmin, fy-rect->ymin, 
 | |
| 			  cumap, 0.0f, 1.0f, 0, 0, "");
 | |
| }
 | |
| 
 | |
| /* still unsure how this call evolves... we use labeltype for defining what curve-channels to show */
 | |
| void curvemap_layout(uiLayout *layout, CurveMapping *cumap, char labeltype, short event, short redraw, rctf *rect)
 | |
| {
 | |
| 	uiLayout *row;
 | |
| 	uiBlock *block;
 | |
| 	uiBut *bt;
 | |
| 	float dx, fy= rect->ymax-18.0f;
 | |
| 	int icon;
 | |
| 	
 | |
| 	block= uiLayoutGetBlock(layout);
 | |
| 	
 | |
| 	/* curve choice options + tools/settings, 8 icons + spacer */
 | |
| 	dx= UI_UNIT_X;
 | |
| 	
 | |
| 	row= uiLayoutRow(layout, 0);
 | |
| 	uiLayoutSetAlignment(row, UI_LAYOUT_ALIGN_RIGHT);
 | |
| 
 | |
| 	if(labeltype=='v') {	/* vector */
 | |
| 		row= uiLayoutRow(layout, 1);
 | |
| 
 | |
| 		if(cumap->cm[0].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "X", 0, 0, dx, 16, &cumap->cur, 0.0, 0.0, 0.0, 0.0, "");
 | |
| 		if(cumap->cm[1].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "Y", 0, 0, dx, 16, &cumap->cur, 0.0, 1.0, 0.0, 0.0, "");
 | |
| 		if(cumap->cm[2].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "Z", 0, 0, dx, 16, &cumap->cur, 0.0, 2.0, 0.0, 0.0, "");
 | |
| 	}
 | |
| 	else if(labeltype=='c') { /* color */
 | |
| 		row= uiLayoutRow(layout, 1);
 | |
| 
 | |
| 		if(cumap->cm[3].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "C", 0, 0, dx, 16, &cumap->cur, 0.0, 3.0, 0.0, 0.0, "");
 | |
| 		if(cumap->cm[0].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "R", 0, 0, dx, 16, &cumap->cur, 0.0, 0.0, 0.0, 0.0, "");
 | |
| 		if(cumap->cm[1].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "G", 0, 0, dx, 16, &cumap->cur, 0.0, 1.0, 0.0, 0.0, "");
 | |
| 		if(cumap->cm[2].curve)
 | |
| 			uiDefButI(block, ROW, redraw, "B", 0, 0, dx, 16, &cumap->cur, 0.0, 2.0, 0.0, 0.0, "");
 | |
| 	}
 | |
| 
 | |
| 	row= uiLayoutRow(row, 1);
 | |
| 
 | |
| 	uiBlockSetEmboss(block, UI_EMBOSSN);
 | |
| 	bt= uiDefIconBut(block, BUT, redraw, ICON_ZOOMIN, 0, 0, dx, 14, NULL, 0.0, 0.0, 0.0, 0.0, "Zoom in");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_zoom_in, cumap, NULL);
 | |
| 	
 | |
| 	bt= uiDefIconBut(block, BUT, redraw, ICON_ZOOMOUT, 0, 0, dx, 14, NULL, 0.0, 0.0, 0.0, 0.0, "Zoom out");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_zoom_out, cumap, NULL);
 | |
| 	
 | |
| 	bt= uiDefIconBlockBut(block, curvemap_tools_func, cumap, event, ICON_MODIFIER, 0, 0, dx, 18, "Tools");
 | |
| 	
 | |
| 	if(cumap->flag & CUMA_DO_CLIP) icon= ICON_CLIPUV_HLT; else icon= ICON_CLIPUV_DEHLT;
 | |
| 	bt= uiDefIconBlockBut(block, curvemap_clipping_func, cumap, event, icon, 0, 0, dx, 18, "Clipping Options");
 | |
| 	
 | |
| 	bt= uiDefIconBut(block, BUT, event, ICON_X, 0, 0, dx, 18, NULL, 0.0, 0.0, 0.0, 0.0, "Delete points");
 | |
| 	uiButSetFunc(bt, curvemap_buttons_delete, cumap, NULL);
 | |
| 	
 | |
| 	uiBlockSetEmboss(block, UI_EMBOSS);
 | |
| 	
 | |
| 	row= uiLayoutRow(layout, 0);
 | |
| 	uiDefBut(block, BUT_CURVE, event, "", 
 | |
| 			  rect->xmin, rect->ymin, rect->xmax-rect->xmin, fy-rect->ymin, 
 | |
| 			  cumap, 0.0f, 1.0f, 0, 0, "");
 | |
| }
 | |
| 
 | |
| 
 | |
| #define B_BANDCOL 1
 | |
| 
 | |
| static int vergcband(const void *a1, const void *a2)
 | |
| {
 | |
| 	const CBData *x1=a1, *x2=a2;
 | |
| 
 | |
| 	if( x1->pos > x2->pos ) return 1;
 | |
| 	else if( x1->pos < x2->pos) return -1;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void colorband_pos_cb(bContext *C, void *coba_v, void *unused_v)
 | |
| {
 | |
| 	ColorBand *coba= coba_v;
 | |
| 	int a;
 | |
| 	
 | |
| 	if(coba->tot<2) return;
 | |
| 	
 | |
| 	for(a=0; a<coba->tot; a++) coba->data[a].cur= a;
 | |
| 	qsort(coba->data, coba->tot, sizeof(CBData), vergcband);
 | |
| 	for(a=0; a<coba->tot; a++) {
 | |
| 		if(coba->data[a].cur==coba->cur) {
 | |
| 			// XXX if(coba->cur!=a) addqueue(curarea->win, REDRAW, 0);	/* button cur */
 | |
| 			coba->cur= a;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	WM_event_add_notifier(C, NC_TEXTURE, NULL);
 | |
| }
 | |
| 
 | |
| static void colorband_cb(bContext *C, void *coba_v, void *unused_v)
 | |
| {
 | |
| 	WM_event_add_notifier(C, NC_TEXTURE, NULL);
 | |
| }
 | |
| 
 | |
| static void colorband_add_cb(bContext *C, void *coba_v, void *unused_v)
 | |
| {
 | |
| 	ColorBand *coba= coba_v;
 | |
| 	
 | |
| 	if(coba->tot < MAXCOLORBAND-1) coba->tot++;
 | |
| 	coba->cur= coba->tot-1;
 | |
| 	
 | |
| 	colorband_pos_cb(C, coba, NULL);
 | |
| 	ED_undo_push(C, "Add colorband");
 | |
| 	WM_event_add_notifier(C, NC_TEXTURE, NULL);
 | |
| }
 | |
| 
 | |
| static void colorband_del_cb(bContext *C, void *coba_v, void *unused_v)
 | |
| {
 | |
| 	ColorBand *coba= coba_v;
 | |
| 	int a;
 | |
| 	
 | |
| 	if(coba->tot<2) return;
 | |
| 	
 | |
| 	for(a=coba->cur; a<coba->tot; a++) {
 | |
| 		coba->data[a]= coba->data[a+1];
 | |
| 	}
 | |
| 	if(coba->cur) coba->cur--;
 | |
| 	coba->tot--;
 | |
| 	
 | |
| 	ED_undo_push(C, "Delete colorband");
 | |
| 	// XXX BIF_preview_changed(ID_TE);
 | |
| 	WM_event_add_notifier(C, NC_TEXTURE, NULL);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* offset aligns from bottom, standard width 300, height 115 */
 | |
| static void colorband_buttons_large(uiBlock *block, ColorBand *coba, int xoffs, int yoffs, int redraw)
 | |
| {
 | |
| 	CBData *cbd;
 | |
| 	uiBut *bt;
 | |
| 	
 | |
| 	if(coba==NULL) return;
 | |
| 	
 | |
| 	bt= uiDefBut(block, BUT, redraw,	"Add",			0+xoffs,100+yoffs,50,20, 0, 0, 0, 0, 0, "Add a new color stop to the colorband");
 | |
| 	uiButSetFunc(bt, colorband_add_cb, coba, NULL);
 | |
| 	bt= uiDefBut(block, BUT, redraw,	"Delete",		60+xoffs,100+yoffs,50,20, 0, 0, 0, 0, 0, "Delete the active position");
 | |
| 	uiDefButS(block, NUM, redraw,		"",				120+xoffs,100+yoffs,80, 20, &coba->cur, 0.0, (float)(coba->tot-1), 0, 0, "Choose active color stop");
 | |
| 	
 | |
| 	uiButSetFunc(bt, colorband_del_cb, coba, NULL);
 | |
| 	
 | |
| 	bt= uiDefButS(block, MENU, redraw,		"Interpolation %t|Ease %x1|Cardinal %x3|Linear %x0|B-Spline %x2|Constant %x4",
 | |
| 		210+xoffs, 100+yoffs, 90, 20,		&coba->ipotype, 0.0, 0.0, 0, 0, "Set interpolation between color stops");
 | |
| 	uiButSetFunc(bt, colorband_cb, coba, NULL);
 | |
| 	uiBlockEndAlign(block);
 | |
| 
 | |
| 	bt= uiDefBut(block, BUT_COLORBAND, redraw, "", 	xoffs,65+yoffs,300,30, coba, 0, 0, 0, 0, "");
 | |
| 	uiButSetFunc(bt, colorband_cb, coba, NULL);
 | |
| 	
 | |
| 	cbd= coba->data + coba->cur;
 | |
| 	
 | |
| 	bt= uiDefButF(block, NUM, redraw, "Pos:",			0+xoffs,40+yoffs,100, 20, &cbd->pos, 0.0, 1.0, 10, 0, "The position of the active color stop");
 | |
| 	uiButSetFunc(bt, colorband_pos_cb, coba, NULL);
 | |
| 	bt= uiDefButF(block, COL, redraw,		"",				110+xoffs,40+yoffs,80,20, &(cbd->r), 0, 0, 0, B_BANDCOL, "The color value for the active color stop");
 | |
| 	uiButSetFunc(bt, colorband_cb, coba, NULL);
 | |
| 	bt= uiDefButF(block, NUMSLI, redraw,	"A ",			200+xoffs,40+yoffs,100,20, &cbd->a, 0.0, 1.0, 10, 0, "The alpha value of the active color stop");
 | |
| 	uiButSetFunc(bt, colorband_cb, coba, NULL);
 | |
| 
 | |
| }
 | |
| 
 | |
| static void colorband_buttons_small(uiBlock *block, ColorBand *coba, rctf *butr, int event)
 | |
| {
 | |
| 	CBData *cbd;
 | |
| 	uiBut *bt;
 | |
| 	float unit= (butr->xmax-butr->xmin)/14.0f;
 | |
| 	float xs= butr->xmin;
 | |
| 	
 | |
| 	cbd= coba->data + coba->cur;
 | |
| 	
 | |
| 	
 | |
| 	bt= uiDefBut(block, BUT, event,	"Add",			xs,butr->ymin+20.0f,2.0f*unit,20,	NULL, 0, 0, 0, 0, "Add a new color stop to the colorband");
 | |
| 	uiButSetFunc(bt, colorband_add_cb, coba, NULL);
 | |
| 	bt= uiDefBut(block, BUT, event,	"Delete",		xs+2.0f*unit,butr->ymin+20.0f,2.0f*unit,20,	NULL, 0, 0, 0, 0, "Delete the active position");
 | |
| 	uiButSetFunc(bt, colorband_del_cb, coba, NULL);
 | |
| 	
 | |
| 	uiDefButF(block, COL, event,		"",			xs+4.0f*unit,butr->ymin+20.0f,2.0f*unit,20,				&(cbd->r), 0, 0, 0, B_BANDCOL, "The color value for the active color stop");
 | |
| 	uiDefButF(block, NUMSLI, event,		"A:",		xs+6.0f*unit,butr->ymin+20.0f,4.0f*unit,20,	&(cbd->a), 0.0f, 1.0f, 10, 2, "The alpha value of the active color stop");
 | |
| 	
 | |
| 	uiDefButS(block, MENU, event,		"Interpolation %t|Ease %x1|Cardinal %x3|Linear %x0|B-Spline %x2|Constant %x4",
 | |
| 		xs+10.0f*unit, butr->ymin+20.0f, unit*4, 20,		&coba->ipotype, 0.0, 0.0, 0, 0, "Set interpolation between color stops");
 | |
| 
 | |
| 	uiDefBut(block, BUT_COLORBAND, event, "",		xs,butr->ymin,butr->xmax-butr->xmin,20.0f, coba, 0, 0, 0, 0, "");
 | |
| 	uiBlockEndAlign(block);
 | |
| }
 | |
| 
 | |
| void colorband_buttons(uiBlock *block, ColorBand *coba, rctf *butr, int small)
 | |
| {
 | |
| 	if(small)
 | |
| 		colorband_buttons_small(block, coba, butr, 0);
 | |
| 	else
 | |
| 		colorband_buttons_large(block, coba, 0, 0, 0);
 | |
| }
 | |
| 
 |