From e26f25e163bc0791282950732c6f01c32b3acc31 Mon Sep 17 00:00:00 2001 From: Agent X <44549182+Agent-11@users.noreply.github.com> Date: Mon, 1 Jan 2024 12:03:01 -0500 Subject: [PATCH] Hopefully fix Git corruption and readd Hide and Seek --- .gitattributes | 3 - .../textures/luigi-icon.tex | Bin 3372 -> 3373 bytes mods/character-select-coop/z-dev-32.lua | Bin 1456 -> 1457 bytes mods/character-select-coop/z-dev-64.lua | Bin 1456 -> 1457 bytes mods/hide-and-seek.lua | 566 ++++++++++++++++++ mods/sex-in-minecraft/sound/funnyjoel.mp3 | Bin 26218212 -> 26218719 bytes mods/sex-in-minecraft/textures/grandma.tex | Bin 953923 -> 953935 bytes res/icon.ico | Bin 167571 -> 167575 bytes 8 files changed, 566 insertions(+), 3 deletions(-) create mode 100644 mods/hide-and-seek.lua diff --git a/.gitattributes b/.gitattributes index d66b590e..e0b19036 100644 --- a/.gitattributes +++ b/.gitattributes @@ -1,9 +1,6 @@ # Set the default behavior, in case people don't have core.autocrlf set. * text=auto -# Fix line endings -*.* text eol=lf - # List text files in case git doesn't characterize correctly *.c text *.h text diff --git a/mods/char-select-cjes-and-vl/textures/luigi-icon.tex b/mods/char-select-cjes-and-vl/textures/luigi-icon.tex index 441b1ba31ee82f7f11bc2c6e1aa8acb9c6b439a2..ccf90885e0744f1f9c1049965ba13f44b8be1e1e 100644 GIT binary patch delta 12 TcmZ1@wN`3^2qW)C5eZ%Z8Fm9I delta 10 RcmZ20wMJ@!$VO2KUH}x(104VW diff --git a/mods/character-select-coop/z-dev-32.lua b/mods/character-select-coop/z-dev-32.lua index b69a01d3396a8359264f8836d855785ef1bc15c3..37f02d16c31327ccf10c509ae5afdf87cdef3042 100644 GIT binary patch delta 18 ZcmdnMy^)(!+NU%zm_c$f??#RltN=2e1uy^r delta 17 YcmdnUy@8uU+NU%zm_c&#M$Q$i04|sXBLDyZ diff --git a/mods/character-select-coop/z-dev-64.lua b/mods/character-select-coop/z-dev-64.lua index 292b920386a35c9636454a1006016c8324dfd66a..6485acdf313705b1c29a999c1901908f185b553f 100644 GIT binary patch delta 18 ZcmdnMy^)(!+NU%zm_c$f??#RltN=2e1uy^r delta 17 YcmdnUy@8uU+NU%zm_c&#M$Q$i04|sXBLDyZ diff --git a/mods/hide-and-seek.lua b/mods/hide-and-seek.lua new file mode 100644 index 00000000..2bc16c90 --- /dev/null +++ b/mods/hide-and-seek.lua @@ -0,0 +1,566 @@ +-- name: Hide and Seek +-- incompatible: gamemode +-- description: A simple hide-and-seek gamemode for\nCo-op.\n\nThe game is split into two teams:\n\nHiders and Seekers. The goal is for all\n\Hiders to be converted into a Seeker within a certain timeframe.\n\nAll Seekers appear as a metal character.\n\nEnjoy! :D\n\nConcept by: Super Keeberghrh + +-- constants +local ROUND_STATE_WAIT = 0 +local ROUND_STATE_ACTIVE = 1 +local ROUND_STATE_SEEKERS_WIN = 2 +local ROUND_STATE_HIDERS_WIN = 3 +local ROUND_STATE_UNKNOWN_END = 4 + +-- globals +gGlobalSyncTable.roundState = ROUND_STATE_WAIT -- current round state +gGlobalSyncTable.touchTag = false +gGlobalSyncTable.hiderCaps = false +gGlobalSyncTable.seekerCaps = false +gGlobalSyncTable.banKoopaShell = true +gGlobalSyncTable.disableBLJ = true +gGlobalSyncTable.displayTimer = 0 -- the displayed timer + +-- variables +local sRoundTimer = 0 -- the server's round timer +local sRoundStartTimeout = 15 * 30 -- fifteen seconds +local sRoundEndTimeout = 3 * 60 * 30 -- three minutes +local pauseExitTimer = 0 +local canLeave = false +local sFlashingIndex = 0 +local puX = 0 +local puZ = 0 +local np = gNetworkPlayers[0] +local cannonTimer = 0 + +-- server settings +gServerSettings.bubbleDeath = 0 + +--localize functions to improve performance +local +hook_chat_command, network_player_set_description, hook_on_sync_table_change, network_is_server, +hook_event, djui_popup_create, network_get_player_text_color_string, play_sound, +play_character_sound, djui_chat_message_create, djui_hud_set_resolution, djui_hud_set_font, +djui_hud_set_color, djui_hud_render_rect, djui_hud_print_text, djui_hud_get_screen_width, djui_hud_get_screen_height, +djui_hud_measure_text, tostring, warp_to_level, warp_to_start_level, stop_cap_music, dist_between_objects, +math_floor, math_ceil, table_insert, set_camera_mode += +hook_chat_command, network_player_set_description, hook_on_sync_table_change, network_is_server, +hook_event, djui_popup_create, network_get_player_text_color_string, play_sound, +play_character_sound, djui_chat_message_create, djui_hud_set_resolution, djui_hud_set_font, +djui_hud_set_color, djui_hud_render_rect, djui_hud_print_text, djui_hud_get_screen_width, djui_hud_get_screen_height, +djui_hud_measure_text, tostring, warp_to_level, warp_to_start_level, stop_cap_music, dist_between_objects, +math.floor, math.ceil, table.insert, set_camera_mode + +local function on_or_off(value) + if value then return "enabled" end + return "disabled" +end + +local function server_update() + -- increment timer + sRoundTimer = sRoundTimer + 1 + gGlobalSyncTable.displayTimer = math_floor(sRoundTimer / 30) + + -- figure out state of the game + local hasSeeker = false + local hasHider = false + local activePlayers = {} + local connectedCount = 0 + for i = 0, (MAX_PLAYERS-1) do + if gNetworkPlayers[i].connected then + connectedCount = connectedCount + 1 + table_insert(activePlayers, gPlayerSyncTable[i]) + if gPlayerSyncTable[i].seeking then + hasSeeker = true + else + hasHider = true + end + end + end + + -- only change state if there are 2+ players + if connectedCount < 2 then + gGlobalSyncTable.roundState = ROUND_STATE_WAIT + return + elseif gGlobalSyncTable.roundState == ROUND_STATE_WAIT then + gGlobalSyncTable.roundState = ROUND_STATE_UNKNOWN_END + sRoundTimer = 0 + gGlobalSyncTable.displayTimer = 0 + end + + -- check to see if the round should end + if gGlobalSyncTable.roundState == ROUND_STATE_ACTIVE then + if not hasHider or not hasSeeker or sRoundTimer > sRoundEndTimeout then + if not hasHider then + gGlobalSyncTable.roundState = ROUND_STATE_SEEKERS_WIN + elseif sRoundTimer > sRoundEndTimeout then + gGlobalSyncTable.roundState = ROUND_STATE_HIDERS_WIN + else + gGlobalSyncTable.roundState = ROUND_STATE_UNKNOWN_END + end + sRoundTimer = 0 + gGlobalSyncTable.displayTimer = 0 + else + return + end + end + + -- start round + if sRoundTimer >= sRoundStartTimeout then + -- reset seekers + for i=0,(MAX_PLAYERS-1) do + gPlayerSyncTable[i].seeking = false + end + hasSeeker = false + + -- pick random seeker + if not hasSeeker then + local randNum = math.random(#activePlayers) + local s = activePlayers[randNum] + s.seeking = true + end + + -- set round state + gGlobalSyncTable.roundState = ROUND_STATE_ACTIVE + sRoundTimer = 0 + gGlobalSyncTable.displayTimer = 0 + + end +end + +local function update() + pauseExitTimer = pauseExitTimer + 1 + + if pauseExitTimer >= 900 and not canLeave then + canLeave = true + end + -- only allow the server to figure out the seeker + if network_is_server() then + server_update() + end +end + +local function screen_transition(trans) + -- if the local player died next to a seeker, make them a seeker + local s = gPlayerSyncTable[0] + if not s.seeking then + for i=1,(MAX_PLAYERS-1) do + if gNetworkPlayers[i].connected and gNetworkPlayers[i].currLevelNum == np.currLevelNum and + gNetworkPlayers[i].currActNum == np.currActNum and gNetworkPlayers[i].currAreaIndex == np.currAreaIndex + and gPlayerSyncTable[i].seeking then + + local m = gMarioStates[0] + local a = gMarioStates[i] + + if trans == WARP_TRANSITION_FADE_INTO_BOWSER or (m.floor.type == SURFACE_DEATH_PLANE and m.pos.y <= m.floorHeight + 2048) then + if dist_between_objects(m.marioObj, a.marioObj) <= 4000 and m.playerIndex == 0 then + s.seeking = true + end + end + end + end + end +end + +--- @param m MarioState +local function mario_update(m) + if (m.flags & MARIO_VANISH_CAP) ~= 0 then + m.flags = m.flags & ~MARIO_VANISH_CAP --Always Remove Vanish Cap + stop_cap_music() + end + + if gGlobalSyncTable.disableBLJ and m.forwardVel <= -55 then + m.forwardVel = -55 + end + + -- this code runs for all players + local s = gPlayerSyncTable[m.playerIndex] + + if m.playerIndex == 0 and m.action == ACT_IN_CANNON and m.actionState == 2 then + cannonTimer = cannonTimer + 1 + if cannonTimer >= 150 then -- 150 is 5 seconds + m.forwardVel = 100 * coss(m.faceAngle.x) + + m.vel.y = 100 * sins(m.faceAngle.x) + + m.pos.x = m.pos.x + 120 * coss(m.faceAngle.x) * sins(m.faceAngle.y) + m.pos.y = m.pos.y + 120 * sins(m.faceAngle.x) + m.pos.z = m.pos.z + 120 * coss(m.faceAngle.x) * coss(m.faceAngle.y) + + play_sound(SOUND_ACTION_FLYING_FAST, m.marioObj.header.gfx.cameraToObject) + play_sound(SOUND_OBJ_POUNDING_CANNON, m.marioObj.header.gfx.cameraToObject) + + m.marioObj.header.gfx.node.flags = m.marioObj.header.gfx.node.flags | GRAPH_RENDER_ACTIVE + set_camera_mode(m.area.camera, m.area.camera.defMode, 1) + + set_mario_action(m, ACT_SHOT_FROM_CANNON, 0) + queue_rumble_data_mario(m, 60, 70) + m.usedObj.oAction = 2 + cannonTimer = 0 + end + end + + -- remove caps + if m.playerIndex == 0 or gGlobalSyncTable.roundState ~= ROUND_STATE_ACTIVE then + if gGlobalSyncTable.seekerCaps and gPlayerSyncTable[m.playerIndex].seeking then + m.flags = m.flags & ~MARIO_WING_CAP -- remove wing cap if seeking + m.flags = m.flags & ~MARIO_METAL_CAP -- remove metal cap if seeking + stop_cap_music() + m.capTimer = 0 + elseif gGlobalSyncTable.hiderCaps and not gPlayerSyncTable[m.playerIndex].seeking then + m.flags = m.flags & ~MARIO_WING_CAP -- remove wing cap if hiding + m.flags = m.flags & ~MARIO_METAL_CAP -- remove metal cap if hiding + stop_cap_music() + m.capTimer = 0 + end + end + + -- warp to the beninging + if m.playerIndex == 0 then + if gPlayerSyncTable[m.playerIndex].seeking and gGlobalSyncTable.displayTimer == 0 and gGlobalSyncTable.roundState == ROUND_STATE_ACTIVE then + warp_to_start_level() + end + end + + -- display all seekers as metal + if s.seeking then + m.marioBodyState.modelState = m.marioBodyState.modelState | MODEL_STATE_METAL + end + + -- pu prevention + if m.pos.x >= 0 then + puX = math_floor((8192 + m.pos.x) / 65536) + else + puX = math_ceil((-8192 + m.pos.x) / 65536) + end + if m.pos.z >= 0 then + puZ = math_floor((8192 + m.pos.z) / 65536) + else + puZ = math_ceil((-8192 + m.pos.z) / 65536) + end + if puX ~= 0 or puZ ~= 0 then + s.seeking = true + warp_restart_level() + end +end + +---@param m MarioState +---@param action integer +local function before_set_mario_action(m, action) + if m.playerIndex == 0 then + if action == ACT_WAITING_FOR_DIALOG or action == ACT_READING_SIGN or action == ACT_READING_NPC_DIALOG or action == ACT_JUMBO_STAR_CUTSCENE then + return 1 + elseif action == ACT_READING_AUTOMATIC_DIALOG and get_id_from_behavior(m.interactObj.behavior) ~= id_bhvDoor and get_id_from_behavior(m.interactObj.behavior) ~= id_bhvStarDoor then + return 1 + elseif action == ACT_EXIT_LAND_SAVE_DIALOG then + set_camera_mode(m.area.camera, m.area.camera.defMode, 1) + return ACT_IDLE + end + end +end + +--- @param m MarioState +local function before_phys_step(m) + -- prevent physics from being altered when bubbled + local s = gPlayerSyncTable[m.playerIndex] + + if m.action == ACT_BUBBLED or s.seeking then return end + + -- only make seekers faster + + local hScale = 1.0 + local vScale = 1.0 + + -- make swimming seekers 5% faster + if (m.action & ACT_FLAG_SWIMMING) ~= 0 then + hScale = hScale * 1.05 + if m.action ~= ACT_WATER_PLUNGE then + vScale = vScale * 1.05 + end + end +end + +local function on_pvp_attack(attacker, victim) + -- this code runs when a player attacks another player + local sAttacker = gPlayerSyncTable[attacker.playerIndex] + local sVictim = gPlayerSyncTable[victim.playerIndex] + + -- only consider local player + if victim.playerIndex ~= 0 then + return + end + + -- make victim a seeker + if sAttacker.seeking and not sVictim.seeking then + sVictim.seeking = true + end +end + +--- @param m MarioState +local function on_player_connected(m) + -- start out as a seeker + local s = gPlayerSyncTable[m.playerIndex] + s.seeking = true + network_player_set_description(gNetworkPlayers[m.playerIndex], "seeker", 255, 64, 64, 255) +end + +local function hud_top_render() + local seconds = 0 + local text = "" + + if gGlobalSyncTable.roundState == ROUND_STATE_WAIT then + seconds = 60 + text = "waiting for players" + elseif gGlobalSyncTable.roundState == ROUND_STATE_ACTIVE then + seconds = math_floor(sRoundEndTimeout / 30 - gGlobalSyncTable.displayTimer) + if seconds < 0 then seconds = 0 end + text = "seekers have " .. seconds .. " seconds" + else + seconds = math_floor(sRoundStartTimeout / 30 - gGlobalSyncTable.displayTimer) + if seconds < 0 then seconds = 0 end + text = "next round in " .. seconds .. " seconds" + end + + local scale = 0.5 + + -- get width of screen and text + local screenWidth = djui_hud_get_screen_width() + local width = djui_hud_measure_text(text) * scale + + local x = (screenWidth - width) * 0.5 + local y = 0 + + local background = 0.0 + if seconds < 60 and gGlobalSyncTable.roundState == ROUND_STATE_ACTIVE then + background = (math.sin(sFlashingIndex * 0.1) * 0.5 + 0.5) * 1 + background = background * background + background = background * background + end + + -- render top + djui_hud_set_color(255 * background, 0, 0, 128) + djui_hud_render_rect(x - 6, y, width + 12, 16) + + djui_hud_set_color(255, 255, 255, 255) + djui_hud_print_text(text, x, y, scale) +end + +local function hud_center_render() + if gGlobalSyncTable.displayTimer > 3 then return end + + -- set text + local text = "" + if gGlobalSyncTable.roundState == ROUND_STATE_SEEKERS_WIN then + text = "Seekers Win!" + elseif gGlobalSyncTable.roundState == ROUND_STATE_HIDERS_WIN then + text = "Hiders Win!" + elseif gGlobalSyncTable.roundState == ROUND_STATE_ACTIVE then + text = "Go!" + else + return + end + + -- set scale + local scale = 1 + + -- get width of screen and text + local screenWidth = djui_hud_get_screen_width() + local screenHeight = djui_hud_get_screen_height() + local width = djui_hud_measure_text(text) * scale + local height = 32 * scale + + local x = (screenWidth - width) * 0.5 + local y = (screenHeight - height) * 0.5 + + -- render + djui_hud_set_color(0, 0, 0, 128) + djui_hud_render_rect(x - 6 * scale, y, width + 12 * scale, height) + + djui_hud_set_color(255, 255, 255, 255) + djui_hud_print_text(text, x, y, scale) +end + +local function on_hud_render() + -- render to N64 screen space, with the HUD font + djui_hud_set_resolution(RESOLUTION_N64) + djui_hud_set_font(FONT_NORMAL) + + hud_top_render() + hud_center_render() + + sFlashingIndex = sFlashingIndex + 1 +end + +local function on_touch_tag_command() + gGlobalSyncTable.touchTag = not gGlobalSyncTable.touchTag + djui_chat_message_create("Touch tag: " .. on_or_off(gGlobalSyncTable.touchTag)) + return true +end + +local function on_hider_cap_command() + gGlobalSyncTable.hiderCaps = not gGlobalSyncTable.hiderCaps + djui_chat_message_create("Hider Caps: " .. on_or_off(gGlobalSyncTable.hiderCaps)) + return true +end + +local function on_seeker_cap_command() + gGlobalSyncTable.seekerCaps = not gGlobalSyncTable.seekerCaps + djui_chat_message_create("Seeker Caps: " .. on_or_off(gGlobalSyncTable.seekerCaps)) + return true +end + +local function on_koopa_shell_command() + gGlobalSyncTable.banKoopaShell = not gGlobalSyncTable.banKoopaShell + djui_chat_message_create("Koopa Shells: " .. on_or_off(not gGlobalSyncTable.banKoopaShell)) + return true +end + +local function on_blj_command() + gGlobalSyncTable.disableBLJ = not gGlobalSyncTable.disableBLJ + djui_chat_message_create("BLJS: " .. on_or_off(not gGlobalSyncTable.disableBLJ)) + return true +end + + +local function level_init() + local s = gPlayerSyncTable[0] + + pauseExitTimer = 0 + canLeave = false + + if s.seeking then canLeave = true end +end + +local function on_pause_exit() + local s = gPlayerSyncTable[0] + + if not canLeave and not s.seeking then + djui_popup_create(tostring(math_floor(30 - pauseExitTimer / 30)).." Seconds until you can leave!", 2) + return false + end +end + +----------------------- +-- network callbacks -- +----------------------- + +local function on_round_state_changed() + local rs = gGlobalSyncTable.roundState + + if rs == ROUND_STATE_ACTIVE then + play_character_sound(gMarioStates[0], CHAR_SOUND_HERE_WE_GO) + elseif rs == ROUND_STATE_SEEKERS_WIN then + play_sound(SOUND_MENU_CLICK_CHANGE_VIEW, gMarioStates[0].marioObj.header.gfx.cameraToObject) + elseif rs == ROUND_STATE_HIDERS_WIN then + play_sound(SOUND_MENU_CLICK_CHANGE_VIEW, gMarioStates[0].marioObj.header.gfx.cameraToObject) + end +end + +local function on_seeking_changed(tag, oldVal, newVal) + local m = gMarioStates[tag] + local npT = gNetworkPlayers[tag] + + -- play sound and create popup if became a seeker + if newVal and not oldVal then + play_sound(SOUND_OBJ_BOWSER_LAUGH, m.marioObj.header.gfx.cameraToObject) + playerColor = network_get_player_text_color_string(m.playerIndex) + djui_popup_create(playerColor .. npT.name .. "\\#ffa0a0\\ is now a seeker", 2) + sRoundTimer = 32 + end + + if newVal then + network_player_set_description(npT, "seeker", 255, 64, 64, 255) + else + network_player_set_description(npT, "hider", 128, 128, 128, 255) + end +end + +local function check_touch_tag_allowed(i) + if gMarioStates[i].action ~= ACT_TELEPORT_FADE_IN and gMarioStates[i].action ~= ACT_TELEPORT_FADE_OUT and gMarioStates[i].action ~= ACT_PULLING_DOOR and gMarioStates[i].action ~= ACT_PUSHING_DOOR and gMarioStates[i].action ~= ACT_WARP_DOOR_SPAWN and gMarioStates[i].action ~= ACT_ENTERING_STAR_DOOR and gMarioStates[i].action ~= ACT_STAR_DANCE_EXIT and gMarioStates[i].action ~= ACT_STAR_DANCE_NO_EXIT and gMarioStates[i].action ~= ACT_STAR_DANCE_WATER and gMarioStates[i].action ~= ACT_PANTING and gMarioStates[i].action ~= ACT_UNINITIALIZED and gMarioStates[i].action ~= ACT_WARP_DOOR_SPAWN then + return true + end + + return false +end + +local function on_interact(m, obj, intee) + if intee == INTERACT_PLAYER then + + if not gGlobalSyncTable.touchTag then + return + end + + if m ~= gMarioStates[0] then + for i=0,(MAX_PLAYERS-1) do + if gNetworkPlayers[i].connected and gNetworkPlayers[i].currAreaSyncValid then + if gPlayerSyncTable[m.playerIndex].seeking and not gPlayerSyncTable[i].seeking and obj == gMarioStates[i].marioObj and check_touch_tag_allowed(i) then + gPlayerSyncTable[i].seeking = true + + network_player_set_description(gNetworkPlayers[i], "seeker", 255, 64, 64, 255) + end + end + end + end + end +end + +local function allow_interact(_, _, intee) + if intee == INTERACT_KOOPA_SHELL and gGlobalSyncTable.banKoopaShell then + return false + end +end + +function allow_pvp_attack(m1, m2) + local s1 = gPlayerSyncTable[m1.playerIndex] + local s2 = gPlayerSyncTable[m2.playerIndex] + if s1.seeking == s2.seeking then + return false + end + return true +end + +gLevelValues.disableActs = true + +----------- +-- hooks -- +----------- + +hook_event(HOOK_UPDATE, update) +hook_event(HOOK_ON_SCREEN_TRANSITION, screen_transition) +hook_event(HOOK_BEFORE_SET_MARIO_ACTION, before_set_mario_action) +hook_event(HOOK_MARIO_UPDATE, mario_update) +hook_event(HOOK_BEFORE_PHYS_STEP, before_phys_step) +hook_event(HOOK_ALLOW_PVP_ATTACK, allow_pvp_attack) +hook_event(HOOK_ON_PVP_ATTACK, on_pvp_attack) +hook_event(HOOK_ON_PLAYER_CONNECTED, on_player_connected) +hook_event(HOOK_ON_HUD_RENDER, on_hud_render) +hook_event(HOOK_ON_LEVEL_INIT, level_init) +hook_event(HOOK_ON_PAUSE_EXIT, on_pause_exit) -- timer +hook_event(HOOK_ON_INTERACT, on_interact) +hook_event(HOOK_ALLOW_INTERACT, allow_interact) +hook_event(HOOK_USE_ACT_SELECT, function () return false end) + +if network_is_server() then + hook_chat_command("touch-to-tag", "Turn touch tag on or off", on_touch_tag_command) + hook_chat_command("hiders-caps", "Turn caps for hiders on or off", on_hider_cap_command) + hook_chat_command("seekers-caps", "Turn caps for seekers on or off", on_seeker_cap_command) + hook_chat_command("koopa-shell", "Turn the koopa shell on or off", on_koopa_shell_command) + hook_chat_command("bljs", "Turn bljs on or off", on_blj_command) +end + +-- call functions when certain sync table values change +hook_on_sync_table_change(gGlobalSyncTable, "roundState", 0, on_round_state_changed) + +for i = 0, (MAX_PLAYERS - 1) do + gPlayerSyncTable[i].seeking = true + hook_on_sync_table_change(gPlayerSyncTable[i], "seeking", i, on_seeking_changed) + network_player_set_description(gNetworkPlayers[i], "seeker", 255, 64, 64, 255) +end + +_G.HideAndSeek = { + is_player_seeker = function (playerIndex) + return gPlayerSyncTable[playerIndex].seeking + end, + + set_player_seeker = function (playerIndex, seeking) + gPlayerSyncTable[playerIndex].seeking = seeking + end, +} diff --git a/mods/sex-in-minecraft/sound/funnyjoel.mp3 b/mods/sex-in-minecraft/sound/funnyjoel.mp3 index 7cd7737d0d9b130423cbf76096c088e68d8e4b81..e16e9fe30fb85bf782f543cebaa8b9ad8233cda8 100644 GIT binary patch delta 5557 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 X_*GYjRQuwW#h1(^2TAc)AUhv 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 diff --git a/mods/sex-in-minecraft/textures/grandma.tex b/mods/sex-in-minecraft/textures/grandma.tex index a88c5ba3d1d66fea1a97e3a2d7db73e13590ffaa..fcf3fd93be8dc5963f7b9f78ee3c9fa5c59aefc2 100644 GIT binary patch delta 156 zcmX>+$LjnXDn1G zD@+2%8F{xe&SCbp0E+osVcza@g~d4)$nLjf1!A`C{g&)=41s*@Z1(Ni*&N#WKz7(I yj_qN$I2ULD^?qaG>QexUwJ39KZ&Bv%2?MePs<$LjDLD?ujq^rFPPl-xw8K0b!d06+JQLZ955Sv|J1dN980Y5#JCar>7mOajNZ zv(91mwrCH$!n{533X5}U`&3I-AZFV>)slUVVY^W_`*x#j4(