#include #include #include #include #include #include #include #include "../config.h" #include "controls.h" #include "gfx.h" static s32 pointer_owner = -1; WPADData *wpads[WPAD_MAX_WIIMOTES]; static int rumbling = 0; void controls_init (void) { WiiDRC_Init (); if(WiiDRC_Inited() && WiiDRC_Connected()) { //Wii VC cannot go back to this channel memset((u32 *) 0x80001800, 0, 0x1800); DCFlushRange((u32 *) 0x80001800, 0x1800); } int i; i = WPAD_Init (); if(i < 0) { gprintf("WPAD_Init failed: %d\n",i); return; } for(i=0;iexp.type) { case WPAD_EXP_NUNCHUK: if(btns & WPAD_NUNCHUK_BUTTON_Z) btns |= PADW_BUTTON_NET_INIT; if(btns & WPAD_NUNCHUK_BUTTON_C) btns |= PADW_BUTTON_SCREENSHOT; break; case WPAD_EXP_CLASSIC: if(btns & WPAD_CLASSIC_BUTTON_LEFT) btns |= WPAD_BUTTON_LEFT; if(btns & WPAD_CLASSIC_BUTTON_RIGHT) btns |= WPAD_BUTTON_RIGHT; if(btns & WPAD_CLASSIC_BUTTON_UP) btns |= WPAD_BUTTON_UP; if(btns & WPAD_CLASSIC_BUTTON_DOWN) btns |= WPAD_BUTTON_DOWN; if(btns & WPAD_CLASSIC_BUTTON_A) btns |= WPAD_BUTTON_A; if(btns & WPAD_CLASSIC_BUTTON_B) btns |= WPAD_BUTTON_B; if(btns & WPAD_CLASSIC_BUTTON_X) btns |= WPAD_BUTTON_1; if(btns & WPAD_CLASSIC_BUTTON_Y) btns |= WPAD_BUTTON_2; if((btns & WPAD_CLASSIC_BUTTON_FULL_L) || (btns & WPAD_CLASSIC_BUTTON_MINUS)) btns |= WPAD_BUTTON_MINUS; if((btns & WPAD_CLASSIC_BUTTON_FULL_R) || (btns & WPAD_CLASSIC_BUTTON_PLUS)) btns |= WPAD_BUTTON_PLUS; if(btns & WPAD_CLASSIC_BUTTON_HOME) btns |= WPAD_BUTTON_HOME; if(btns & WPAD_CLASSIC_BUTTON_ZR) btns |= PADW_BUTTON_NET_INIT; if(btns & WPAD_CLASSIC_BUTTON_ZL) btns |= PADW_BUTTON_SCREENSHOT; break; case WPAD_EXP_GUITARHERO3: if(btns & WPAD_GUITAR_HERO_3_BUTTON_STRUM_UP) btns |= WPAD_BUTTON_UP; if(btns & WPAD_GUITAR_HERO_3_BUTTON_STRUM_DOWN) btns |= WPAD_BUTTON_DOWN; if(btns & WPAD_GUITAR_HERO_3_BUTTON_GREEN) btns |= WPAD_BUTTON_A; if(btns & WPAD_GUITAR_HERO_3_BUTTON_RED) btns |= WPAD_BUTTON_B; if(btns & WPAD_GUITAR_HERO_3_BUTTON_PLUS) btns |= WPAD_BUTTON_PLUS; if(btns & WPAD_GUITAR_HERO_3_BUTTON_MINUS) btns |= WPAD_BUTTON_MINUS; if(btns & WPAD_GUITAR_HERO_3_BUTTON_YELLOW) btns |= WPAD_BUTTON_LEFT; if(btns & WPAD_GUITAR_HERO_3_BUTTON_BLUE) btns |= WPAD_BUTTON_RIGHT; if(btns & WPAD_GUITAR_HERO_3_BUTTON_ORANGE) btns |= WPAD_BUTTON_1; break; } return (btns&0xffff) << 16; } static u32 wiidrc_to_wpad(u32 btns) { u32 ret = 0; if(btns & WIIDRC_BUTTON_LEFT) ret |= WPAD_BUTTON_LEFT; if(btns & WIIDRC_BUTTON_RIGHT) ret |= WPAD_BUTTON_RIGHT; if(btns & WIIDRC_BUTTON_UP) ret |= WPAD_BUTTON_UP; if(btns & WIIDRC_BUTTON_DOWN) ret |= WPAD_BUTTON_DOWN; if(btns & WIIDRC_BUTTON_A) ret |= WPAD_BUTTON_A; if(btns & WIIDRC_BUTTON_B) ret |= WPAD_BUTTON_B; if(btns & WIIDRC_BUTTON_X) ret |= WPAD_BUTTON_1; if(btns & WIIDRC_BUTTON_Y) ret |= WPAD_BUTTON_2; if((btns & WIIDRC_BUTTON_ZL) || (btns & WIIDRC_BUTTON_MINUS)) ret |= WPAD_BUTTON_MINUS; if((btns & WIIDRC_BUTTON_ZR) || (btns & WIIDRC_BUTTON_PLUS)) ret |= WPAD_BUTTON_PLUS; if(btns & WIIDRC_BUTTON_HOME) ret |= WPAD_BUTTON_HOME; if(btns & WIIDRC_BUTTON_L) ret |= PADW_BUTTON_NET_INIT; if(btns & WIIDRC_BUTTON_R) ret |= PADW_BUTTON_SCREENSHOT; return (ret&0xffff) << 16; } void controls_scan (u32 *down, u32 *held, u32 *up) { u32 bd, bh, bu; int i; s32 last_owner; PAD_ScanPads (); bd = PAD_ButtonsDown (0); bh = PAD_ButtonsHeld (0); bu = PAD_ButtonsUp (0); WPAD_ScanPads (); for(i=0;i= 0) && (!wpads[pointer_owner] || !wpads[pointer_owner]->ir.valid)) { pointer_owner = -1; } // find a new pointer owner if necessary if(pointer_owner < 0) { for(i=0;iir.valid) { pointer_owner = i; break; } } } // pointer owner owns buttons if(pointer_owner >= 0) { bd |= wpad_button_transform(wpads[pointer_owner], wpads[pointer_owner]->btns_d); bh |= wpad_button_transform(wpads[pointer_owner], wpads[pointer_owner]->btns_h); bu |= wpad_button_transform(wpads[pointer_owner], wpads[pointer_owner]->btns_u); } else { // otherwise just mix all buttons together for(i=0;ibtns_d); bh |= wpad_button_transform(wpads[i], wpads[i]->btns_h); bu |= wpad_button_transform(wpads[i], wpads[i]->btns_u); } } } if(last_owner >= 0 && last_owner != pointer_owner && rumbling) { WPAD_Rumble (last_owner, 0); if(pointer_owner >= 0) WPAD_Rumble(pointer_owner, 1); } if(WiiDRC_Inited() && WiiDRC_Connected()) { WiiDRC_ScanPads(); bd |= wiidrc_to_wpad(WiiDRC_ButtonsDown()); bh |= wiidrc_to_wpad(WiiDRC_ButtonsHeld()); bu |= wiidrc_to_wpad(WiiDRC_ButtonsUp()); } if (down) *down = bd; if (held) *held = bh; if (up) *up = bu; } bool controls_ir (s32 *x, s32 *y, f32 *roll) { if (pointer_owner >= 0 && wpads[pointer_owner] && wpads[pointer_owner]->ir.valid) { *x = wpads[pointer_owner]->ir.x - 64; *y = wpads[pointer_owner]->ir.y - 64; if(roll) *roll = -wpads[pointer_owner]->ir.angle / 180.0 * M_PI; return true; } else { return false; } } void controls_rumble(int rumble) { if(pointer_owner >= 0) WPAD_Rumble(pointer_owner, rumble); rumbling = rumble; } static s32 wpad_sticky(int chan) { s32 sy = 0; if(wpads[chan]) { switch(wpads[chan]->exp.type) { case WPAD_EXP_NUNCHUK: sy = wpads[chan]->exp.nunchuk.js.pos.y - wpads[chan]->exp.nunchuk.js.center.y; break; case WPAD_EXP_CLASSIC: sy = (wpads[chan]->exp.classic.ljs.pos.y - wpads[chan]->exp.classic.ljs.center.y) * 4; break; case WPAD_EXP_GUITARHERO3: sy = (wpads[chan]->exp.gh3.js.pos.y - wpads[chan]->exp.gh3.js.center.y) * 4; break; } } return sy; } #define DEADZONE 10 s32 deadzone(s32 v) { if(v > DEADZONE) return v - DEADZONE; if(v < -DEADZONE) return v + DEADZONE; return 0; } s32 controls_sticky(void) { s32 sy; int i; sy = deadzone(PAD_StickY(0)); if(pointer_owner >= 0) { sy += deadzone(wpad_sticky(pointer_owner)); } else { for(i=0;i