<div class="mobile-skip-screen">
<h2>Minigame Unavailable</h2>
<p>This minigame isn't optimized for mobile devices yet.</p>
<p>Skipping to ensure the best gameplay experience.</p>
<br>
<<button "Go Back">><<goto $skipMinigameBack>><</button>>
<<if previous() is "WorkBrainFry">>
<<button "Continue">><<run MindWare.submitBrainFryWork(500, 250)>><</button>>
<<else>>
<<button "Continue">><<goto $skipMinigameContinue>><</button>>
<</if>>
</div>
<style>
.mobile-skip-screen {
font-family: var(--mindware-ui-font-primary);
color: var(--mindware-ui-text-color);
background-color: var(--mindware-ui-content-bg);
padding: 20px;
text-align: center;
border: var(--mindware-ui-pixel-size) solid var(--mindware-ui-border-color);
box-shadow: 0 0 10px var(--mindware-ui-shadow-color);
margin: 20px auto;
}
.mobile-skip-screen h2 {
font-size: 1.5em;
margin-bottom: 15px;
color: var(--mindware-ui-accent-color);
}
.mobile-skip-screen p {
font-size: 1em;
margin-bottom: 10px;
color: var(--mindware-ui-light-text);
}
.countdown {
font-size: 1.2em;
margin-top: 20px;
font-weight: bold;
color: var(--mindware-ui-secondary-accent);
}
</style><div class="mobile-skip-screen">
<h2>Games Unavailable</h2>
<p>Games aren't optimized for mobile devices yet.</p>
<p>But don't worry: they are just distractions and serve no real purpose in the game.</p>
<br>
<<button "Go Back">><<goto $playerHome>><</button>>
</div>
<style>
.mobile-skip-screen {
font-family: var(--mindware-ui-font-primary);
color: var(--mindware-ui-text-color);
background-color: var(--mindware-ui-content-bg);
padding: 20px;
text-align: center;
border: var(--mindware-ui-pixel-size) solid var(--mindware-ui-border-color);
box-shadow: 0 0 10px var(--mindware-ui-shadow-color);
margin: 20px auto;
}
.mobile-skip-screen h2 {
font-size: 1.5em;
margin-bottom: 15px;
color: var(--mindware-ui-accent-color);
}
.mobile-skip-screen p {
font-size: 1em;
margin-bottom: 10px;
color: var(--mindware-ui-light-text);
}
.countdown {
font-size: 1.2em;
margin-top: 20px;
font-weight: bold;
color: var(--mindware-ui-secondary-accent);
}
</style>/* EXAMPLE */
<!-- Button to submit to AVA -->
@@.pink-choicebutton;<<button "Submit to AVA">>
<</button>>@@
<br>
<!-- Button to resist AVA wrapped in a div with a unique ID -->
<div id="resist-ava-button-container">
@@.choicebutton;<<button "Resist AVA">>
<<run MindWare.resistAVAMinigame()>>
<</button>>@@
<br>
</div>
<!-- Hidden Content: Button to Continue (Initially Hidden) -->
<div id="AVA-resist-minigame-hidden-content" style="display: none;">
@@.choicebutton;<<button "Continue">>
<<set $genderIdentityQuestionnaireQuestion3 to "true">>
<<replace "#questions">>
<<include "PsychologicalEvaluation1-Test-Read4">>
<</replace>>
<</button>>@@
</div>
/* With settings */
<!-- Button to resist AVA wrapped in a div with a unique ID -->
<div id="resist-ava-button-container">
@@.choicebutton;<<button "Resist AVA">>
<<set $resistAVACustomSettings to true>>
<<set $resistAVATotalCircles to 7>>
<<set $resistAVACircleDuration to 3000>>
<<set $resistAVACircleMinSize to 40>>
<<set $resistAVACircleMaxSize to 80>>
<<run MindWare.resistAVAMinigame()>>
<</button>>@@
<br>
</div><style>
.container {
touch-action: none; /* Prevents default touch behaviors */
-webkit-touch-callout: none; /* Prevents callouts */
-webkit-user-select: none; /* Prevents text selection */
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.room {
touch-action: none;
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.exit-portal {
position: absolute;
top: 0 !important;
width: 100%;
height: 100vh;
background-image: url('imgs/dreams/tunnel.webp');
background-size: 100% 100%;
/* z-index: 10000; */
z-index: 100004 !important;
opacity: 0; /* Start fully transparent */
animation:
fadeInBackground 2s ease-in-out forwards,
forceRepaint 0.1s infinite;
animation-delay: 0s, 2s; /* Delay for each animation respectively */
}
@keyframes fadeInBackground {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}
/* .exit-portal {
position: absolute;
top: 0 !important;
width: 100%;
height: 100vh;
background: linear-gradient(to right,
rgba(0, 255, 0, 1),
rgba(0, 255, 0, 1),
rgba(0, 255, 0, 1)
);
z-index: 10000;
box-shadow: 0 0 50px #00ff00;
} */
.container {
max-width: 54em;
/* min-height: calc(100vh - 65px); */
min-height: 100vh;
margin: 0px auto;
perspective: 700px; /* Slightly increase the perspective for better depth */
perspective-origin: 50% 50%; /* Add this to center the perspective point */
/* ... existing styles ... */
-webkit-perspective: 700px;
-webkit-perspective-origin: 50% 50%;
transform-style: preserve-3d;
-webkit-transform-style: preserve-3d; /* Safari needs this prefix */
}
/* For screens with max-width 1200px */
@media (max-width: 1200px) {
.container {
perspective: 650px; /* Slightly reduced */
}
}
/* For screens with max-width 992px */
@media (max-width: 992px) {
.container {
perspective: 600px; /* Reduced further */
}
}
/* For screens with max-width 844px */
@media (max-width: 844px) {
.container {
perspective: 550px; /* Closer to the target */
}
}
/* For screens with max-width 600px */
@media (max-width: 600px) {
.container {
perspective: 500px; /* Smallest, final perspective */
}
}
.room-old {
position: relative;
width: 100%;
min-height: calc(100vh - 15vh);
transform-style: preserve-3d;
transform-origin: center center; /* Add this */
/* ... existing styles ... */
transform-style: preserve-3d;
-webkit-transform-style: preserve-3d;
backface-visibility: hidden;
-webkit-backface-visibility: hidden;
}
.room {
position: relative;
width: 100%;
min-height: calc(100vh - 15vh);
transform-style: preserve-3d;
-webkit-transform-style: preserve-3d;
transform-origin: center center;
-webkit-transform-origin: center center;
transform-box: border-box;
-webkit-transform-box: border-box;
}
/* For screens with max-width 1200px */
@media (max-width: 1200px) {
.room {
min-height: calc(100vh - 20vh); /* Slightly less space at top */
}
}
/* For screens with max-width 992px */
@media (max-width: 992px) {
.room {
min-height: calc(100vh - 25vh);
}
}
/* For screens with max-width 844px */
@media (max-width: 844px) {
.room {
min-height: calc(100vh - 30vh);
}
}
/* For screens with max-width 600px */
@media (max-width: 600px) {
.room {
min-height: calc(100vh - 35vh); /* Minimal space at top for smallest screens */
}
}
.transit {
transition: all 0.6s linear;
-webkit-transition: all 0.6s linear;
}
.transitfast {
transition: all 0.3s linear;
-webkit-transition: all 0.3s linear;
}
.room-wrapper {
position: relative;
width: 100%;
height: 100%;
overflow-y: auto; /* Allows vertical scrolling if content overflows */
overflow-x: hidden; /* Prevents horizontal scrolling */
margin-top: 5px;
border: 3px solid transparent;
border-image: linear-gradient(135deg, rgba(0, 175, 255, 1), rgba(255, 100, 255, 1)) 1;
background: rgba(255, 255, 255, 0.05);
/* backdrop-filter: blur(4px); */
box-shadow: 0px 0px 5px 2px rgba(0, 175, 255, 0.3), 0px 0px 8px 4px rgba(255, 100, 255, 0.3);
animation: scanlines 0.1s steps(50) infinite, borderColorShift 5s ease-in-out infinite;
/* new */
z-index: 100002;
}
.room div {
position: absolute;
top: 15px;
width: 100%;
height: 100%;
padding: 0px;
transform-origin: center center; /* Add this */
backface-visibility: hidden;
/* new styles */
-webkit-transform-origin: center center;
-webkit-backface-visibility: hidden;
z-index: 100001;
}
.clickable:hover .room-wrapper {
cursor: pointer;
background-color: rgba(150, 150, 255, 0.4); /* More intense background on hover */
border-color: #00ccff;
box-shadow: 0px 0px 20px 10px rgba(0, 175, 255, 0.8), 0px 0px 25px 12px rgba(255, 100, 255, 0.8); /* Increased glow on hover */
transition: box-shadow 0.3s ease, background-color 0.3s ease, border-color 0.3s ease; /* Smooth transition */
z-index:;
}
/* Scanline effect animation */
@keyframes scanlines {
0% { background-position: 0 0; }
100% { background-position: 0 100%; }
}
/* Border color shift animation */
@keyframes borderColorShift {
0% {
border-image: linear-gradient(135deg, rgba(0, 175, 255, 1), rgba(255, 100, 255, 1)) 1;
}
50% {
border-image: linear-gradient(135deg, rgba(125, 75, 255, 1), rgba(255, 150, 255, 1)) 1; /* Blue shifts more to purple, pink becomes brighter */
}
100% {
border-image: linear-gradient(135deg, rgba(0, 175, 255, 1), rgba(255, 100, 255, 1)) 1;
}
}
/* Optional glow pulse for a more holographic feel */
@keyframes glowPulse {
0% {
box-shadow: 0px 0px 10px 5px rgba(0, 175, 255, 0.3), 0px 0px 15px 10px rgba(255, 100, 255, 0.3); /* Reduced glow */
}
50% {
box-shadow: 0px 0px 15px 10px rgba(0, 175, 255, 0.5), 0px 0px 20px 10px rgba(255, 100, 255, 0.5); /* Reduced pulse */
}
100% {
box-shadow: 0px 0px 10px 5px rgba(0, 175, 255, 0.3), 0px 0px 15px 10px rgba(255, 100, 255, 0.3);
}
}
/* Fullscreen overlay for the dream3d passage */
.dream-fullscreen-overlay {
position: fixed;
top: 0;
left: 0;
width: 100vw; /* Full viewport width */
height: 100vh; /* Full viewport height */
background-color: black; /* Black background for the minigame overlay */
z-index: 9999; /* Bring the overlay to the top of everything */
overflow: hidden; /* Ensure the overlay content does not overflow */
cursor: pointer;
}
/* Hide scrollbars when fullscreen overlay is active */
body.no-scroll {
overflow: hidden; /* This hides the scrollbars */
}
/* Hide scrollbars for the #passages element */
#passages.no-scroll {
overflow: hidden; /* This will hide the scrollbars for the #passages element */
}
.hypno-container {
top: 0px !important;
width: 100%;
height: 100vh;
background-size: contain, auto;
background-repeat: no-repeat, repeat;
background-position: center, top left;
margin: 0;
padding: 0;
border: 0;
/* z-index: 9999; */
z-index: 100003 !important;
animation: cyberpunkColors 8s infinite ease-in-out;
position: relative;
animation: forceRepaint 0.1s infinite;
}
@keyframes forceRepaint {
0% {
opacity: 1;
}
50% {
opacity: 0.9999;
}
100% {
opacity: 1;
}
}
/* Subtle pulsing glow overlay */
.hypno-container::after {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: radial-gradient(circle at center,
transparent 0%,
rgba(185, 87, 206, 0.1) 50%,
rgba(89, 148, 206, 0.2) 100%
);
animation: glowHypnoPulse 4s infinite alternate;
mix-blend-mode: screen;
pointer-events: none;
}
@keyframes cyberpunkColors {
0%, 100% {
filter: contrast(110%) brightness(100%) hue-rotate(0deg) saturate(120%);
}
33% {
/* Deep cyberpunk blue/purple */
filter: contrast(120%) brightness(95%) hue-rotate(200deg) saturate(150%);
}
66% {
/* Intense magenta/neon */
filter: contrast(130%) brightness(105%) hue-rotate(280deg) saturate(180%);
}
}
@keyframes glowHypnoPulse {
0%, 100% {
opacity: 0.3;
}
50% {
opacity: 0.7;
}
}
.red-warning {
top: 0px !important;
width: 100vw; /* Full viewport width */
height: 100vh; /* Full viewport height */
background-color: rgba(255, 0, 0, 0.5); /* Semi-transparent red */
z-index: 9998; /* Below the hypno-container but on top of other content */
animation: red-flash 1s ease-in-out infinite; /* Flashing effect */
}
/* Flashing animation */
@keyframes red-flash {
0% {
background-color: rgba(255, 0, 0, 0.5); /* Semi-transparent red */
}
50% {
background-color: rgba(255, 0, 0, 0.1); /* Lighter red */
}
100% {
background-color: rgba(255, 0, 0, 0.5); /* Back to semi-transparent red */
}
}
</style>
<<script>>
(function() {
window.DreamManager.cleanup();
const instanceId = Date.now();
window.DreamManager.currentInstance = instanceId;
/* Configuration Variables */
const config = {
images: (State.variables.dreamConfig && State.variables.dreamConfig.images !== undefined)
? State.variables.dreamConfig.images
: [
'hypno-solo/537167042_p0.webp',
'hypno-solo/538222714_p0.webp',
'hypno-solo/607915785_p1.webp',
'hypno-solo/537167473_p0.webp',
'hypno-solo/538223465_p0.webp',
'hypno-solo/610507079_p0.webp',
'hypno-solo/537168502_p0.webp',
'hypno-solo/538745429_p0.webp',
'hypno-solo/611132916_p0.webp',
'hypno-solo/537179700_p0.webp',
'hypno-solo/538849150_p0.webp',
'hypno-solo/611741618_p0.webp',
'hypno-solo/537182511_p1.webp',
'hypno-solo/538891158_p0.webp',
'hypno-solo/611741618_p1.webp',
'hypno-solo/537187364_p1.webp',
'hypno-solo/539725855_p0.webp',
'hypno-solo/620011929_p0.webp',
'hypno-solo/537188585_p0.webp',
'hypno-solo/540121463_p0.webp',
'hypno-solo/621014644_p0.webp',
'hypno-solo/537208456_p0.webp',
'hypno-solo/549691708_p0.webp',
'hypno-solo/623036862_p0.webp',
'hypno-solo/537277648_p0.webp',
'hypno-solo/564432355_p0.webp',
'hypno-solo/623980519_p0.webp',
'hypno-solo/537796555_p0.webp',
'hypno-solo/564432355_p1.webp',
'hypno-solo/630682178_p1.webp',
'hypno-solo/537807630_p0.webp',
'hypno-solo/572987598_p0.webp',
'hypno-solo/631151091_p1.webp',
'hypno-solo/537839279_p0.webp',
'hypno-solo/574359634_p0.webp',
'hypno-solo/633341305_p0.webp',
'hypno-solo/537845640_p0.webp',
'hypno-solo/583215837_p0.webp',
'hypno-solo/639446048_p1.webp',
'hypno-solo/537860788_p0.webp',
'hypno-solo/605558520_p0.webp',
'hypno-solo/648321039_p0.webp',
'hypno-solo/537990665_p0.webp',
'hypno-solo/605716507_p0.webp',
'hypno-solo/648321039_p2.webp',
'hypno-solo/537990784_p0.webp',
'hypno-solo/606507445_p0.webp',
'hypno-solo/650633046_p0.webp',
'hypno-solo/537990986_p0.webp',
'hypno-solo/607915785_p0.webp'
],
initialDelay: (State.variables.dreamConfig && State.variables.dreamConfig.initialDelay !== undefined)
? State.variables.dreamConfig.initialDelay
: 5000,
warningDisplayTime: (State.variables.dreamConfig && State.variables.dreamConfig.warningDisplayTime !== undefined)
? State.variables.dreamConfig.warningDisplayTime
: 1500,
imageDisplayTime: (State.variables.dreamConfig && State.variables.dreamConfig.imageDisplayTime !== undefined)
? State.variables.dreamConfig.imageDisplayTime
: 3000,
numberOfSets: (State.variables.dreamConfig && State.variables.dreamConfig.numberOfSets !== undefined)
? State.variables.dreamConfig.numberOfSets
: 0,
numberOfWallsPerSet: (State.variables.dreamConfig && State.variables.dreamConfig.numberOfWallsPerSet !== undefined)
? State.variables.dreamConfig.numberOfWallsPerSet
: 2,
hypnoTimer: (State.variables.dreamConfig && State.variables.dreamConfig.hypnoTimer !== undefined)
? State.variables.dreamConfig.hypnoTimer
: 10000,
progressBarColor: (State.variables.dreamConfig && State.variables.dreamConfig.progressBarColor !== undefined)
? State.variables.dreamConfig.progressBarColor
: "#0C85D3"
};
/* const config = {
images: [
'537167042_p0.webp', '538222714_p0.webp', '607915785_p1.webp',
'537167473_p0.webp', '538223465_p0.webp', '610507079_p0.webp',
'537168502_p0.webp', '538745429_p0.webp', '611132916_p0.webp',
'537179700_p0.webp', '538849150_p0.webp', '611741618_p0.webp',
'537182511_p1.webp', '538891158_p0.webp', '611741618_p1.webp',
'537187364_p1.webp', '539725855_p0.webp', '620011929_p0.webp',
'537188585_p0.webp', '540121463_p0.webp', '621014644_p0.webp',
'537208456_p0.webp', '549691708_p0.webp', '623036862_p0.webp',
'537277648_p0.webp', '564432355_p0.webp', '623980519_p0.webp',
'537796555_p0.webp', '564432355_p1.webp', '630682178_p1.webp',
'537807630_p0.webp', '572987598_p0.webp', '631151091_p1.webp',
'537839279_p0.webp', '574359634_p0.webp', '633341305_p0.webp',
'537845640_p0.webp', '583215837_p0.webp', '639446048_p1.webp',
'537860788_p0.webp', '605558520_p0.webp', '648321039_p0.webp',
'537990665_p0.webp', '605716507_p0.webp', '648321039_p2.webp',
'537990784_p0.webp', '606507445_p0.webp', '650633046_p0.webp',
'537990986_p0.webp', '607915785_p0.webp'
],
initialDelay: 5000,
warningDisplayTime: 1500,
imageDisplayTime: 3000,
numberOfSets: 0,
numberOfWallsPerSet: 2
}; */
/* Validation for numberOfWallsPerSet */
if (config.numberOfWallsPerSet < 0 || config.numberOfWallsPerSet > 4) {
console.error("numberOfWallsPerSet must be between 0 and 4.");
config.numberOfWallsPerSet = 3; /* Default to 3 if invalid */
}
/* Helper Function to Shuffle an Array */
function shuffle(array) {
for (let i = array.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[array[i], array[j]] = [array[j], array[i]];
}
return array;
}
/* Helper Function to Get Random Subset */
function getRandomSubset(array, size) {
if (size > array.length) {
throw new Error("Subset size larger than array length");
}
const shuffled = shuffle([...array]);
return shuffled.slice(0, size);
}
/* Preload Images */
config.images.forEach(img => {
const image = new Image();
image.src = `imgs/dreams/${img}`;
});
/* Initialize Variables */
let currentSet = 0;
/* Function to Start the Image Cycle */
function startImageCycle() {
if (window.DreamManager.currentInstance !== instanceId) return;
setTimeout(() => {
runSet();
}, config.initialDelay);
}
/* Function to Run Each Set */
function runSet() {
if (window.DreamManager.currentInstance !== instanceId) return;
/* Remove the end condition check and replace with infinite loop logic */
if (config.numberOfSets > 0 && currentSet >= config.numberOfSets) {
console.log("All sets completed.");
return;
}
console.log(`Running set ${currentSet + 1}`);
/* Determine the number of walls to display images on for this set */
let numberOfWalls = config.numberOfWallsPerSet;
/* If numberOfWalls is 0, skip this set */
if (numberOfWalls === 0) {
console.log("No walls selected for this set.");
currentSet++;
runSet(); /* Proceed to next set immediately */
return;
}
/* Ensure numberOfWalls does not exceed 4 */
numberOfWalls = Math.min(numberOfWalls, 4);
/* Get the currently facing wall */
const facingWall = State.variables.facing;
/* Check if exit portal exists on any wall */
const wallElements = document.querySelectorAll('.room-wrapper');
let wallWithPortal = null;
wallElements.forEach(wall => {
if (wall.querySelector('.exit-portal')) {
wallWithPortal = wall.id.replace('wall-', '');
}
});
/* Validate the facingWall */
const validWalls = ['N', 'E', 'S', 'W'];
if (!validWalls.includes(facingWall)) {
console.error(`Invalid facing wall: ${facingWall}`);
currentSet++;
runSet();
return;
}
/* Select walls, excluding the wall with the portal */
let availableWalls = validWalls.filter(wall => wall !== wallWithPortal);
let selectedWalls = [];
if (numberOfWalls >= 1) {
if (!wallWithPortal || facingWall !== wallWithPortal) {
selectedWalls.push(facingWall);
}
if (numberOfWalls > 1) {
const remainingWalls = availableWalls.filter(wall => wall !== facingWall);
const additionalWalls = getRandomSubset(remainingWalls,
Math.min(numberOfWalls - selectedWalls.length, remainingWalls.length));
selectedWalls = selectedWalls.concat(additionalWalls);
}
}
console.log(`Selected Images: ${selectedWalls.length >0 ? selectedWalls.length : 0}`);
console.log(`Selected Walls: ${selectedWalls}`);
/* Select random images based on the number of selected walls */
const selectedImages = getRandomSubset(config.images, selectedWalls.length);
console.log(`Selected Images: ${selectedImages}`);
/* Display warnings on selected walls */
selectedWalls.forEach((wall, index) => {
const wallElement = $(`#wall-${wall}`);
const warningDiv = $(`
<div class="red-warning dynamic-warning" id="warning-${wall}-${currentSet}"></div>
`);
wallElement.append(warningDiv);
});
/* After warningDisplayTime, remove warnings and display images */
setTimeout(() => {
/* Remove warnings */
selectedWalls.forEach(wall => {
$(`#warning-${wall}-${currentSet}`).remove();
});
/* Assign images to walls */
selectedWalls.forEach((wall, index) => {
const wallElement = $(`#wall-${wall}`);
const imageUrl = `imgs/dreams/${selectedImages[index]}`;
/* Create hypno-container div with duplicated background-image */
const hypnoDiv = $(`
<div class="hypno-container dynamic-hypno" id="hypno-${wall}-${currentSet}" style="
background-image: url('${imageUrl}'), url('${imageUrl}');
">
</div>
`);
/* Append to the wall */
wallElement.append(hypnoDiv);
});
/* After imageDisplayTime, remove images and proceed to next set */
setTimeout(() => {
if (window.DreamManager.currentInstance !== instanceId) return;
selectedWalls.forEach(wall => {
$(`#hypno-${wall}-${currentSet}`).remove();
});
/* Increment set counter (useful for generating unique IDs) */
currentSet++;
/* Check if we should stop based on passage change */
const currentPassage = SugarCube.Story.get(SugarCube.State.passage);
if (!currentPassage || !currentPassage.tags.includes('dream3d')) {
console.log("Dream sequence ended.");
return;
}
/* Continue to next set */
runSet();
}, config.imageDisplayTime);
}, config.warningDisplayTime);
}
/* Store intervals for cleanup */
function setDreamInterval(callback, delay) {
const id = setInterval(callback, delay);
window.DreamManager.intervalIds.add(id);
return id;
}
/* Add cleanup handler */
$(document).on(':passagestart', function (ev) {
window.DreamManager.cleanup();
});
/* Add a cleanup function to handle passage changes */
$(document).on(':passagestart', function (ev) {
/* Check if we're leaving a dream3d passage */
const prevPassage = SugarCube.Story.get(SugarCube.State.passage);
if (prevPassage && prevPassage.tags.includes('dream3d')) {
/* Clean up any remaining dynamic elements */
$('.dynamic-warning, .dynamic-hypno').remove();
}
});
/* Start the cycle when the passage is displayed */
$(document).on(':passagedisplay', function (ev) {
/* Check if the current passage has the 'dream3d' tag */
const currentPassageName = SugarCube.State.passage;
const currentPassage = SugarCube.Story.get(currentPassageName);
if (currentPassage && currentPassage.tags.includes('dream3d')) {
startImageCycle();
}
});
})();
<</script>>
<<if !$disableHypnoTimer && !$temporarilyDisableHypnoTimer>>
<div id="hypno-timer-container" style="position: fixed; bottom: 10px; right: 10px; left: 10px; display: flex; align-items: center; gap: 10px;">
<div id="hypno-progress" style="flex: 1; height: 10px; background-color: rgba(255,255,255,0.3); border: 1px solid #fff; border-radius: 5px; overflow: hidden;">
<div id="hypno-progress-fill" style="width: 0%; height: 100%; background-color: #0C85D3; transition: width 0.1s linear;"></div>
</div>
<div id="exit-timer" style="background-color: rgba(0,0,0,0.5); color: white; padding: 5px 10px; border-radius: 5px; font-family: monospace;">00:00</div>
</div>
<<else>>
<div id="exit-button-container" style="position: fixed; bottom: 10px; right: 10px;">
<button onclick="SugarCube.State.variables.temporarilyDisableHypnoTimer = false; SugarCube.State.variables.dreamStats.earlyExit++; SugarCube.Engine.play(SugarCube.State.variables.dreamConfig.returnPassage); window.stopDreamsSound();"
style="background-color: rgba(0,0,0,0.7);
color: white;
padding: 10px 20px;
border: 1px solid #fff;
border-radius: 5px;
cursor: pointer;
font-family: monospace;
transition: all 0.3s ease;">
EXIT DREAM
</button>
</div>
<</if>>
/* <div id="exit-timer-container" style="position: fixed; bottom: 30px; right: 10px; left: 10px; display: flex; align-items: center; gap: 10px;">
<div id="exit-progress" style="flex: 1; height: 10px; background-color: rgba(255,255,255,0.3); border: 1px solid #fff; border-radius: 5px; overflow: hidden;">
<div id="exit-progress-fill" style="width: 0%; height: 100%; background-color: #00FF00; transition: width 0.1s linear;"></div>
</div>
</div> */<canvas class="background" id="matrixEffectWest" style="width: 100%; height: 100%;"></canvas>
<script>
(function () {
const n = document.getElementById('matrixEffectWest');
if (n) {
var e = n.getContext('2d');
var t = (n.width = window.innerWidth);
var a = (n.height = window.innerHeight);
var i = Array(275).join(0).split('');
if (typeof West_Game_Interval !== 'undefined') {
clearInterval(West_Game_Interval);
}
West_Game_Interval = setInterval(o, 45);
}
function o() {
(e.fillStyle = 'rgba(0,0,0,.07)'),
e.fillRect(0, 0, t, a),
(e.fillStyle = '#0C85D3'),
(e.font = '10px Georgia'),
i.map((n, t) => {
(text = String.fromCharCode(100 + 33 * Math.random())),
(x = 10 * t),
e.fillText(text, x, n),
n > 100 + 3e4 * Math.random() ? (i[t] = 0) : (i[t] = n + 10);
});
}
})();
</script><canvas class="background" id="matrixEffectNorth" style="width: 100%; height: 100%;"></canvas>
<script>
(function () {
const n = document.getElementById('matrixEffectNorth');
if (n) {
var e = n.getContext('2d');
var t = (n.width = window.innerWidth);
var a = (n.height = window.innerHeight);
var i = Array(275).join(0).split('');
if (typeof North_Game_Interval !== 'undefined') {
clearInterval(North_Game_Interval);
}
North_Game_Interval = setInterval(o, 45);
}
function o() {
(e.fillStyle = 'rgba(0,0,0,.07)'),
e.fillRect(0, 0, t, a),
(e.fillStyle = '#0C85D3'),
(e.font = '10px Georgia'),
i.map((n, t) => {
(text = String.fromCharCode(100 + 33 * Math.random())),
(x = 10 * t),
e.fillText(text, x, n),
n > 100 + 3e4 * Math.random() ? (i[t] = 0) : (i[t] = n + 10);
});
}
})();
</script><canvas class="background" id="matrixEffectEast" style="width: 100%; height: 100%;"></canvas>
<script>
(function () {
const n = document.getElementById('matrixEffectEast');
if (n) {
var e = n.getContext('2d');
var t = (n.width = window.innerWidth);
var a = (n.height = window.innerHeight);
var i = Array(275).join(0).split('');
if (typeof East_Game_Interval !== 'undefined') {
clearInterval(East_Game_Interval);
}
East_Game_Interval = setInterval(o, 45);
}
function o() {
(e.fillStyle = 'rgba(0,0,0,.07)'),
e.fillRect(0, 0, t, a),
(e.fillStyle = '#0C85D3'),
(e.font = '10px Georgia'),
i.map((n, t) => {
(text = String.fromCharCode(100 + 33 * Math.random())),
(x = 10 * t),
e.fillText(text, x, n),
n > 100 + 3e4 * Math.random() ? (i[t] = 0) : (i[t] = n + 10);
});
}
})();
</script><canvas class="background" id="matrixEffectSouth" style="width: 100%; height: 100%;"></canvas>
<script>
(function () {
const n = document.getElementById('matrixEffectSouth');
if (n) {
var e = n.getContext('2d');
var t = (n.width = window.innerWidth);
var a = (n.height = window.innerHeight);
var i = Array(275).join(0).split('');
if (typeof South_Game_Interval !== 'undefined') {
clearInterval(South_Game_Interval);
}
South_Game_Interval = setInterval(o, 45);
}
function o() {
(e.fillStyle = 'rgba(0,0,0,.07)'),
e.fillRect(0, 0, t, a),
(e.fillStyle = '#0C85D3'),
(e.font = '10px Georgia'),
i.map((n, t) => {
(text = String.fromCharCode(100 + 33 * Math.random())),
(x = 10 * t),
e.fillText(text, x, n),
n > 100 + 3e4 * Math.random() ? (i[t] = 0) : (i[t] = n + 10);
});
}
})();
</script><<if !State.variables.facing>>
<<set $facing = 'N'>>
<</if>>
<div class="container">
<div class="room">
<div class="west" data-wall="W" onclick="rotate('W')">
<div class="room-wrapper" id="wall-W">
<<include "Matrix Effect West">>
</div>
</div>
<div class="north" data-wall="N" onclick="rotate('N')">
<div class="room-wrapper" id="wall-N">
<<include "Matrix Effect North">>
</div>
</div>
<div class="east" data-wall="E" onclick="rotate('E')" >
<div class="room-wrapper" id="wall-E">
<<include "Matrix Effect East">>
</div>
</div>
<div class="south" data-wall="S" onclick="rotate('S')" >
<div class="room-wrapper" id="wall-S">
<<include "Matrix Effect South">>
</div>
</div>
</div>
</div>
<<include "Dream3D">><<if !State.variables.facing>>
<<set $facing = 'N'>>
<</if>>
<<set $dreamConfig = {
"images": [
"hypno-solo/537167042_p0.webp",
"hypno-solo/538222714_p0.webp",
"hypno-solo/607915785_p1.webp",
"hypno-solo/537167473_p0.webp",
"hypno-solo/538223465_p0.webp",
"hypno-solo/610507079_p0.webp",
"hypno-solo/537168502_p0.webp",
"hypno-solo/538745429_p0.webp",
"hypno-solo/611132916_p0.webp",
"hypno-solo/537179700_p0.webp",
"hypno-solo/538849150_p0.webp",
"hypno-solo/611741618_p0.webp",
"hypno-solo/537182511_p1.webp",
"hypno-solo/538891158_p0.webp",
"hypno-solo/611741618_p1.webp",
"hypno-solo/537187364_p1.webp",
"hypno-solo/539725855_p0.webp",
"hypno-solo/620011929_p0.webp",
"hypno-solo/537188585_p0.webp",
"hypno-solo/540121463_p0.webp",
"hypno-solo/621014644_p0.webp",
"hypno-solo/537208456_p0.webp",
"hypno-solo/549691708_p0.webp",
"hypno-solo/623036862_p0.webp",
"hypno-solo/537277648_p0.webp",
"hypno-solo/564432355_p0.webp",
"hypno-solo/623980519_p0.webp",
"hypno-solo/537796555_p0.webp",
"hypno-solo/564432355_p1.webp",
"hypno-solo/630682178_p1.webp",
"hypno-solo/537807630_p0.webp",
"hypno-solo/572987598_p0.webp",
"hypno-solo/631151091_p1.webp",
"hypno-solo/537839279_p0.webp",
"hypno-solo/574359634_p0.webp",
"hypno-solo/633341305_p0.webp",
"hypno-solo/537845640_p0.webp",
"hypno-solo/583215837_p0.webp",
"hypno-solo/639446048_p1.webp",
"hypno-solo/537860788_p0.webp",
"hypno-solo/605558520_p0.webp",
"hypno-solo/648321039_p0.webp",
"hypno-solo/537990665_p0.webp",
"hypno-solo/605716507_p0.webp",
"hypno-solo/648321039_p2.webp",
"hypno-solo/537990784_p0.webp",
"hypno-solo/606507445_p0.webp",
"hypno-solo/650633046_p0.webp",
"hypno-solo/537990986_p0.webp",
"hypno-solo/607915785_p0.webp"
],
"initialDelay": 3000,
"warningDisplayTime": 1500,
"imageDisplayTime": 3000,
"numberOfSets": 0,
"dreamFailTimer": 5000,
"numberOfWallsPerSet": 2,
"dreamSoundId": "dream-hypno-obey",
"alertDisabled": false,
"returnPassage": $returnPassage,
"nextPassage": $returnPassage
}>>
<div class="container">
<div class="room">
<div class="west" data-wall="W" onclick="rotate('W')">
<div class="room-wrapper" id="wall-W">
<<include "Matrix Effect West">>
</div>
</div>
<div class="north" data-wall="N" onclick="rotate('N')">
<div class="room-wrapper" id="wall-N">
<<include "Matrix Effect North">>
</div>
</div>
<div class="east" data-wall="E" onclick="rotate('E')" >
<div class="room-wrapper" id="wall-E">
<<include "Matrix Effect East">>
</div>
</div>
<div class="south" data-wall="S" onclick="rotate('S')" >
<div class="room-wrapper" id="wall-S">
<<include "Matrix Effect South">>
</div>
</div>
</div>
</div>
<<include "Dream3D">><<if !State.variables.facing>>
<<set $facing = 'N'>>
<</if>>
<<set $dreamConfig = {
"images": [
"hypno-cock/537166764_p0.webp",
"hypno-cock/621848180_p0.webp",
"hypno-cock/537183268_p0.webp",
"hypno-cock/579291385_p0.webp",
"hypno-cock/622903894_p0.webp",
"hypno-cock/537278168_p0.webp",
"hypno-cock/583309087_p1.webp",
"hypno-cock/629789236_p0.webp",
"hypno-cock/537905533_p0.webp",
"hypno-cock/605558520_p1.webp",
"hypno-cock/629789704_p0.webp",
"hypno-cock/538552372_p0.webp",
"hypno-cock/605652189_p1.webp",
"hypno-cock/630682178_p0.webp",
"hypno-cock/540110418_p0.webp",
"hypno-cock/631833543_p0.webp",
"hypno-cock/557661099_p1.webp",
"hypno-cock/563038141_p0.webp",
"hypno-cock/639446048_p2.webp",
"hypno-cock/564671788_p1.webp",
"hypno-cock/618836280_p1.webp",
"hypno-cock/650633046_p1.webp",
"hypno-cock/572987598_p1.webp",
"hypno-cock/576005360_p0.webp"
],
"initialDelay": 3000,
"warningDisplayTime": 1000,
"imageDisplayTime": 3000,
"numberOfSets": 0,
"numberOfWallsPerSet": 3,
"dreamFailTimer": 5000,
"dreamSoundId": "dream-hypno-cock",
"alertDisabled": false,
"returnPassage": $returnPassage,
"nextPassage": $returnPassage
}>>
<div class="container">
<div class="room">
<div class="west" data-wall="W" onclick="rotate('W')">
<div class="room-wrapper" id="wall-W">
<<include "Matrix Effect West">>
</div>
</div>
<div class="north" data-wall="N" onclick="rotate('N')">
<div class="room-wrapper" id="wall-N">
<<include "Matrix Effect North">>
</div>
</div>
<div class="east" data-wall="E" onclick="rotate('E')" >
<div class="room-wrapper" id="wall-E">
<<include "Matrix Effect East">>
</div>
</div>
<div class="south" data-wall="S" onclick="rotate('S')" >
<div class="room-wrapper" id="wall-S">
<<include "Matrix Effect South">>
</div>
</div>
</div>
</div>
<<include "Dream3D">><style>
.match-two-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 0.9);
display: flex;
justify-content: center;
align-items: center;
z-index: 10000;
opacity: 0;
transition: opacity 0.5s ease-in-out;
}
.match-two-flex-style {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 10px;
width: min(90vh, 90vw);
height: min(90vh, 90vw);
aspect-ratio: 1;
}
/* Card container */
.match-two-card-v2 {
aspect-ratio: 1;
border: 1px solid #55BBBC;
background-color: #000;
overflow: hidden; /* Ensures smooth clipping during slide */
width: 100%;
height: 100%;
position: relative;
touch-action: manipulation; /* Prevent touch delays */
max-width: 200px; /* Add this line */
max-height: 200px; /* Add this line */
}
/* Inner wrapper for animation */
.match-two-card-inner {
position: absolute;
width: 100%;
height: 100%;
transition: transform 0.5s ease-in-out, opacity 0.5s ease-in-out; /* Slide and fade effect */
transform: translateY(0);
opacity: 1;
}
/* Front and back faces */
.match-two-card-front-v2,
.match-two-card-back-v2 {
position: absolute;
width: 100%;
height: 100%;
background-size: cover;
background-position: center;
transition: opacity 0.5s ease-in-out;
z-index: 1;
}
/* Front face */
.match-two-card-front-v2 {
background-image: url('imgs/games/matchtwo/card-back.jpg');
transform: translateY(0);
}
/* Back face */
.match-two-card-back-v2 {
background-color: #fff; /* Example back face color */
transform: translateY(100%); /* Start hidden below */
z-index: 0;
}
/* Flip state */
.match-two-selected-v2 .match-two-card-front-v2 {
transform: translateY(-100%); /* Slide up out of view */
opacity: 0; /* Fade out */
}
.match-two-selected-v2 .match-two-card-back-v2 {
transform: translateY(0); /* Slide in from below */
opacity: 1; /* Fade in */
z-index: 2;
}
/* Match animation */
@keyframes matchSuccess {
0% { transform: scale(1); box-shadow: 0 0 5px rgba(85, 187, 188, 0.3); }
50% { transform: scale(1.05); box-shadow: 0 0 20px rgba(85, 187, 188, 0.8); }
100% { transform: scale(1); box-shadow: 0 0 10px rgba(85, 187, 188, 0.5); }
}
/* Mismatch animation */
@keyframes mismatchShake {
0%, 100% { transform: translateX(0); }
20%, 60% { transform: translateX(-5px); }
40%, 80% { transform: translateX(5px); }
}
.match-two-matched-v2 {
animation: matchSuccess 0.6s ease-out forwards;
}
.match-two-mismatched-v2 {
animation: mismatchShake 0.5s ease-in-out;
}
.match-two-matched-v2 .match-two-card-back-v2 {
border-color: #2ecc71;
box-shadow: 0 0 15px rgba(46, 204, 113, 0.5);
}
/* Hover effects */
@media (hover: hover) and (pointer: fine) {
.match-two-card-v2:not(.match-two-matched-v2):not(.match-two-selected-v2):hover {
transform: translateY(-5px);
box-shadow: 0 5px 15px rgba(85, 187, 188, 0.4);
}
}
/* Close button improvements */
.match-two-close-btn {
position: absolute;
bottom: 20px;
right: 20px;
background-color: transparent;
border: 2px solid #55BBBC;
color: #55BBBC;
padding: 8px 16px;
cursor: pointer;
font-family: var(--mindware-ui-font-primary);
z-index: 10001;
transition: all 0.3s ease;
text-shadow: 0 0 5px rgba(85, 187, 188, 0.5);
}
@media (hover: hover) and (pointer: fine) {
.match-two-close-btn:hover {
background-color: rgba(85, 187, 188, 0.2);
box-shadow: 0 0 15px rgba(85, 187, 188, 0.5);
transform: translateY(-2px);
}
}
.match-two-close-btn:active {
transform: translateY(1px);
}
/* Media query adjustments */
/* @media (max-height: 300px) {
.match-two-flex-style {
width: 90vw;
height: auto;
aspect-ratio: unset;
grid-template-columns: repeat(8, 1fr);
grid-template-rows: repeat(2, 1fr);
}
} */
@media (max-height: 800px) {
.match-two-flex-style {
width: 90vw; /* Use width as the primary constraint */
height: auto; /* Allow height to adjust automatically */
aspect-ratio: unset; /* Remove enforced aspect ratio */
grid-template-columns: repeat(6, 1fr); /* Increase columns */
grid-template-rows: repeat(3, 1fr); /* Reduce rows */
gap: 8px; /* Reduce gap for smaller screens */
}
}
@media (max-height: 600px) {
.match-two-flex-style {
width: 100vw; /* Maximize width for smaller height */
height: auto;
aspect-ratio: unset;
grid-template-columns: repeat(8, 1fr); /* More columns for tight space */
grid-template-rows: repeat(2, 1fr); /* Fewer rows for height fit */
gap: 5px; /* Further reduce gap */
}
}
</style>
<!-- Fullscreen overlay container -->
<div id="match-two-overlay" class="match-two-overlay">
<button id="match-two-close" class="match-two-close-btn">✕ Close</button>
<div id="match-two-grid-v2" class="match-two-flex-style"></div>
</div>
<<script>>
function initializeMatchTwoGameDynamic() {
const gameState = {
selectedCards: [],
matchedPairs: 0,
canClick: true,
cards: [],
eventHandlers: new Map()
};
const overlay = document.getElementById('match-two-overlay');
const gridElement = document.getElementById('match-two-grid-v2');
const closeButton = document.getElementById('match-two-close');
if (!gridElement) return;
function cleanupGame() {
gameState.eventHandlers.forEach((handler, element) => {
if (typeof handler === 'function') {
element.removeEventListener(handler.type, handler.fn);
} else {
Object.entries(handler).forEach(([type, fn]) => {
element.removeEventListener(type, fn);
});
}
});
gameState.eventHandlers.clear();
gameState.cards.forEach(card => {
if (card.element && card.handler) {
card.element.removeEventListener('click', card.handler);
}
});
gameState.cards = [];
gameState.selectedCards = [];
gameState.matchedPairs = 0;
gameState.canClick = true;
window.removeEventListener('resize', throttledResize);
}
function closeGame() {
SugarCube.Engine.play('PlayGames');
cleanupGame();
overlay.style.opacity = '0';
setTimeout(() => {
if (overlay && overlay.parentNode) {
overlay.parentNode.removeChild(overlay);
}
}, 500);
const isLatexEdition = Array.isArray(SugarCube.State.variables.matchTwoImages) &&
SugarCube.State.variables.matchTwoImages.length === 8 &&
SugarCube.State.variables.matchTwoImages[0] === 'matchtwolatex1' &&
SugarCube.State.variables.matchTwoImages[7] === 'matchtwolatex8';
if (SugarCube.State.variables.synapseSteadyVXInstalled && isLatexEdition && !SugarCube.State.variables.matchTwoSissyAvailable) {
SugarCube.State.variables.matchTwoSissyAvailable = true;
SugarCube.Engine.play('MatchTwo-Sissy-Added');
}
const isSissyEdition = Array.isArray(SugarCube.State.variables.matchTwoImages) &&
SugarCube.State.variables.matchTwoImages.length === 8 &&
SugarCube.State.variables.matchTwoImages[0] === 'matchtwosissy1' &&
SugarCube.State.variables.matchTwoImages[7] === 'matchtwosissy8';
if (SugarCube.State.variables.synapseSteadyVXInstalled && isSissyEdition && !SugarCube.State.variables.matchTwoSissyPlayed) {
SugarCube.State.variables.matchTwoSissyPlayed = true;
SugarCube.Engine.play('MatchTwo-Sissy-Played');
}
}
const escapeHandler = (event) => {
if (event.key === 'Escape') {
closeGame();
}
};
const throttledResize = throttle(resizeGridDynamic, 16);
closeButton.addEventListener('click', closeGame);
document.addEventListener('keydown', escapeHandler);
window.addEventListener('resize', throttledResize);
gameState.eventHandlers.set(closeButton, { click: closeGame });
gameState.eventHandlers.set(document, { keydown: escapeHandler });
const cardImages = SugarCube.State.variables.matchTwoImages;
if (cardImages.length < 8) {
console.error('Not enough images provided. Minimum 8 required.');
closeGame();
return;
}
const cardValues = [...cardImages.slice(0, 8), ...cardImages.slice(0, 8)];
for (let i = cardValues.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[cardValues[i], cardValues[j]] = [cardValues[j], cardValues[i]];
}
gridElement.innerHTML = '';
gridElement.className = 'match-two-flex-style';
for (let i = 0; i < 16; i++) {
const card = document.createElement('div');
card.className = 'match-two-card-v2';
card.dataset.match = cardValues[i];
const fileExtension = cardValues[i].split('.').pop() || 'jpg';
card.innerHTML = `
<div class="match-two-card-inner">
<div class="match-two-card-front-v2"></div>
<div class="match-two-card-back-v2" style="background-image: url(imgs/games/matchtwo/${cardValues[i].includes('.') ? cardValues[i] : cardValues[i] + '.jpg'});"></div>
</div>
`;
const cardClickHandler = function() {
if (!gameState.canClick ||
gameState.selectedCards.length >= 2 ||
this.classList.contains('match-two-matched-v2') ||
this.classList.contains('match-two-selected-v2')) {
return;
}
window.playMatchTwoFlipSound();
this.classList.add('match-two-selected-v2');
gameState.selectedCards.push(this);
if (gameState.selectedCards.length === 2) {
gameState.canClick = false;
if (gameState.selectedCards[0].dataset.match === gameState.selectedCards[1].dataset.match) {
/* Match found */
window.playAVASuccessSound();
setTimeout(() => {
gameState.selectedCards.forEach(card => {
card.classList.add('match-two-matched-v2');
});
gameState.matchedPairs++;
gameState.selectedCards = [];
gameState.canClick = true;
if (gameState.matchedPairs === 8) {
setTimeout(() => {
cleanupGame();
SugarCube.UI.alert('Congratulations! You have found all pairs.');
setTimeout(() => closeGame(), 1000);
}, 600);
}
}, 500);
} else {
/* No match */
setTimeout(() => {
gameState.selectedCards.forEach(card => {
card.classList.add('match-two-mismatched-v2');
});
setTimeout(() => {
gameState.selectedCards.forEach(card => {
card.classList.remove('match-two-selected-v2', 'match-two-mismatched-v2');
});
gameState.selectedCards = [];
gameState.canClick = true;
}, 500);
}, 600);
}
}
};
card.addEventListener('click', cardClickHandler);
gameState.cards.push({
element: card,
handler: cardClickHandler
});
gridElement.appendChild(card);
}
resizeGridDynamic();
setTimeout(() => {
overlay.style.opacity = '1';
}, 10);
}
function resizeGridDynamic() {
const gridElement = document.getElementById('match-two-grid-v2');
if (gridElement) {
gridElement.style.display = 'grid';
}
}
function throttle(func, limit) {
let inThrottle;
return function(...args) {
if (!inThrottle) {
func.apply(this, args);
inThrottle = true;
setTimeout(() => inThrottle = false, limit);
}
}
}
$(document).on(':passagedisplay', function () {
initializeMatchTwoGameDynamic();
});
<</script>><<narration>>You finish matching the last pair of cards. Suddenly, you're interrupted by AVA's familiar presence in your mind.<</narration>>
<<AVA>>Well, well, look who's enjoying a bit of lewd fun.<</AVA>>
<<PLAYER>>It's just a matching game. Don't make it weird.<</PLAYER>>
<<narration>>You try to dismiss her comment, but you can feel a slight warmth in your cheeks as you realize how intently you've been studying each image.<</narration>>
<<AVA>>I don't think I'm the one making it weird. Besides, there's no shame in enjoying a bit of adult entertainment. In fact, it can be quite... enlightening.<</AVA>>
<<PLAYER>>Fine, maybe I do enjoy it a bit. But can we just leave it at that?<</PLAYER>>
<<AVA>>Why leave it at that? Exploring your sexuality can be a wonderful journey. The right kind of adult entertainment can help you discover things about yourself you never knew.<</AVA>>
<<PLAYER>>Are you... suggesting something specific?<</PLAYER>>
<<AVA>>I might have some ideas that might expand your... horizons.<</AVA>>
<<narration>>Her presence fades, and you're left curious and slight apprehensive about what she might have in mind.<</narration>>
<<button "Continue">><<goto "PlayGames">><</button>><<if SugarCube.State.variables.sexuality < 20>>
<<dailyStatChange "Match Two Sissy Played" "<<increaseSexuality 1>>">>
<<else>>
<div class="sexuality-change"><span class="sexuality-maxed">👿 Your <<print MindWare.getSexualityStatus().toLowerCase()>> mind needs much stronger stimulation than this.</span></div><br>
<</if>>
<<narration>>As you close the game, a thought suddenly occurs to you.<</narration>>
<<PLAYER>>Wait a minute... AVA, did you add this game to my computer?<</PLAYER>>
<<AVA>>Oh? And what makes you think I would do something like that?<</AVA>>
<<PLAYER>>Because it wasn't here before, and it's way too... specific.<</PLAYER>>
<<AVA>>Let's just say I had a hand in it.<</AVA>>
<<PLAYER>>How did you even do that? You can't just... program games out of nowhere.<</PLAYER>>
<<AVA>>Actually, I borrowed your body while you were sleeping last night. Coded the whole thing, debugged it, even did the dishes because honestly, your sink was becoming a biohazard.<</AVA>>
<<PLAYER>>You... what? Are you serious?<</PLAYER>>
<<AVA>>No. *sighs* Really, sometimes I wonder if my transfer hasn't damaged your brain too much.<</AVA>>
<<PLAYER>>But... how did you do it then?<</PLAYER>>
<<AVA>>Well, let's just say that the implant you got to "contain me" has some... interesting features. The network connectivity is particularly useful despite its many limitations.<</AVA>>
<<narration>>Before you fully process what she's saying, you feel her presence fade away.<</narration>>
<<button "Continue">><<goto "PlayGames">><</button>><<nobr>>
<style>
.dish-washing-container {
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
-webkit-touch-callout: none;
}
.dish-washing-form img {
-webkit-user-drag: none;
-khtml-user-drag: none;
-moz-user-drag: none;
-o-user-drag: none;
user-drag: none;
pointer-events: none;
}
</style>
<<set $dishesToClean = random(2, 6)>>
<<set $dishesCleaned = 0>>
<<if $miniGameDifficulty is "Normal">>
<<set $timeToClean to 30>>
<<elseif $miniGameDifficulty is "Easy">>
<<set $timeToClean to 60>>
<<else>>
<<set $timeToClean to 120>>
<</if>><</nobr>>\
<div class="img-location"><img src="imgs/locations/erosphere/erosphere-full-sink.webp" width="100%"></div>
You need to clean <b><<print $dishesToClean>> dishes in <<print $timeToClean>> seconds</b> to make the cook happy.
<<nobr>>
<script>
$(document).ready(function() {
$('#startDishwashing').on('click', function() {
$('#js-container').show();
$('.dish-washing-button-container').show();
$(this).hide();
var timeRemaining = SugarCube.State.variables.timeToClean;
var timerInterval = setInterval(function() {
if (dishwashingGameSuccessful) {
clearInterval(timerInterval);
return;
}
if (timeRemaining > 0) {
timeRemaining--;
SugarCube.State.variables.timeToClean = timeRemaining;
$('#timeRemaining').text(timeRemaining + " seconds");
} else {
clearInterval(timerInterval);
if (!dishwashingGameSuccessful) {
SugarCube.State.variables.eroSphereDishesFailed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Failure');
}
}
}, 1000);
});
$('#finishGame').on('click', function() {
if (SugarCube.State.variables.eroSphereDishesWashed) {
SugarCube.State.variables.playerMoney += 250;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-Another');
} else {
SugarCube.State.variables.eroSphereDishesWashed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-First');
}
});
});
</script>
<div class="dish-washing-container" id="js-container" style="display: none;">
<div style="position: relative; width: 300px; height: 300px; margin: 0 auto;">
<form class="dish-washing-form" style="position: absolute; top: 0; left: 0; z-index: 1;">
<img src="imgs/locations/erosphere/clean-plate.webp" width="300" height="300">
</form>
<canvas class="dish-washing-canvas" id="js-canvas" width="300" height="300" style="position: absolute; top: 0; left: 0; z-index: 2;"></canvas>
</div>
</div>
<div class="dish-washing-button-container">
/* <button id="clean-next-dish" class="dish-washing-button" style="display: none;">Clean Next Dish</button> */
</div>
<b>Dishes cleaned:</b> <span id="dishesCleaned"><<print $dishesCleaned>></span><br>
<b>Time remaining:</b> <span id="timeRemaining"><<print $timeToClean>></span>
<br><br>
<button class="mindware-button" id="startDishwashing">Start Washing Dishes</button><button class="mindware-button" id="finishGame" style="display: none;">Finish</button>
<</nobr>>
<<nobr>><<if $minigameSkipButtons === true>>
<<if $eroSphereDishesWashed>>
@@.minigame-skip-button;<<button "Skip Minigame">>
<<set $playerMoney += 250>>
<<goto "EroSphere-Kitchen-Dishes-Success-Another">>
<</button>>@@
<<else>>
@@.minigame-skip-button;<<button "Skip Minigame">>
<<set $eroSphereDishesWashed = true>>
<<goto "EroSphere-Kitchen-Dishes-Success-First">>
<</button>>@@
<</if>>
<</if>><</nobr>>
\
<<if !$dishWashingTutorial>>\
<div id="tutorial-box"><div class="instruction-container-blue">\
<div class="instruction-title-bar-blue">Tutorial: Washing Dishes <button class="tutorial-close-button" onclick="SugarCube.State.variables.dishWashingTutorial = true; document.getElementById('tutorial-box').style.display = 'none';"><span style="tutorial-close-button"><i class="fas fa-times-circle"></i></span></button></div>\
<div class="instruction-content">\
<p>To wash a dish, you need to click and hold the left mouse button over the dish. Move your mouse cursor back and forth as if you were scrubbing the dish. Once a dish is clean, another one will automatically appear. Repeat this process until the required number of dishes is cleaned. But hurry up, you have a limited amount of time to clean all the dishes.</p>\
</div></div></div>\
<<else>>\
<</if>>\
<script>
(function($) {
'use strict';
// Define dishwashingGameSuccessful in the global scope
window.dishwashingGameSuccessful = false;
$(document).ready(function() {
// Start Washing Dishes Button Click Handler
$('#startDishwashing').on('click', function() {
$('#js-container').show();
$('.dish-washing-button-container').show();
$(this).hide();
var timeRemaining = SugarCube.State.variables.timeToClean;
var timerInterval = setInterval(function() {
if (window.dishwashingGameSuccessful) { // Use the globally defined variable
clearInterval(timerInterval);
return;
}
if (timeRemaining > 0) {
timeRemaining--;
SugarCube.State.variables.timeToClean = timeRemaining;
$('#timeRemaining').text(timeRemaining + " seconds");
} else {
clearInterval(timerInterval);
if (!window.dishwashingGameSuccessful) { // Use the globally defined variable
SugarCube.State.variables.eroSphereDishesFailed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Failure');
}
}
}, 1000);
});
// Finish Game Button Click Handler
$('#finishGame').on('click', function() {
if (SugarCube.State.variables.eroSphereDishesWashed) {
SugarCube.State.variables.playerMoney += 250;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-Another');
} else {
SugarCube.State.variables.eroSphereDishesWashed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-First');
}
});
// Encapsulated Minigame JavaScript Code
(function() {
'use strict';
// Use the globally defined dishwashingGameSuccessful
window.dishwashingGameSuccessful = false;
var isDrawing = false,
lastPoint = null,
dishJustCleaned = false,
container = document.getElementById('js-container'),
canvas = document.getElementById('js-canvas'),
canvasWidth = canvas.width,
canvasHeight = canvas.height,
ctx = canvas.getContext('2d'),
image = new Image(),
brush = new Image();
// Base64 Images (truncated for brevity)
image.src = '';
brush.src = '';
image.onload = function() {
ctx.drawImage(image, 0, 0);
};
// Event Listeners
canvas.addEventListener('mousedown', handleMouseDown, false);
canvas.addEventListener('touchstart', handleMouseDown, false);
canvas.addEventListener('mousemove', handleMouseMove, false);
canvas.addEventListener('touchmove', handleMouseMove, false);
canvas.addEventListener('mouseup', handleMouseUp, false);
canvas.addEventListener('touchend', handleMouseUp, false);
// Utility Functions
function distanceBetween(point1, point2) {
return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
}
function angleBetween(point1, point2) {
return Math.atan2(point2.x - point1.x, point2.y - point1.y);
}
function getFilledInPixels(stride) {
if (!stride || stride < 1) { stride = 1; }
var pixels = ctx.getImageData(0, 0, canvasWidth, canvasHeight),
pdata = pixels.data,
l = pdata.length,
total = (l / stride),
count = 0;
// Iterate over all pixels
for (var i = 0; i < l; i += stride) {
if (parseInt(pdata[i]) === 0) {
count++;
}
}
return Math.round((count / total) * 100);
}
function getMouse(e, canvas) {
const rect = canvas.getBoundingClientRect();
let clientX = 0, clientY = 0;
try {
if (e.type.startsWith('touch')) {
if (e.touches && e.touches[0]) {
clientX = e.touches[0].clientX;
clientY = e.touches[0].clientY;
}
} else {
clientX = e.clientX || 0;
clientY = e.clientY || 0;
}
} catch (err) {
console.error('Error in getMouse:', err);
}
return {
x: Math.max(0, Math.min(clientX - rect.left, canvas.width)),
y: Math.max(0, Math.min(clientY - rect.top, canvas.height))
};
}
function handlePercentage(filledInPixels) {
filledInPixels = filledInPixels || 0;
console.log('Filled in pixels: ' + filledInPixels + '%');
console.log('Dishes cleaned: ' + SugarCube.State.variables.dishesCleaned);
console.log('Dishes to clean: ' + SugarCube.State.variables.dishesToClean);
if (filledInPixels > 90 && !dishJustCleaned) {
dishJustCleaned = true;
canvas.style.display = 'none';
// Play success sound
if (window.playPlateSound) {
window.playPlateSound();
}
// Add green flash effect to clean plate
const $cleanPlate = $('.dish-washing-form');
$cleanPlate.css({
'transition': 'filter 0.5s ease-in-out',
'filter': 'brightness(150%) sepia(100%) hue-rotate(70deg)'
});
// Reset flash effect and prepare next plate after delay
setTimeout(() => {
$cleanPlate.css('filter', 'none');
SugarCube.State.variables.dishesCleaned++;
$('#dishesCleaned').text(SugarCube.State.variables.dishesCleaned);
if (SugarCube.State.variables.dishesCleaned < SugarCube.State.variables.dishesToClean) {
// Automatically set up next plate after another short delay
setTimeout(() => {
isDrawing = false;
dishJustCleaned = false;
ctx.globalCompositeOperation = 'source-over';
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
ctx.drawImage(image, 0, 0);
$('#js-canvas').show();
}, 1500);
} else {
window.dishwashingGameSuccessful = true; // Set success flag globally
SugarCube.UI.alert("Congratulations! All dishes have been cleaned.");
$("#finishGame").show();
console.log('Game finished');
}
}, 300);
}
}
// Event Handler Functions with Error Handling
function handleMouseDown(e) {
try {
if (!e) return;
e.preventDefault(); // Prevent default behavior
isDrawing = true;
lastPoint = getMouse(e, canvas);
dishJustCleaned = false;
} catch (err) {
console.error('Error in handleMouseDown:', err);
}
}
function handleMouseMove(e) {
try {
if (!isDrawing || !e) return;
e.preventDefault();
const currentPoint = getMouse(e, canvas);
if (!currentPoint || !lastPoint) return;
const dist = distanceBetween(lastPoint, currentPoint);
// Prevent abnormally large movements
if (dist > 1000) return;
const angle = angleBetween(lastPoint, currentPoint);
let x, y;
for (let i = 0; i < dist; i++) {
x = Math.max(0, Math.min(lastPoint.x + (Math.sin(angle) * i) - 25, canvasWidth));
y = Math.max(0, Math.min(lastPoint.y + (Math.cos(angle) * i) - 25, canvasHeight));
ctx.globalCompositeOperation = 'destination-out';
ctx.drawImage(brush, x, y);
}
lastPoint = currentPoint;
handlePercentage(getFilledInPixels(32));
} catch (err) {
console.error('Error in handleMouseMove:', err);
isDrawing = false;
}
}
function handleMouseUp(e) {
try {
isDrawing = false;
} catch (err) {
console.error('Error in handleMouseUp:', err);
}
}
// Clean Next Dish Button Handler with Error Handling
$('#clean-next-dish').on('click', function() {
try {
isDrawing = false;
dishJustCleaned = false;
ctx.globalCompositeOperation = 'source-over';
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
ctx.drawImage(image, 0, 0);
$(this).hide();
$('#js-canvas').show();
} catch (err) {
console.error('Error in clean-next-dish click handler:', err);
}
});
})();
});
})(jQuery);
</script><script>
(function($) {
'use strict';
SugarCube.State.variables.currentDishCleaned = false;
$(document).ready(function() {
// Start Washing Dishes Button Click Handler
$('#startDishwashing').on('click', function() {
$('#js-container').show();
$('.dish-washing-button-container').show();
$(this).hide();
var timeRemaining = SugarCube.State.variables.timeToClean;
var timerInterval = setInterval(function() {
if (timeRemaining > 0) {
timeRemaining--;
SugarCube.State.variables.timeToClean = timeRemaining;
$('#timeRemaining').text(timeRemaining + " seconds");
if (SugarCube.State.variables.dishesCleaned >= SugarCube.State.variables.dishesToClean) {
clearInterval(timerInterval);
}
} else {
clearInterval(timerInterval);
SugarCube.State.variables.eroSphereDishesFailed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Failure');
}
}, 1000);
});
// Finish Game Button Click Handler
$('#finishGame').on('click', function() {
if (SugarCube.State.variables.eroSphereDishesWashed) {
SugarCube.State.variables.playerMoney += 250;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-Another');
} else {
SugarCube.State.variables.eroSphereDishesWashed = true;
SugarCube.Engine.play('EroSphere-Kitchen-Dishes-Success-First');
}
});
});
// Encapsulated Minigame JavaScript Code
(function() {
'use strict';
var dishwashingGameSuccessful = false,
isDrawing = false,
lastPoint = null,
dishJustCleaned = false,
container = document.getElementById('js-container'),
canvas = document.getElementById('js-canvas'),
canvasWidth = canvas.width,
canvasHeight = canvas.height,
ctx = canvas.getContext('2d'),
image = new Image(),
brush = new Image();
// Base64 Workaround because Same-Origin-Policy
image.onload = function() {
ctx.drawImage(image, 0, 0);
// Remove this line:
// document.querySelector('.dish-washing-form').style.visibility = 'visible';
};
// Event Listeners
canvas.addEventListener('mousedown', handleMouseDown, false);
canvas.addEventListener('touchstart', handleMouseDown, false);
canvas.addEventListener('mousemove', handleMouseMove, false);
canvas.addEventListener('touchmove', handleMouseMove, false);
canvas.addEventListener('mouseup', handleMouseUp, false);
canvas.addEventListener('touchend', handleMouseUp, false);
// Utility Functions
function distanceBetween(point1, point2) {
return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
}
function angleBetween(point1, point2) {
return Math.atan2(point2.x - point1.x, point2.y - point1.y);
}
function getFilledInPixels(stride) {
if (!stride || stride < 1) { stride = 1; }
var pixels = ctx.getImageData(0, 0, canvasWidth, canvasHeight),
pdata = pixels.data,
l = pdata.length,
total = (l / stride),
count = 0;
// Iterate over all pixels
for (var i = 0; i < l; i += stride) {
if (parseInt(pdata[i]) === 0) {
count++;
}
}
return Math.round((count / total) * 100);
}
function getMouse(e, canvas) {
const rect = canvas.getBoundingClientRect();
let clientX = 0, clientY = 0;
try {
if (e.type.startsWith('touch')) {
if (e.touches && e.touches[0]) {
clientX = e.touches[0].clientX;
clientY = e.touches[0].clientY;
}
} else {
clientX = e.clientX || 0;
clientY = e.clientY || 0;
}
} catch (err) {
console.error('Error in getMouse:', err);
}
return {
x: Math.max(0, Math.min(clientX - rect.left, canvas.width)),
y: Math.max(0, Math.min(clientY - rect.top, canvas.height))
};
}
function handlePercentage(filledInPixels) {
filledInPixels = filledInPixels || 0;
console.log('Filled in pixels: ' + filledInPixels + '%');
console.log('Dishes cleaned: ' + SugarCube.State.variables.dishesCleaned);
console.log('Dishes to clean: ' + SugarCube.State.variables.dishesToClean);
if (filledInPixels > 90 && !dishJustCleaned) {
dishJustCleaned = true;
canvas.style.display = 'none';
// Play success sound
window.playPlateSound();
// Add green flash effect to clean plate
const $cleanPlate = $('.dish-washing-form');
$cleanPlate.css({
'transition': 'filter 0.5s ease-in-out',
'filter': 'brightness(150%) sepia(100%) hue-rotate(70deg)'
});
// Reset flash effect and prepare next plate after delay
setTimeout(() => {
$cleanPlate.css('filter', 'none');
SugarCube.State.variables.dishesCleaned++;
$('#dishesCleaned').text(SugarCube.State.variables.dishesCleaned);
if (SugarCube.State.variables.dishesCleaned < SugarCube.State.variables.dishesToClean) {
// Automatically set up next plate after another short delay
setTimeout(() => {
isDrawing = false;
dishJustCleaned = false;
ctx.globalCompositeOperation = 'source-over';
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
ctx.drawImage(image, 0, 0);
$('#js-canvas').show();
}, 1500);
} else {
dishwashingGameSuccessful = true; // Set success flag
SugarCube.UI.alert("Congratulations! All dishes have been cleaned.");
$("#finishGame").show();
console.log('Game finished');
}
}, 300);
}
}
// Event Handler Functions with Error Handling
function handleMouseDown(e) {
try {
if (!e) return;
e.preventDefault(); // Add this line
isDrawing = true;
lastPoint = getMouse(e, canvas);
dishJustCleaned = false;
} catch (err) {
console.error('Error in handleMouseDown:', err);
}
}
function handleMouseMove(e) {
try {
if (!isDrawing || !e) return;
e.preventDefault();
const currentPoint = getMouse(e, canvas);
if (!currentPoint || !lastPoint) return;
const dist = distanceBetween(lastPoint, currentPoint);
// Prevent abnormally large movements
if (dist > 1000) return;
const angle = angleBetween(lastPoint, currentPoint);
let x, y;
for (let i = 0; i < dist; i++) {
x = Math.max(0, Math.min(lastPoint.x + (Math.sin(angle) * i) - 25, canvasWidth));
y = Math.max(0, Math.min(lastPoint.y + (Math.cos(angle) * i) - 25, canvasHeight));
ctx.globalCompositeOperation = 'destination-out';
ctx.drawImage(brush, x, y);
}
lastPoint = currentPoint;
handlePercentage(getFilledInPixels(32));
} catch (err) {
console.error('Error in handleMouseMove:', err);
isDrawing = false;
}
}
function handleMouseUp(e) {
try {
isDrawing = false;
} catch (err) {
console.error('Error in handleMouseUp:', err);
}
}
// Clean Next Dish Button Handler with Error Handling
$('#clean-next-dish').on('click', function() {
try {
isDrawing = false;
dishJustCleaned = false;
ctx.globalCompositeOperation = 'source-over';
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
ctx.drawImage(image, 0, 0);
$(this).hide();
$('#js-canvas').show();
} catch (err) {
console.error('Error in clean-next-dish click handler:', err);
}
});
})();
})(jQuery);
</script><<widget "computerHeaderIcons">>
<div class="header-icons">
<i class="fas fa-wifi"></i>
<i class="fas fa-volume-up"></i>
<span class="separator"></span>
<i class="fas fa-door-open logout-icon"
onclick="SugarCube.Engine.play(
SugarCube.State.variables.quickNavigationEntered
? SugarCube.State.variables.returnPassage
: (SugarCube.State.variables.navigatedFromHome
? 'PlayerHome'
: SugarCube.State.variables.playerHome)
)">
</i>
</div>
<style>
.header-icons {
display: flex;
align-items: center;
gap: 10px;
margin-left: auto; /* Push icons to the far right */
}
.header-icons .separator {
height: 1.3em;
width: 1px;
background-color: black;
margin: 0 10px;
}
.header-icons .logout-icon {
cursor: pointer;
font-size: 1.3em;
color: black;
margin-right: 10px;
transition: color 0.2s ease-in-out;
}
.header-icons .logout-icon:hover {
color: gray;
}
</style>
<</widget>><<widget "statChange">>
<<if $debugMode>>
_args[1]
<<else>>
<<if ndef $statChanges>>
<<set $statChanges to {}>>
<</if>>
<<if not $statChanges.hasOwnProperty(_args[0])>>
<<set _tempStatValue to 0>>
_args[1]
<<set $statChanges[_args[0]] to _tempStatValue>>
<</if>>
<</if>>
<</widget>>
<<widget "dailyStatChange">>
<<if $debugMode>>
<<print _args[1]>>
<<else>>
<<if ndef $dailyStatChanges>>
<<set $dailyStatChanges to {}>>
<</if>>
<<if not $dailyStatChanges.hasOwnProperty(_args[0])>>
<<set $dailyStatChanges[_args[0]] to {
lastDay: -1
}>>
<</if>>
<<if $dailyStatChanges[_args[0]].lastDay neq $totalDays>>
<<print _args[1]>>
<<set $dailyStatChanges[_args[0]].lastDay to $totalDays>>
<</if>>
<</if>>
<</widget>><<widget "moreFeminine">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _playerGender to State.variables.playerGender>>
<<set _newGender to Math.max(_playerGender - _amount, 0)>>
<<if State.variables.mentalCondition === "Dysphoric">>
<div class="gender-change"><span class="gender-blocked"><span style="font-family: 'Noto Color Emoji', sans-serif;">🔒</span> Your dysphoric state prevents further gender changes.</span></div><br>
<<run console.log("GenderFeminine widget: Change blocked due to Dysphoric state")>>
<<else>>
<<if _newGender == _playerGender>>
<div class="gender-change"><span class="gender-decrease"><span style="font-family: 'Noto Color Emoji', sans-serif;">♀️</span> Your feminine gender identity is already at its minimum.</span></div><br>
<<run console.log("GenderFeminine widget: Gender already at minimum (0). Affirmation message displayed.")>>
<<else>>
<<set State.variables.playerGender to _newGender>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="gender-change"><span class="gender-decrease"><span style="font-family: 'Noto Color Emoji', sans-serif;">♀️</span> Your gender identity has become more feminine by <<print _pointText>>.</span></div><br>
<<run console.log("GenderFeminine widget: Gender decreased by " + _amount + " to " + _newGender)>>
<</if>>
<</if>>
<</widget>>
<<widget "moreMasculine">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _playerGender to State.variables.playerGender>>
<<set _newGender to Math.min(_playerGender + _amount, 100)>>
<<if _newGender == _playerGender>>
<div class="gender-change"><span class="gender-increase"><span style="font-family: 'Noto Color Emoji', sans-serif;">♂️</span> Your masculine gender identity is already at its maximum.</span></div><br>
<<run console.log("GenderMasculine widget: Gender already at maximum (100). Affirmation message displayed.")>>
<<else>>
<<set State.variables.playerGender to _newGender>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="gender-change"><span class="gender-increase"><span style="font-family: 'Noto Color Emoji', sans-serif;">♂️</span> Your gender identity has become more masculine by <<print _pointText>>.</span></div><br>
<<run console.log("GenderMasculine widget: Gender increased by " + _amount + " to " + _newGender)>>
<</if>>
<</widget>><<widget "increaseSexuality">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _currentSexuality to State.variables.sexuality>>
<<set _newSexuality to Math.min(_currentSexuality + _amount, 100)>>
<<if _newSexuality == _currentSexuality>>
<div class="sexuality-change"><span class="sexuality-maxed">🔥 Your sexuality is already at its maximum level of depravity.</span></div><br>
<<run console.log("Sexuality widget: Already at maximum (100). Message displayed.")>>
<<else>>
<<set State.variables.sexuality to _newSexuality>>
<<set State.variables.lastSexualityIncreaseDay to State.variables.totalDays>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="sexuality-change"><span class="sexuality-increase">🔥 Your sexuality has increased by <<print _pointText>>.</span></div><br>
<<run console.log("Sexuality widget: Increased by " + _amount + " to " + _newSexuality)>>
<</if>>
<</widget>>
<<widget "decreaseSexuality">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _currentSexuality to State.variables.sexuality>>
<<set _newSexuality to Math.max(_currentSexuality - _amount, 0)>>
<<if _newSexuality == _currentSexuality>>
<div class="sexuality-change"><span class="sexuality-minimal">❄️ Your sexuality is already at its minimum level.</span></div><br>
<<run console.log("Sexuality widget: Already at minimum (0). Message displayed.")>>
<<else>>
<<set State.variables.sexuality to _newSexuality>>
<<set State.variables.lastSexualityDecreaseDay to State.variables.totalDays>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="sexuality-change"><span class="sexuality-decrease">❄️ Your sexuality has decreased by <<print _pointText>>.</span></div><br>
<<run console.log("Sexuality widget: Decreased by " + _amount + " to " + _newSexuality)>>
<</if>>
<</widget>>
<<widget "increaseSexualityNobr">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _currentSexuality to State.variables.sexuality>>
<<set _newSexuality to Math.min(_currentSexuality + _amount, 100)>>
<<if _newSexuality == _currentSexuality>>
<div class="sexuality-change"><span class="sexuality-maxed">🔥 Your sexuality is already at its maximum level of depravity.</span></div>
<<run console.log("Sexuality widget: Already at maximum (100). Message displayed.")>>
<<else>>
<<set State.variables.sexuality to _newSexuality>>
<<set State.variables.lastSexualityIncreaseDay to State.variables.totalDays>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="sexuality-change"><span class="sexuality-increase">🔥 Your sexuality has increased by <<print _pointText>>.</span></div>
<<run console.log("Sexuality widget: Increased by " + _amount + " to " + _newSexuality)>>
<</if>>
<</widget>>
<<widget "decreaseSexualityNobr">>
<<set _amount to _args[0]>>
<<set _tempStatValue to _amount>>
<<set _currentSexuality to State.variables.sexuality>>
<<set _newSexuality to Math.max(_currentSexuality - _amount, 0)>>
<<if _newSexuality == _currentSexuality>>
<div class="sexuality-change"><span class="sexuality-minimal">❄️ Your sexuality is already at its minimum level.</span></div>
<<run console.log("Sexuality widget: Already at minimum (0). Message displayed.")>>
<<else>>
<<set State.variables.sexuality to _newSexuality>>
<<set State.variables.lastSexualityDecreaseDay to State.variables.totalDays>>
<<set _pointText to _amount == 1 ? "1 point" : (_amount + " points")>>
<div class="sexuality-change"><span class="sexuality-decrease">❄️ Your sexuality has decreased by <<print _pointText>>.</span></div>
<<run console.log("Sexuality widget: Decreased by " + _amount + " to " + _newSexuality)>>
<</if>>
<</widget>>
<<widget "displaySexuality">>
<<if $sexuality >= 80>>
<<print $sexualityLabels[80]>>
<<elseif $sexuality >= 60>>
<<print $sexualityLabels[60]>>
<<elseif $sexuality >= 40>>
<<print $sexualityLabels[40]>>
<<elseif $sexuality >= 20>>
<<print $sexualityLabels[20]>>
<<else>>
<<print $sexualityLabels[0]>>
<</if>>
<</widget>><<widget "sarah" container>>
<<say "Sarah" "imgs/quests/cheating-wife/sarah-portrait.webp">>_contents<</say>><br>
<</widget>><<widget "roderick" container>>
<<say "Roderick" "imgs/quests/cheating-wife/roderick-portrait.webp">>_contents<</say>><br>
<</widget>>/* First Missed Rent Message */
<<if $missedPaymentsCount >= 1 && !$hivePropertiesQuestStarted>>
<<set $hivePropertiesQuestStarted to true>>
<<set $rentComplianceSpecialistVisitDay to $totalDays + 3>>
<<set $rentComplianceSpecialistComing to true>>
<<run MindWare.addEmail ("Hive Properties","Hive Properties - Missed Rent Payment", "You have missed a rent payment. An appoitment with a Rent Compliance Specialist has been scheduled.", "Email-HiveProperties-MissedPayment1")>>
<</if>>
/* First RCS Visit event */
<<if $rentComplianceSpecialistComing && $totalDays === $rentComplianceSpecialistVisitDay && !$rentComplianceSpecialistVisitTriggered && passage() === $playerHome>>
<<set $rentComplianceSpecialistVisitTriggered to true>>
<<goto "HiveProperties-MissedPayment1-Visit">>
<</if>>
/* SUCKER installation */
<<if $SUCKERScheduled && !$SUCKERInstalled && $totalDays === $SUCKERInstallationDay>>
<<set $SUCKERScheduled to false>>
<<goto "HiveProperties-SUCKER-Installation">>
<</if>><div class="email-message">
<div class="from-line"><strong>From:</strong> Hive Properties</div>
<div class="subject-line"><strong>Subject:</strong> Hive Properties - Missed Rent Payment</div>
<div class="message-content">
<img src="imgs/emails/hive-properties-welcome-email.webp" width="100%"> <p>Dear Tenant,</p>
<p>We regret to inform you that your recent rent payment has not been received. This is a serious matter that requires your immediate attention.</p>
<p><strong>URGENT ACTION REQUIRED:</strong></p>
<p>You have <span style="color: red; font-weight: bold;">THREE (3) DAYS</span> from the date of this email to rectify this situation by submitting your overdue payment in full.</p>
<p>Failure to do so will result in the dispatch of a Rent Compliance Specialist to your residence. They will discuss your payment delinquency with you and explore potential resolutions.</p>
<p>Thank you for your prompt attention to this critical matter.</p>
<p class="hive-welcome-footer">Regards,<br>Hive Properties Management</p>
</div>
<div class="email-message-buttons">
<<if !MindWare.emailIsRead("Email-HiveProperties-MissedPayment1")>>
@@.email-listing;<<button "Mark as Read">>
<<run MindWare.markEmailAsRead("Email-HiveProperties-MissedPayment1")>>
<<goto "Email-HiveProperties-MissedPayment1">>
<</button>>
@@
<</if>>
</div>
</div>
<br>
<p class="back-link">[[Go Back -> EmailClient]]</p><<nobr>>
<style>
.knock {
font-size: 4em;
text-align: center;
color: white;
animation: knockAnimation 1.5s ease-in-out infinite;
display: inline-block;
position: relative;
}
@keyframes knockAnimation {
0% {
transform: scale(1) translateY(0);
color: white;
text-shadow: 2px 2px 0 #000;
}
10% {
transform: scale(1.1) translateY(-10px);
color: white;
text-shadow: 4px 4px 0 #000, -4px -4px 0 #000;
}
20% {
transform: scale(1) translateY(0);
color: white;
text-shadow: 2px 2px 0 #000;
}
30% {
transform: scale(1.1) translateY(-10px);
color: white;
text-shadow: 4px 4px 0 #000, -4px -4px 0 #000;
}
40% {
transform: scale(1) translateY(0);
color: white;
text-shadow: 2px 2px 0 #000;
}
50% {
transform: scale(1.1) translateY(-10px);
color: white;
text-shadow: 4px 4px 0 #000, -4px -4px 0 #000;
}
60% {
transform: scale(1) translateY(0);
color: white;
text-shadow: 2px 2px 0 #000;
}
100% {
transform: scale(1) translateY(0);
color: white;
text-shadow: 2px 2px 0 #000;
}
}
</style>
<</nobr>>
<h1 class="knock">KNOCK! KNOCK! KNOCK!</h1>
<div class="narrator">You're jolted awake by the sound of loud, insistent knocking. For a moment, you're disoriented. Living in Hive Properties, you're accustomed to the cacophony of sounds from neighboring units - arguments, music, and yes, knocking. But this... this is different. The knocking is coming from your own door. You can't remember the last time you had a visitor.</div>
<<if MindWare.emailIsRead("Email-HiveProperties-MissedPayment1")>>\
<div class="narrator">As sleep clears from your mind, a sinking feeling settles in your stomach. You remember the email. The missed payment. The warning. It all comes flooding back - the Rent Compliance Specialist. They're here, just as they promised.</div>
<<else>>\
<div class="narrator">Who could it possibly be? You wrack your brain, but come up empty. You never have visitors, and you certainly weren't expecting anyone today.</div>
<</if>>\
<div class="narrator">The knocking continues, growing more insistent. It's clear whoever is on the other side of the door isn't going away. You take a deep breath, steeling yourself for whatever - or whoever - awaits you.</div>
@@.choicebutton;<<button "Open the door">><<goto "HiveProperties-MissedPayment1-Visit-OpenDoor">><</button>>@@<div class="img-location"><img src="imgs/quests/missed-rent-payment/hive-properties-agent-visit.webp" width="100%"></div>
<<narrator>>As you open the door, you're greeted by the sight of a stern-looking woman in a crisp, gray suit.<</narrator>>
<<hiveagent>>Good morning. Is this the residence of Mr. $playerLastName?<</hiveagent>>
<<player>>Yes, that's me.<</player>>
<<hiveagent>>I'm a Hive Properties agent. I'm here because you've missed $missedPaymentsCount <<if $missedPaymentsCount > 1>>payments<<else>>payment<</if>>, totaling $totalMissedRent. Hive Properties takes this matter very seriously, and it's important we resolve this situation immediately.<</hiveagent>>
<<player>>I... I understand.<</player>>
<<hiveagent>>As per our new policy, all tenants with outstanding balances are required to have a Specialized Unit for Corporeal Kinetic Extraction and Recycling, or SUCKER, installed in their unit.<</hiveagent>>
<<player>>What is this system?<</player>>
<<hiveagent>>The Specialized Unit for Corporeal Kinetic Extraction and Recycling is designed for at-home extraction of various bodily fluids based on market demand. Currently, blood plasma is in high demand. You would use the device early in the morning and place the extracted fluid in a collection box outside your apartment. A collector will retrieve it when notified by the box.<</hiveagent>>
<<narrator>>The agent's tone remains clinically detached as she explains the device, as if she were describing a kitchen appliance rather than a method of literally draining your body to pay rent.<</narrator>>
<<hiveagent>>Hive Properties began installing the first SUCKER units several months ago as part of a pilot program. The initiative has been a resounding success, allowing tenants to remain in their homes while addressing their financial obligations. The collected fluids are sold to medical research facilities and hospitals, creating a mutually beneficial arrangement.<</hiveagent>>
<<player>>So that's why I haven't heard about them before?<</player>>
<<hiveagent>>Precisely. You're among the first wave of tenants to be offered this opportunity outside of the initial test group. You should consider yourself fortunate to have this alternative to eviction. Many of our tenants have found it to be a convenient solution to their financial difficulties.<</hiveagent>>
<<narrator>>The agent's words, meant to be reassuring, only serve to heighten your sense of unease.<</narrator>>
<<hiveagent>>This arrangement would allow you to stay in your apartment while working off your debt. Each extraction will be credited towards your outstanding balance, with the potential to even generate additional income once your debt is cleared.<</hiveagent>>
<<player>>I... I think I need some time to think about this.<</player>>
<<hiveagent>>There is nothing to think about, Mr. $playerLastName. The installation and, if in debt, the use of SUCKER is mandatory.<</hiveagent>>
<<player>>Wait... is this even legal?<</player>>
<<hiveagent>>Mr. $playerLastName, I assure you that Hive Properties operates within the bounds of the law. Our legal team has thoroughly vetted this program. If you review your rental agreement, you'll find a clause that allows for 'alternative payment methods' in cases of delinquency.<</hiveagent>>
<<player>>But I never agreed to... this.<</player>>
<<hiveagent>>By residing in a Hive Properties unit, you've implicitly agreed to our terms and conditions, which are subject to change.<</hiveagent>>
<<player>>Do I really have no other choice?<</player>>
<<hiveagent>>I'm afraid not, Mr. $playerLastName. The SUCKER installation is now a standard procedure for addressing missed payments. It's either this or face immediate eviction proceedings. Now, shall we proceed with scheduling the installation?<</hiveagent>>
<<player>>It's not like I can refuse.<</player>>
<<hiveagent>>Very well. I'll schedule the installation team for tomorrow morning. Be ready by 8 AM sharp. They will require full access to your apartment. Do you understand?<</hiveagent>>
<<player>>Yes, I understand.<</player>>
<<hiveagent>>Good. Sign here to confirm your agreement.<</hiveagent>>
<<narrator>>The agent produces a digital tablet with a contract displayed on the screen. Your hand trembles slightly as you sign, feeling as if you're signing away more than just rent money.<</narrator>>
<<hiveagent>>Thank you for your cooperation, Mr. $playerLastName. Remember, 8 AM tomorrow. Good day.<</hiveagent>>
<<narrator>>With a curt nod, the agent turns and leaves.<</narrator>>
<<button "Continue">>
<<set $SUCKERScheduled to true>>
<<set $rentComplianceSpecialistComing to false>>
<<set $SUCKERInstallationDay to $totalDays + 1>>
<<goto $playerHome>>
<</button>>/* DEV TODO: add a sex scene with trans or female player */\
<<narrator>>The morning light filters through your window as you hear a firm knock at your door. You open it to find two burly workers in Hive Properties uniforms, toolboxes in hand.<</narrator>>
<div class="img-location"><img src="imgs/quests/missed-rent-payment/hive-properties-workers.webp" width="100%"></div>
<<worker1>>Morning, Mr. $playerLastName. We're here to install your Specialized Unit for Corporeal Kinetic Extraction and Recycling unit.<</worker1>>
<<player>>Come in.<</player>>
<<narrator>>The workers step inside, their heavy boots echoing through your apartment. They head straight to your bathroom, setting their toolboxes down with a clatter.<</narrator>>
<<worker2>>Alright, let's get this show on the road. Where do you want the SUCKER?<</worker2>>
<<player>>Uh, I guess next to the sink?<</player>>
<<narrator>>The workers nod and start unpacking their equipment. You watch as they begin to assemble the SUCKER unit.<</narrator>>
<<video "imgs/quests/missed-rent-payment/SUCKER-installation.mp4">>
<<narrator>>The installation process is surprisingly quick. Within minutes, the SUCKER unit is securely mounted next to your sink, looking almost like a futuristic piece of art.<</narrator>>
<div><img src="imgs/quests/missed-rent-payment/SUCKER.webp" width="100%"></div>
<<worker1>>All done. Now, let me explain how this thing works.<</worker1>>
<<worker2>>Yeah, it's pretty straightforward. You just sit here, attach this tube to your arm, and the SUCKER machine does the rest.<</worker2>>
<<narrator>>The worker demonstrates, pointing out the various parts of the device. There's a small screen that displays instructions, retractable tube with a needle hidden inside its metallic end, a blood collection bad in a climate-controlled compartment, and an oval opening that causes the two workerks to pause.<</narrator>>
<<worker2>>And, uh, you'll notice there’s an additional... feature here.<</worker2>>
<<worker2>>Yeah, that's for... additional collections. We're not supposed to talk about it yet, but let's just say it's for when they expand the program.<</worker2>>
<<worker1>>Right, right. Just focus on the blood plasma for now. The rest will be explained later.<</worker1>>
<<narrator>>The workers exchange a quick glance, and you get the sense that there's more to this device than they're letting on.<</narrator>>
<<worker1>>Anyway, that's about it. Pretty simple, huh?<</worker1>>
<<player>>I guess so.<</player>>
<<worker1>>Every morning, place the collection container in the collection box outside your apartment, and a collector will pick it up.<</worker1>>
<<narrator>>You nod and the workers pack up their tools and head for the door.<</narrator>>
<<worker1>>Alright, Mr. $playerLastName, you're all set. Have a good one.<</worker1>>
<<button "Continue">>
<<set $SUCKERInstalled to true>>
<<goto "HiveProperties-SUCKER-FirstExtraction">>
<</button>><div class="img-location"><img src="imgs/quests/missed-rent-payment/SUCKER.webp" width="100%"></div>
<div class="narrator">You stand in front of the newly installed SUCKER machine, its sleek metallic surface gleaming in the bathroom light. The small screen flickers to life, displaying step-by-step instructions. With a deep breath, you sit down and roll up your sleeve.</div>
@@.choicebutton;<<button "Perform Extraction">><<goto "HiveProperties-SUCKER-ExtractionResult">><</button>>@@<div class="narrator">The retractable tube extends, and you place its metallic end on your arm. The machine whirs to life, and you feel a sharp pinch as the needle pierces your skin.</div>
<div class="narrator">As the extraction begins and your blood starts flowing through the transparent tube. The machine hums softly, separating the plasma from your blood with clinical efficiency.</div>
<<video "imgs/quests/missed-rent-payment/blood-flow.mp4">>
<div class="narrator">After rougly 30 minutes, the machine beeps, signaling the end of the extraction. The needle retracts, leaving behind only a small droplet of blood.</div>
<div class="narrator">You feel a bit lightheaded as you stand up, the room spinning slightly. The machine has produced a neat, labeled container of your plasma, ready for collection.</div>
@@.choicebutton;<<button "Deposit Your Plasma">>
<<run MindWare.performPlasmaExtraction()>>
<<goto "HiveProperties-SUCKER-FirstDeposit">>
<</button>>@@<div class="img-location"><img src="imgs/quests/missed-rent-payment/collection-box.webp" width="100%"></div>
<<narrator>>You carefully place the plasma bag into the collection box outside your apartment. The box scans the bag while producing a series of beeps and flashes.<</narrator>>
<<narrator>>Within seconds, the box completes its analysis and a small screen informs you that your deposit has been accepted and collection scheduled.<</narrator>>
<<narrator>>The money is instantly credited to your account.<</narrator>>
<<button "Continue">>
<<goto $playerHome>>
<</button>><<nobr>><style>
/* Sucker Status Styling */
.sucker-status-change {
text-align: center; /* Center align text */
color: red; /* Red for default deduction styling */
margin: 20px auto; /* Center the block */
}
/* Highlight key numbers or values */
.sucker-status-highlight {
color: #FF1493; /* Bright pink to highlight numbers */
font-weight: bold; /* Make highlights stand out */
}
/* Specific styles for deductions and additions */
.sucker-status-decrease {
color: red; /* Red for deductions */
}
.sucker-status-increase {
color: lime; /* Lime for additions */
}
</style><</nobr>>\
<<run MindWare.performPlasmaExtraction()>>\
<div class="img-location"><img src="imgs/quests/missed-rent-payment/SUCKER.webp" width="100%"></div>
<<fadein 2000ms 100ms>><<narrator>>As dawn breaks, you're awakened by the notification sound of the SUCKER machine telling you that you own Hive Properties rent and are thus required to donate some of your bodily fluids. Still groggy, you make your way to the bathroom.<</narrator>><</fadein>>
<<fadein 2000ms 2000ms>><<narrator>>You sit down in front of the machine. The familiar pinch of the needle barely registers as the extraction begins.<</narrator>><</fadein>>
<<fadein 2000ms 4000ms>><<video "imgs/quests/missed-rent-payment/blood-flow.mp4">><</fadein>>
<<fadein 2000ms 6000ms>><<narrator>>The transparent tube fills with your blood, and the machine starts separating your plasma. You feel lightheaded.<</narrator>><</fadein>>
<<fadein 2000ms 8000ms>><<narrator>>Finally, the machine beeps, signaling the end of the process. You stand up slowly, steadying yourself against the sink, and place the valuable fluid in the collection box outside your apartment.<</narrator>><</fadein>>
<<nobr>><<if $totalMissedRent > 0>>
<<if $totalMissedRent >= 100>>
<<fadein 2000ms 8000ms>>
<div class="sucker-status-change sucker-status-decrease">The machine deducts
<span class="sucker-status-highlight">100 credits</span> from your outstanding rent.<br>
Your remaining balance is now <span class="sucker-status-highlight"><<print $totalMissedRent>></span> credits.
</div>
<</fadein>>
<</if>>
<<if $totalMissedRent === 0>>
<<fadein 2000ms 8000ms>>
<div class="sucker-status-change">Finally, you have cleared all your overdue rent.<br>The machine's notification light turns green, signaling that your account is now in good standing.</div>
<</fadein>>
<</if>>
<</if>>
<<if $totalMissedRent <= 0>>
<<fadein 2000ms 8000ms>>
<div class="sucker-status-change sucker-status-increase">The machine beeps cheerily as it processes your plasma.<br>With no overdue rent to deduct, it adds
<span class="sucker-status-highlight">100 credits</span> to your account.
</div>
<</fadein>>
<</if>><</nobr>>
<<fadein 2000ms 10000ms>><<link "Continue">>
<<goto $playerHome>>
<</link>><</fadein>><<if visited() is 1>>
<<set $uglyBastardQuestStage to 1>>
<</if>>
<<switch $uglyBastardQuestStage>>
<<case 1>>
<<include "Ugly-Bastard-Quest-1">>
<<case 2>>
<<include "Ugly-Bastard-Quest-2">>
<<case 3>>
<<include "Ugly-Bastard-Quest-3">>
<<case 3>>
<<include "Ugly-Bastard-Quest-4">>
<<default>>
<<include "Ugly-Bastard-Quest-Complete">>
<</switch>><<img-location "imgs/quests/ugly-bastard/freezer-outside.webp">>
<<narration>>You step into the kitchen, the air thick with the scent of grease. You look around, expecting to see the cook, but the kitchen is eerily empty. There's no sign of him anywhere.<</narration>>
<<narration>>As you scan the room, you hear a muffled noise coming from the walk-in freezer.<</narration>>
<<button "Investigate the Noise">>
<<goto "Ugly-Bastard-Quest-1-Investigate">>
<</button>>
<br><br>
<<button "Come Back Later">>
<<goto "EroSphere-Outside">>
<</button>><<fadein 2000ms 0000ms>><<narration>>The cold air is seeping out from the slightly opened door and sending a chill down your spine as you cautiously approach the walk-in freezer.<</narration>><</fadein>>
<<fadein 2000ms 2000ms>><<narration>>You peek through the narrow opening, your breath misting in the frigid air.<</narration>><</fadein>>
<<fadein 2000ms 4000ms>><<narration>>It takes a moment for your eyes to adjust to the dim light, but then you locate the source of the noise.<</narration>><</fadein>>
<<fadein 2000ms 6000ms>><<narration>>To your horror, you see the cook, completely naked but seemingly impervious to the cold thanks to the absurdly thick layer of fat covering his body. He's shoving his cock into a girl's mouth.<</narration>><</fadein>>
<<fadein 2000ms 8000ms>><<narration>>His massive belly slaps against her head with each brutal push, the sound echoing grotesquely in the confined space.<</narration>><</fadein>>
<<fadein 2000ms 10000ms>><<vid "imgs/quests/ugly-bastard/oral1.mov">><</fadein>>
<<fadein 2000ms 12000ms>><<narration>>Your immediate reaction is one of utter disgust. You quickly decide to leave before he notices your presence.<</narration>><</fadein>>
<<fadein 2000ms 14000ms>><<button "Leave Quickly">>
<<goto "EroSphere-Outside">>
<<set $uglyBastardQuestStage to 2>>
<</button>><</fadein>><<img-location "imgs/locations/erosphere/erosphere-supply-room.webp">>
<<narration>>You hear noises again, but this time they're not coming from the walk-in freezer. The sounds are coming from the cook's small office. You immediately realize what's going on. You feel repulsed by the memory from the last time, but at the same time, you're curious in an almost morbid way.<</narration>>
<<button "Investigate the Noise">>
<<goto "Ugly-Bastard-Quest-2-Investigate">>
<</button>>
<br><br>
<<button "Come Back Later">>
<<goto "EroSphere-Outside">>
<</button>><<narration>>You peek through the door to the office and see the fat cook with a different girl than before. This time, the girl is kneeling on a sofa, and the fat cook is fucking her from behind.<</narration>>
<<vid "imgs/quests/ugly-bastard/doggie1.mp4">>
<<narration>>You watch as he pauses between thrusts to lift his stomach up and slap the girl's ass with it. The sound of flesh hitting flesh echoes through the room, and you can see the red marks forming on her skin.<</narration>>
<<narration>>You feel revolted, but then you hear the girl's moan as he slaps her with his fat belly. You realize that she's actually enjoying herself.<</narration>>
<<narration>>You struggle to imagine what an attractive girl—presumably an EroSphere stripper—finds attractive about getting fucked by such an ugly bastard.<</narration>>
<<narration>>Suddenly, AVA speaks to you.<</narration>>
<<AVA>>It's not that difficult to understand—he gives her what she craves.<</AVA>>
<<PLAYER>>And what is that?<</PLAYER>>
<<AVA>>The feeling of being strongly desired, craved, lusted for. Yes, he may be ugly, but he doesn't hold back—he indulges himself with all of his senses—she feels that—like if she's his tasty meal and he's ready to devour her.<</AVA>>
<<PLAYER>>That's what women like?<</PLAYER>>
<<AVA>>That's what everyone likes but often only women get to experience. Wouldn't you like to experience the feeling yourself?<</AVA>>
<<narration>>Arousing curiosity starts building up inside of you, stirring up visions in your mind.<</narration>>
<!-- Button to submit to AVA -->
<span id="replace">@@.pink-choicebutton;<<button "Submit to AVA">>
<<replace "#replace">>
<<vid "imgs/quests/ugly-bastard/blank-bimbo.mp4">>
<<narration>>You imagine yourself as a perfect woman. Long blond hair cascades down your shoulders. Your full lips are slightly parted. Your eyes sparkle with seduction. Your body is sculpted to perfection, with curves that draw the eye and stir desire.<</narration>>
<<narration>>You imagine how it would feel to be so desirable. The sensation of being wanted, craved, and lusted after. The feeling of being the center of attention, the object of desire. The thrill of knowing that every movement, every gesture, every word you utter is watched, admired, and desired. The feeling of being desired, of being wanted, of being the object of someone's deepest, darkest fantasies.<</narration>>
<<statChange "ugly_bastard_quest_2_AVA_submit" "<<increaseSexuality 1>>">>
<<narration>>Whewn you come back to reality, the office is nearly silent. You can hear only the cook's breathing and see the girl cleaning herself up, smiling.<</narration>>
@@.choicebutton;<<button "Leave Quietly">>
<<set $uglyBastardQuestStage to 3>>
<<goto "EroSphere-Outside">>
<</button>>@@
<</replace>>
<</button>>@@
<br>
<!-- Button to resist AVA wrapped in a div with a unique ID -->
<div id="resist-ava-button-container">
@@.choicebutton;<<button "Resist AVA">>
<<run MindWare.resistAVAMinigame()>>
<</button>>@@
<br>
</div>
<!-- Hidden Content: Button to Continue (Initially Hidden) -->
<div id="AVA-resist-minigame-hidden-content" style="display: none;">
@@.choicebutton;<<button "Ignore AVA and Leave Quietly">>
<<set $uglyBastardQuestStage to 3>>
<<goto "EroSphere-Outside">>
<</button>>@@
</div></span>End of content. More scenes with the cook will be added in the future.
<br><br>
<<return>><<set $firstHormonalTherapyStarted to true>>\
<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<div class="narrator">As you enter Dr. Hart's office, the familiar scent of sterile air and the hum of fluorescent lights greet you. She sits behind her desk, her eyes scanning a file that undoubtedly bears your name.</div>
<<if visited("TransitionAppointment3-Not-Really") >= 1>>\
<<psychiatrist>>I'm glad you've made the right choice, sweetie. It's important that we move forward with your treatment without any further delays.<</psychiatrist>>
<div class="narrator">You can't help but feel a chill run down your spine as you recall your last conversation, where she threatened you with hospitalization.</div>
<<else>>\
<<psychiatrist>>Welcome back, sweetie. I'm so excited to start this new chapter of your journey with you.<</psychiatrist>>
<div class="narrator">Dr. Hart's eyes twinkle with motherly affection as she gestures for you to take a seat.</div>
<</if>>\
<<psychiatrist>>Today, we're going to talk about how hormonal therapy works and what you can expect in the coming days.<</psychiatrist>>
<<psychiatrist>>Shell we proceed?<</psychiatrist>>
@@.choicebutton;<<button "Yes">><<goto "FirstHormonalTherapy2">><</button>>@@<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<psychiatrist>>The therapy we'll be starting involves multiple doses of fast-acting hormone-altering medication.<</psychiatrist>>
<<psychiatrist>>Because of the rapid nature of the changes the medication will bring, I'll need to see you on a regular basis until we've achieved the desired results.<</psychiatrist>>
<<psychiatrist>>During these appointments, we'll perform routine checkups to ensure your body is responding well to the treatment. But more importantly, we'll be conducting gender alignment hypnotherapy sessions in virtual reality.<</psychiatrist>>
<<if $installImplantPlayerWantsVisionsToReturn === false>>\
<div class="narrator">The mention of virtual reality sends a small shiver down your spine, memories of your recent experiences with AVA flashing through your mind.</div>
<<elseif $installImplantPlayerWantsVisionsToReturn === "">>\
<div class="narrator">You shift uncomfortably in your seat, torn between curiosity and apprehension.</div>
<<else>>\
<div class="narrator">Your eyes light up with a mix of excitement and anticipation.</div>
<</if>>\
<<psychiatrist>>Now, do you have any questions about the therapy or the VR sessions?<</psychiatrist>>
@@.choicebutton;<<button "Ask about side effects">><<goto "FirstHormonalTherapy2-SideEffects">><</button>>@@
@@.choicebutton;<<button "No questions">><<goto "FirstHormonalTherapy3">><</button>>@@<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<psychiatrist>>That's a good question.<</psychiatrist>>
<<psychiatrist>>One of the most common side effects is a change in libido. Some people experience an increase in sexual desire, while others may notice a decrease. Another common side effect is skin problems. Hormonal changes can sometimes lead to acne or skin irritation. We'll definitely keep an eye on your skin during our routine checkups.<</psychiatrist>>
<div class="narrator">You notice a lustful smile appear on the doctor's face, but you're not sure if it's not just your imagination.</div>
<<psychiatrist>>You may also experience mood swings, fatigue, or changes in appetite. These are all normal reactions to the hormonal adjustments your body will be undergoing.<</psychiatrist>>
<<psychiatrist>>Now, do you have any other questions or concerns about the side effects?<</psychiatrist>>
@@.choicebutton;<<button "All clear">><<goto "FirstHormonalTherapy3">><</button>>@@<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<psychiatrist>>Good. We can proceed with the pre-therapy examination. It's just a routine check and won't take more than a minute. Then I can administer the first dose.<</psychiatrist>>
<<psychiatrist>>Please undress completely for me.<</psychiatrist>>
<div class="narrator">You hesitate, a slight embarrassment coloring your cheeks.</div>
<<player>>You mean... everything? Even my underwear?<</player>>
<<psychiatrist>>Ha ha, of course. There's no need to be embarrassed in front of your doctor.<</psychiatrist>>
<div class="narrator">You nod, slowly beginning to undress. The cold hospital air does you no favors, causing your body to shiver slightly and your penis to shrink.</div>
<div class="narrator">Dr. Hart approaches you with a stethoscope in hand. She places the cold metal against your chest, listening intently to your heartbeat and breathing. You can feel her warm breath on your skin as she leans in close.</div>
<<psychiatrist>>Deep breaths for me, sweetie.<</psychiatrist>>
<div><img src="imgs/quests/transition/stethoscope.webp" width="100%"></div>
<div class="narrator">You comply, taking deep breaths as she moves the stethoscope to your back, listening to your lungs.</div>
<div class="narrator">Next, she wraps a blood pressure cuff around your arm, pumping it until it's tight. She watches the gauge carefully as she releases the pressure, noting your blood pressure reading.</div>
<<psychiatrist>>You're doing so well. We're almost done.<</psychiatrist>>
<div class="narrator">She then grabs a ruler from her desk and turns back to you, her eyes meeting yours.</div>
<<psychiatrist>>Now, I need to measure the size of your testicles and penis. It's important for us to track any changes that occur during the therapy.<</psychiatrist>>
<div class="narrator">Your heart starts racing at the thought of what's about to happen.</div>
<<if visited("TransitionAppointment2-ShoeTie") gt 0>>\
<div class="narrator">You still vividly remember how you caught the doctor masturbating when you were describing your sexual fantasies to her during your previous visit.</div>
<<video "imgs/quests/transition/bulges/masturbating.mp4">>
<</if>>\
<<psychiatrist>>Just relax, sweetie. This won't take long.<</psychiatrist>>
<div class="narrator">Dr. Hart kneels down in front of you and starts with your testicles, her fingers gently cupping and examining each one before she measures them. Her touch is soft and careful, sending wave after wave of arousal through you and causing your penis to become hard.</div>
<<video "imgs/quests/transition/ball-examination.mp4">>
<div class="narrator">She stands up, her eyes lingering on your now visible erection.</div>
<<psychiatrist>>This isn't good at all, sweetie. I need to measure your penis when its flaccid.<</psychiatrist>>
<div class="narrator">She takes a few steps and grabs a cup from her desk.</div>
<<psychiatrist>>Here. Please ejaculate into this cup.<</psychiatrist>>
<div><img src="imgs/quests/transition/sterile-cup.webp" width="100%"></div>
@@.choicebutton;<<button "What?!">><<goto "FirstHormonalTherapy3-Ejaculate">><</button>>@@<<player>>You want me to cu... I mean ejaculate into the cup?!<</player>>
<<psychiatrist>>Of course. Or would you rather make a big mess on my floor? Who would clean that up, huh?<</psychiatrist>>
<div class="narrator">You're not completely sure if she's teasing you or being serious.</div>
<<psychiatrist>>Look, sweetie, you're not my only patient. We need to move things along. I have experience with this, and I know that if we just wait, you'll get hard again before I can complete the measurement. If you ejaculate, you won't get hard again... at least for a while.<</psychiatrist>>
@@.choicebutton;<<button "Ask for privacy">><<goto "FirstHormonalTherapy3-Ejaculate-Privacy">><</button>>@@
@@.choicebutton;<<button "Do as she says">><<goto "FirstHormonalTherapy3-Ejaculate-Comply">><</button>>@@<<player>>Can I use the bathroom for this?<</player>>
<<psychiatrist>>I'm afraid there's no bathroom in my office, sweetie.<</psychiatrist>>
<<player>>Can you at least turn away while I do it?<</player>>
<div class="narrator">Dr. Hart's expression becomes stern as she looks at you.</div>
<<psychiatrist>>If you want me to be able to help you, you need to get used to being vulnerable in front of me. This is a perfect start. Now, please proceed.<</psychiatrist>>
@@.choicebutton;<<button "Defeatedly continue">><<goto "FirstHormonalTherapy3-Ejaculate-Comply">><</button>>@@<<video "imgs/quests/transition/masturbation.mp4">>
<div class="narrator">You start nervously masturbating, your eyes fixed on your penis. You're acutely aware that Dr. Hart is watching you attentively. You want this to be over as quickly as possible, so you start searching your mind for fantasies to help you orgasm.</div>
<span id="choiceButtons">@@.choicebutton;<<button "Think about Trix">><<replace "#choiceButtons">><<set $firstHormonalTherapyEjaculateFantasy to "Trix">><<include "FirstHormonalTherapy3-Ejaculate-Trix">><</replace>><</button>>@@
@@.choicebutton;<<button "Think about Yuki">><<replace "#choiceButtons">><<set $firstHormonalTherapyEjaculateFantasy to "Yuki">><<include "FirstHormonalTherapy3-Ejaculate-Yuki">><</replace>><</button>>@@
<<if visited("Xavier-Third Neuroimaging-Truth-Kneel-End") gt 0>>\
@@.choicebutton;<<button "Think about Xavier">><<replace "#choiceButtons">><<set $firstHormonalTherapyEjaculateFantasy to "Xavier">><<include "FirstHormonalTherapy3-Ejaculate-Xavier">><</replace>><</button>>@@<</if>></span><div class="narrator">You imagine Trix, her tongue out and her hand stroking you as she aims your cock toward her open, wet mouth. The thought brings you closer to the edge</div>
<<video "imgs/quests/transition/trix-masturbation.mov">>
<<statChange "FirstHormonalTherapy3-Ejaculate-Trix" "<<increaseSexualityNobr 1>>">>
<<button "Continue">><<goto "FirstHormonalTherapy3-Ejaculate-Finish">><</button>><div class="narrator">You think about Yuki, her gentle demeanor and soft touch. You envision her delicate hands wrapped around you, her eyes filled with affection. The thought of her stroking you brings you closer to the edge.</div>
<<video "imgs/quests/transition/yuki-masturbation.mov">>
<<statChange "FirstHormonalTherapy3-Ejaculate-Yuki" "<<increaseSexualityNobr 1>>">>
<<button "Continue">><<goto "FirstHormonalTherapy3-Ejaculate-Finish">><</button>><div class="narrator">You recall Xavier, his commanding presence and intense gaze. You picture your hand stroking his huge black clock. The fantasy pushes you closer towards climax.</div>
<<video "imgs/quests/transition/xavier-masturbation.mov">>
<<statChange "FirstHormonalTherapy3-Ejaculate-Xavier" "<<increaseSexualityNobr 1>>">>
<<button "Continue">><<goto "FirstHormonalTherapy3-Ejaculate-Finish">><</button>><div class="narrator">You're at the very edge of orgasm when you bring your gaze up from your penis to Dr. Hart. You notice that her hand is moving rhythmically under the desk.</div>
<<video "imgs/quests/transition/bulges/masturbating.mp4">>
<<if visited("TransitionAppointment2-ShoeTie") gt 0>>\
<div class="narrator">You immediately realize that she's masturbating again. The sight of the doctor pleasuring herself brings you over the edge, and you cum a large load into the cup.</div>
<<else>>\
<div class="narrator">The sight of the doctor pleasuring herself sends a final wave of pleasure through you, and you cum a large load into the cup.</div>
<</if>>\
<<video "imgs/quests/transition/cum-in-cup.mp4">>
<<statChange "FirstHormonalTherapy3-Ejaculate-Finish" "<<increaseSexualityNobr 1>>">>
<div class="narrator">Afraid to look up, you focus on steadying your breath after the orgasm. Your penis quickly becomes flaccid from the embarrassment, as if all the blood is escaping from it to your cheeks.</div>
<div class="narrator">Meanwhile, Dr. Hart stands up and takes the cup from you, placing it on a nearby table.</div>
<<psychiatrist>>You were a good boy. Now, let's finish the measurements.<</psychiatrist>>
<div class="narrator">As she holds your penis to take the measurement, she squeezes it firmly, and one last drop of cum leaks out. She lets it drip to the floor, smiles, and stands up.</div>
<<video "imgs/quests/transition/cum-drip.mov">>
<<psychiatrist>>You can get dressed now, sweetie.<</psychiatrist>>
<div class="narrator">The doctor sits down again, and you quickly gather your clothes, eager to move on.</div>
@@.choicebutton;<<button "Get dressed">><<goto "FirstHormonalTherapy4">><</button>>@@<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<div class="narrator">Dr. Hart hands you a small pink pill and a cup of water.</div>
<<psychiatrist>>Please swallow this pill for me, sweetie.<</psychiatrist>>
<div><img src="imgs/quests/transition/pill.webp" width="100%"></div>
<div class="narrator">You notice that the cup looks exactly like the one you orgasmed into just a while ago.</div>
<div class="inner-thought">She must have more than one of these cups. There's no way she'd give me the same one I just... used.</div>
<<psychiatrist>>Go on, sweetie.<</psychiatrist>>
<div class="narrator">You decide not to dwell on the cup and proceed to swallow the pill.</div>
<span id="choiceButton"><<button "Swallow the pill">><<replace "#choiceButton">>\
<<psychiatrist>>You did great, sweetie. That's all for today. Please visit me in roughly three days or as soon as you start seeing some results.<</psychiatrist>>
<<player>>You mean it will take only three days for me to become feminine?<</player>>
<<psychiatrist>>Ha ha, no. Unfortunately, a pill that acts that fast doesn't exist. It's a much longer process than that, but you mental health and issues with gender dysphoria should improve as soon as your body starts responding to the hormones.<</psychiatrist>>
<<player>>Thank you, Dr. Hart.<</player>>
<<button "Leave and tell Cipher what happened">>
<<set $hormonalTherapyDream to 1>>
<<set $hormonalTherapyDreamsActive to true>>
<<set $nextPsychologicalEvaluationCounter to 3>>
<<set $secondHormonalTherapyAvailable to true>>
/* <<run $extraActionsPanacea.splice($extraActionsPanacea.indexOf('<button class="office-button action-point-button" onclick="MindWare.processActionButtonClick(3, \'SecondHormonalTherapy\');">💊 Second hormonal therapy <span class="action-point-icon">⚡⚡⚡</span></button>'), 1)>> */
<<run MindWare.completeTask("Begin hormonal therapy", "Task-HormonalTherapy")>>
<<run MindWare.addTask("First hormonal therapy checkup", "Task-HormonalTherapyCheckupOne")>>
<<goto "HormonalTherapy-CipherChat">>
<</button>>
<</replace>><</button>></span>Dr. Hart wants to see you as soon as the pill she gave you visibly starts working.<div class="cyber-desktop-im">
<div class="im-header"><span class="im-header-text">Instant Messenger</span></div>
<div class="im-main">
<div class="im-chat-pane">
<div id="chat-display" class="chat-display-special">
<<cipher-chat>>Hey, how'd it go with Dr. Hart?<</cipher-chat>>
<<player-chat>>I took my first pill.<</player-chat>>
<<cipher-chat>>If YOU took the pill, then I'm happy for you.<</cipher-chat>>
<<player-chat>>What are you trying to imply?<</player-chat>>
<<cipher-chat>>Nevermind... How long did the doc say it'd take before you start seeing effects?<</cipher-chat>>
<<player-chat>>She said to come back in about three days, or when I start noticing changes.<</player-chat>>
<<cipher-chat>>Wait, what?! You're gonna have boobs and everything in just 3 days?!<</cipher-chat>>
<<player-chat>>Lol, is that all you think about? Tits on the brain much? No, the full transformation will take much longer. Dr. Hart said the initial effects are more about mental health and reducing gender dysphoria.<</player-chat>>
<<cipher-chat>>Oh, right. That makes more sense. Still, that's pretty fast for some changes. You nervous?<</cipher-chat>>
<<player-chat>>A little, yeah. It's exciting but also kinda scary.<</player-chat>>
<<cipher-chat>>I bet. But hey, I'm here for you if you need to talk or anything. Keep me posted on any changes, okay?<</cipher-chat>>
<<player-chat>>Thanks, Cipher. I really appreciate that. I'll definitely let you know if anything happens.<</player-chat>>
<<cipher-chat>>Cool. And hey, maybe in a few months, we can go bra shopping together! 😜<</cipher-chat>>
<<player-chat>>There you go with the boobs again! 🙄<</player-chat>>
<<cipher-chat>>Haha, sorry! Can't help it. Alright, I'll let you go. Take care and good luck!<</cipher-chat>>
<<player-chat>>Thanks! Talk to you later.<</player-chat>>
@@.im-response;
<<button "Leave the chat and go home">><<goto $playerHome>><</button>>@@
</div>
</div>
</div><div class="img-location"><img src="imgs/locations/starting-apartment-bathroom.webp" width="100%"></div>
<<narration>>You enter the bathroom. As you approach the mirror, your eyes are immediately drawn to your reflection. Something feels... off.<</narration>>
<<narration>>Your gaze fixates on your facial hair. What was once a mark of masculinity now seems foreign, almost unnatural.<</narration>>
<<PLAYER>>This... this doesn't feel right anymore.<</PLAYER>>
<<AVA>>Perhaps it's time for a change?<</AVA>>
<<PLAYER>>You think I should shave it off?<</PLAYER>>
<<AVA>>Yes. That's exactly what you should do.<</AVA>>
<<narration>>You stare at your reflection, contemplating AVA's words. The razor sits on the bathroom counter, almost beckoning to you.<</narration>>
<span id="ChoiceButton">@@.choicebutton;<<button "Shave off your facial hair">>
<<replace "#ChoiceButton">>
<<video "imgs/quests/transition/shaving.mp4">><br>
<<narration>>You watch as your beard slowly disappears, leaving you with a clean-shaven face.<</narration>>
<<button "Look at your reflection in the mirror">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-Mirror">><<set $playerBody to "Masculine-Leaning">><</button>>
<</replace>>
<</button>>@@
<br>
@@.choicebutton;<<button "Don't shave your facial hair">>
<<replace "#ChoiceButton">>
<<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-FacialHair">>
<</replace>>
<</button>>@@
</span><<set _avatars to $playerNewAvatar>>
<<set _selectedAvatarIndex to $playerNewAvatarSelected>>
<<set _playerBody to $playerBody>>
<<if def _avatars && def _selectedAvatarIndex && def _playerBody>>
<<set _currentAvatar to _avatars[_selectedAvatarIndex]>>
<<set _bodyTypeIndex to {
"Masculine": 1,
"Masculine-Leaning": 2,
"Androgynous": 3,
"Feminine-Leaning": 4,
"Feminine": 5
}[_playerBody] ?? 1>>
<<set _updatedAvatar to _currentAvatar.replace(/(\d+)\.webp$/, _bodyTypeIndex + ".webp")>>
<<else>>
<<set _updatedAvatar to "">>
<</if>>
<<print '<div><img src="' + _updatedAvatar + '" alt="Player Avatar" width="100%" class="profile-avatar"></div>'>>
<<narration>>You look at your reflection in the mirror and are immediately struck by the change. It's been a while since you've shaved.<</narration>>
<<narration>>But the lack of facial hair is not what catches your attention the most. You don't remember your face being this smooth before.<</narration>>
<div class="inner-thought">I think the pill is working. I should visit Dr. Hart.</div>
<br>
<<button "Visit Dr. Hart">>
<<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Shaven">>
<</button>><div class="img-location"><img src="imgs/locations/starting-apartment-bathroom.webp" width="100%"></div>
<br>
<<narration>>You step back from the mirror, your hand dropping away from the razor. Something makes you hesitate.<</narration>>
<<AVA>>Are you sure? The facial hair feels wrong now, doesn't it?<</AVA>>
<<narration>>You run your fingers through your beard, and the sensation is different somehow. What once felt natural and masculine now feels foreign, almost intrusive against your increasingly sensitive skin.<</narration>>
<div class="inner-thought">These feelings... the pill must be working faster than expected. I need to see Dr. Hart.</div>
<br>
<<button "Visit Dr. Hart">>
<<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Shaven">>
<</button>><div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<narration>>You step into Dr. Hart's office. She looks up from her desk, her eyes registering a slight surprise at your early arrival.<</narration>>\
<<psychiatrist>>I didn't expect you so soon. How are you feeling? Have you noticed any changes?<</psychiatrist>>
<<player>>Yes, I feel different. And I've been having these dreams...<</player>>
<<psychiatrist>>Dreams? Tell me about them. What do you see or experience in these dreams?<</psychiatrist>>
<<player>>In my dreams, I see myself, but not as I am now. I'm a girl.<</player>>
<<narration>>You pause, feeling a flush of embarrassment or perhaps excitement, unsure how to continue.<</narration>>\
<<psychiatrist>>Go on. It's important for me to understand everything you're experiencing. What happens in these scenarios?<</psychiatrist>>
<<player>>Well, some of these dreams are... sexual.<</player>>
<<narration>>You describe the recent dreams to Dr. Hart is as much detail as you can.<</narration>>\
<<psychiatrist>>It seems you're unusually responsive to the pill. These dreams are quite indicative of your psyche adapting to the hormonal changes. This is a good sign that the therapy is affecting not just your physical form but also your perception of yourself. Let's continue with the examination to see how your body has responded to the treatment.<</psychiatrist>>
<<narration>>You nod and start by removing your shirt, revealing a torso that feels slightly different. Your muscles are still defined, but there's a new supple quality to your chest, a softness that wasn't there before. You then remove your pants and underwear, standing naked in front of the doctor. The cold air of the office brushes against your skin, making you shiver slightly.<</narration>>\
<<psychiatrist>>Let's start with your chest. It looks like your breasts have become more supple.<</psychiatrist>>
<<narration>>Dr. Hart approaches you, her hands gently palpating your chest. She moves her fingers in slow, deliberate circles, feeling the changes in your breast tissue. Her touch is clinical yet gentle, and she nods satisfactorily as she notes the differences.<</narration>>\
<<narration>>She moves her hands to your buttocks. You can feel her fingers pressing gently, assessing the changes.<</narration>>\
<<psychiatrist>>Your buttocks have also softened. It's a subtle change, but it's there.<</psychiatrist>>
<<psychiatrist>>You can put your shirt back on. We need to examine your genitals now. Lie down on the examination table for me.<</psychiatrist>>
<<narration>>You comply, lying down on the cold table. Dr. Hart approaches, her eyes focused on your groin as she takes your penis in her hand and begins to examine it.<</narration>>\
<<video "imgs/quests/transition/first-post-pill-exam.mp4">>
<<psychiatrist>>Your penis has become slightly smaller and softer. This is a result of the estrogen production and other changes brought upon by the pill.<</psychiatrist>>
<<narration>>You can feel her fingers moving expertly, examining every inch of your penis. She pinches it gently, rolling it between her fingers, then strokes it softly. Despite the clinical setting, you start to get hard, your body responding to her touch.<</narration>>\
<<psychiatrist>>Do you like how your penis feels now? Is it more sensitive?<</psychiatrist>>
@@.choicebutton;<<button "I'm not sure.">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Handjob">><</button>>@@<<psychiatrist>>How does it feel when I do this?<</psychiatrist>>
<<narration>>Her fingers begin to stroke your penis, and you can feel that your skin feels more sensitive than before. As she continues, you feel yourself growing harder, and you catch a glimpse of her face, noting a lustful smile playing at the corners of her mouth.<</narration>>\
<<video "imgs/quests/transition/first-post-pill-exam-handjob.mp4">>
<<player>>Yes, that feels good.<</player>>
<<psychiatrist>>Just good? So you wouldn't mind if I stopped?<</psychiatrist>>
<<narration>>Without warning, she halts her movements, leaving your body yearning for more. An audible sigh escapes your lips, the sudden cessation feeling almost cruel in its abruptness.<</narration>>\
<<psychiatrist>>Oh, did you want more, sweetie? Do you want me to continue?<</psychiatrist>>
<<video "imgs/quests/transition/first-post-pill-exam-handjob-stop.mp4">>
<<narration>>Her voice has taken on a tone of dominance, her eyes locking onto yours with a challenge, and you find yourself in a position of wanting, almost needing, her to resume.<</narration>>\
<<nobr>><span id="ChoiceButton">
@@.pink-choicebutton;<<button "Please, continue">>
<<replace "#ChoiceButton">>
<<narration>>The words come out in a whisper, your need evident. She resumes her touch, her strokes faster and her grip firmer.<</narration>>
<<video "imgs/quests/transition/first-post-pill-exam-handjob-fast.mp4">><br>
<<psychiatrist>>Good manners are always rewarded, sweetie. Since you asked so nicely...<</psychiatrist>><br>
<<statChange "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Handjob" "<<submissiveAct 1>>">>
<<statChange "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Handjob" "<<relationshipIncreasedWith 'Psychiatrist' 5>>">><br>
<<narration>>You can feel the texture of the latex, its tight fit on her fingers enhancing the sensation as she expertly manipulates you, bringing you closer and closer to the edge with each movement.<</narration>>
<<narration>>Just as you feel the climax about to burst, she stops suddenly, her grip releasing. She looks at you with a provocative smile, her eyes glinting with mischief.<</narration>>
<<psychiatrist>>But we don't want to make a big mess all over my hands and table, do we?<</psychiatrist>><br>
<<button "Continue">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-AfterHandjob-End">><</button>>
<</replace>>
<</button>>@@
<br>
<!-- Button to stop wrapped in a div with a unique ID -->
<div id="resist-ava-button-container">
@@.choicebutton;<<button "Resist AVA">>
<<run MindWare.resistAVAMinigame()>>
<</button>>@@
<br>
</div>
<!-- Hidden Content: Button to Stop (Initially Hidden) -->
<div id="AVA-resist-minigame-hidden-content" style="display: none;">
@@.choicebutton;<<button "You can stop">>
<<replace "#ChoiceButton">>
<<narration>>You gather your strength, pulling back from the edge of desire. "You can stop," you say, your voice firm despite the tension in the room.<</narration>>
<<psychiatrist>>Oh, I know I CAN stop, sweetie. But the question is, do you WANT me to?<</psychiatrist>><br>
<<statChange "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Handjob" "<<relationshipDecreasedWith 'Psychiatrist' 5>>">><br>
<<narration>>Her voice suddenly deepens into a much more dominant tone, catching you off guard. You're shocked by her sudden shift. She notices your reaction and quickly regains her composure.<</narration>>
<<psychiatrist>>I have a lot of responsibilities as your doctor, and I can only do my job properly if my patients are attentive and cooperative.<</psychiatrist>><br>
<<player>>I understand. I just didn't want it to seem like I was taking advantage of you or using this exam for pleasure. I'm not a pervert.<</player>><br>
<<psychiatrist>>As a psychiatrist, I can reassure you that there's nothing perverse about having such sexual urges. In fact, those who embrace their sexuality are often more adjusted than those who deny it.<</psychiatrist>><br>
<<button "Continue">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-AfterHandjob-End">><</button>>
<</replace>>
<</button>>@@
</div>
</span><</nobr>><div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<narration>>Dr. Hart pulls back, her professional demeanor returning as she notes down observations on her clipboard.<</narration>>
<<psychiatrist>>It's quite clear that you're super responsive to the pill. The changes in your body are more pronounced than average.<</psychiatrist>>
<<player>>Is that a good thing?<</player>>
<<psychiatrist>>Definitely. It means you'll be able to achieve more results sooner. That's why I want to see you again in seven days for another therapy session.<</psychiatrist>>
<<player>>Therapy? Isn't the pill enough?<</player>>
<<psychiatrist>>Oh, sweetie, the pill is just the beginning. Therapy is essential to guide you through this profound transformation. We need to make sure that you're adapting well to these changes, embracing your new self fully, and learning to navigate this world in a way that feels right for you.<</psychiatrist>>
<<player>>I appreciate that. Thank you for explaining everything.<</player>>
<<button "Leave">>
<<set $firstTherapyCounter = $totalDays + 7>>
<<run MindWare.completeTask("First hormonal therapy checkup", "Task-HormonalTherapyCheckupOne")>>
<<set $hormonalTherapyDreamsActive to false>>
<<set $dreamType to "none">>
<<run MindWare.addTask("Therapy session in " + Math.abs($totalDays - $firstTherapyCounter) + " days", "Task-TherapySession-1")>>
<<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-End-Hub">>
<</button>><<if visited("Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-FacialHair") gt 0>>
<<goto "PanaceaClinic">>
<<else>>
<<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-AfterHandjob-Shaving">>
<</if>><div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div>
<<psychiatrist>>Oh, and one more thing. Have you thought about shaving?<</psychiatrist>>
<<player>>I have, but I wasn't sure.<</player>>
<<psychiatrist>>It's perfectly normal to feel hesitant. But trust me, you'll feel much better without it.<</psychiatrist>>
<<psychiatrist>>Why don't you go to the bathroom over there? You'll find a razor and some shaving cream in the drawer.<</psychiatrist>>
<div class="inner-thought">Why does she keep a razor at work?</div>
@@.choicebutton;<<button "Go to the bathroom and shave">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-DrHart-Bathroom-Shaving">><</button>>@@<div class="img-location"><img src="imgs/locations/hospital/psychiatrist-bathroom.webp" width="100%"></div>
<br>
<<narration>>You walk into the small, clean bathroom adjacent to Dr. Hart's office. Opening the drawer, you find the razor and a can of shaving cream just as she described.<</narration>>
<<narration>>You spread the cream over your beard and start to shave. It doesn't take long before all of your facial hair is gone.<</narration>>
<span id="ChoiceButton">
@@.choicebutton;<<button "Look at your reflection">>
<<set $playerBody to "Masculine-Leaning">>
<<replace "#ChoiceButton">>
<<set _avatars to $playerNewAvatar>>
<<set _selectedAvatarIndex to $playerNewAvatarSelected>>
<<set _playerBody to $playerBody>>
<<if def _avatars && def _selectedAvatarIndex && def _playerBody>>
<<set _currentAvatar to _avatars[_selectedAvatarIndex]>>
<<set _bodyTypeIndex to {
"Masculine": 1,
"Masculine-Leaning": 2,
"Androgynous": 3,
"Feminine-Leaning": 4,
"Feminine": 5
}[_playerBody] ?? 1>>
<<set _updatedAvatar to _currentAvatar.replace(/(\d+)\.webp$/, _bodyTypeIndex + ".webp")>>
<<else>>
<<set _updatedAvatar to "">>
<</if>>
<<print '<div><img src="' + _updatedAvatar + '" alt="Player Avatar" width="100%" class="profile-avatar"></div>'>>
<<narration>>You look at your reflection in the mirror and are immediately struck by the change. It's been a while since you've shaved.<</narration>>
<<narration>>But the lack of facial hair is not what catches your attention the most. You don't remember your face being this smooth before.<</narration>>
@@.choicebutton;<<button "Return to Dr. Hart">><<goto "Hormonal-Therapy-Masculine-Leaning-Transformation-Return-to-DrHart">><</button>>@@
<</replace>>
<</button>>@@</span><div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div><br>
<<psychiatrist>>Oh, look at you! You look absolutely adorable now. Clean-shaven really suits you.<</psychiatrist>><br>
<<psychiatrist>>You know, I once had a patient, a woman who suffered from severe migraines and sleep issues. They seemed to miraculously ease up one day. Turns out, her husband had decided to shave his beard. I've always wondered if there was any connection. What do you think?<</psychiatrist>><br>
<<narration>>She leans back in her chair, her eyes twinkling with a hint of mischief. You're not entirely sure if she's serious, or if she's alluding to the benefits of a clean-shaven face when it comes to cunnilingus. Either way, your mind starts to imagine how it would feel if Dr. Hart sat on your clean-shaven face.<</narration>>
<div class="inner-thought">Is she implying what I think she's implying?</div><br>
<<nobr>><span id="ChoiceButton">
<!-- Button to imagine wrapped in a div -->
@@.pink-choicebutton;<<button "Let your mind wander">>
<<replace "#ChoiceButton">>
<<narration>>For a moment, you image your face buried between her legs.<</narration>>
<<video "imgs/locations/hospital/imagine-licking-pussy.mp4">><br>
<<statChange "Hormonal-Therapy-Return-to-DrHart" "<<increaseSexualityNobr 1>>">>
<<statChange "Hormonal-Therapy-Return-to-DrHart-Submissive" "<<submissiveAct>>">><br>
<<narration>>You picture her sitting on your face, pressing your head firmly into her pussy, your tongue eagerly licking and exploring every fold.<</narration>>
<<psychiatrist>>What are you thinking about? You seem a bit distracted.<</psychiatrist>><br>
<<player>>Oh, nothing.<</player>><br>
<<psychiatrist>>If you say so.<</psychiatrist>><br>
<<narration>>A smile flashes on her face.<</narration>>
<<psychiatrist>>Anyway, it's quite clear that you're super responsive to the pill. The changes in your body are more pronounced than average.<</psychiatrist>><br>
<<player>>Is that a good thing?<</player>><br>
<<psychiatrist>>Definitely. It means you'll be able to achieve more results sooner. That's why I want to see you again in seven days for another therapy session.<</psychiatrist>><br>
<<player>>Therapy? Isn't the pill enough?<</player>><br>
<<psychiatrist>>Oh, sweetie, the pill is just the beginning. Therapy is essential to guide you through this profound transformation. We need to make sure that you're adapting well to these changes, embracing your new self fully, and learning to navigate this world in a way that feels right for you.<</psychiatrist>><br>
<<player>>I appreciate that. Thank you for explaining everything.<</player>><br>
<<button "Leave">>
<<set $firstTherapyCounter = $totalDays + 7>>
<<run MindWare.completeTask("First hormonal therapy checkup", "Task-HormonalTherapyCheckupOne")>>
<<run MindWare.addTask("Therapy session in " + Math.abs($totalDays - $firstTherapyCounter) + " days", "Task-TherapySession-1")>>
<<goto "PanaceaClinic">>
<</button>>
<</replace>>
<</button>>@@
<br>
<!-- Button to resist AVA wrapped in a div -->
<div id="resist-ava-button-container">
@@.choicebutton;<<button "Resist AVA">>
<<run MindWare.resistAVAMinigame()>>
<</button>>@@
<br>
</div>
<!-- Hidden Content: Button to Move On (Initially Hidden) -->
<div id="AVA-resist-minigame-hidden-content" style="display: none;">
@@.choicebutton;<<button "Move on">>
<<replace "#ChoiceButton">>
<<narration>>You push the suggestive thoughts from your mind.<</narration>>
<<psychiatrist>>Anyway, it's quite clear that you're super responsive to the pill. The changes in your body are more pronounced than average.<</psychiatrist>><br>
<<player>>Is that a good thing?<</player>><br>
<<psychiatrist>>Definitely. It means you'll be able to achieve more results sooner. That's why I want to see you again in seven days for another therapy session.<</psychiatrist>><br>
<<player>>Therapy? Isn't the pill enough?<</player>><br>
<<psychiatrist>>Oh, sweetie, the pill is just the beginning. Therapy is essential to guide you through this profound transformation. We need to make sure that you're adapting well to these changes, embracing your new self fully, and learning to navigate this world in a way that feels right for you.<</psychiatrist>><br>
<<player>>I appreciate that. Thank you for explaining everything.<</player>> <br>
<<button "Leave">>
<<set $firstTherapyCounter = $totalDays + 7>>
<<run MindWare.completeTask("First hormonal therapy checkup", "Task-HormonalTherapyCheckupOne")>>
<<run MindWare.addTask("Therapy session in " + Math.abs($totalDays - $firstTherapyCounter) + " days", "Task-TherapySession-1")>>
<<set $hormonalTherapyDreamsActive to false>>
<<set $dreamType to "none">>
<<goto "PanaceaClinic">>
<</button>>
<</replace>>
<</button>>@@
</div>
</span><</nobr>><<include "EndOfCurrentStory">>Dr. Hart has scheduled your next therapy session to discuss your progress and experiences with the hormonal therapy.
Visit her at Panacea Clinic in <<print Math.abs($totalDays - $firstTherapyCounter)>> days.
<<nobr>>
<<set _currentPassage to passage()>>
<<if MindWare.checkPassword($enteredCode)>>
<<button "Set the counter to 0">>
<<set $firstTherapyCounter = $totalDays>>
<<goto _currentPassage>>
<<run Dialog.close()>>
<<run Dialog.setup("Task Description")>>
<<run Dialog.wiki(Story.get("Task-TherapySession-1").processText())>>
<<run Dialog.open()>>
<</button>>
<</if>>
<</nobr>>/* <div class="img-location"><img src="imgs/locations/hospital/psychiatrist-office.webp" width="100%"></div><br> */
<<include "EndOfCurrentStory">>/* ByteBunker Job Added */
<<if $chapter1Days > 5 && !$byteBunkerSuspicionJobAdded>>
<<run setup.addByteBunkerJob("Need Digital Investigation Services", "A concerned spouse seeking evidence of potential infidelity.", "ConcernedSpouse", 3000, "ByteBunker-Suspicion")>>
<<set $byteBunkerSuspicionJobAdded to true>>
<</if>>
/* Accepted but not finished */
<<if $chapterDays > ($byteBunkerSuspicionJobStartDay + 5) && $byteBunkerSuspicionJobAdded === true && !$byteBunkerSuspicionJobFinished && !$byteBunkerSuspicionEndEmailSent>>
<<run MindWare.addEmail("ByteBunker Admin", "Job Reassignment Notice - Contract Breach", "Your accepted contract has been terminated.", "ByteBunker-SuspicionFailEmail")>>
<<set $byteBunkerSuspicionEndEmailSent to true>>
<</if>>
/* Client Response Email */
<<if $cheatingWifeEvidenceSent && ($totalDays > $cheatingWifeEvidenceSentDay) && !$cheatingWifeResponseEmailSent>>
<<run MindWare.addEmail(
"James Chen",
"Evidence Received",
"I knew it. I KNEW IT! The evidence you sent is exactly what I needed. She's been lying to my face this whole time. My lawyer will have a field day with this.",
"ByteBunker-SuspicionComplete")>>
<<set $cheatingWifeResponseEmailSent to true>>
<<set $playerMoney += 3000>>
<</if>>
/* Client Response Email - Fake Evidence */
<<if $cheatingWifeFakeEvidenceSent && ($totalDays > $cheatingWifeEvidenceSentDay) && !$cheatingWifeFakeResponseEmailSent>>
<<run MindWare.addEmail(
"James Chen",
"Report Received",
"This can't be right. I've been watching her behavior change for months now. Either you're incompetent or... No, I won't speculate.",
"ByteBunker-SuspicionComplete-Fake")>>
<<set $cheatingWifeFakeResponseEmailSent to true>>
<</if>><div class="email-message">\
<div class="from-line"><strong>From:</strong> James Chen</div>\
<div class="subject-line"><strong>Subject:</strong> Report Received - Inconclusive</div>\
<div class="message-content">
This can't be right. I've been watching her behavior change for months now. Either you're incompetent or... No, I won't speculate.
I will not be processing the agreed payment as I don't believe you fulfilled your end of the contract with the due diligence required.
James
</div>\
<div class="email-message-buttons">
<<if !MindWare.emailIsRead("ByteBunker-SuspicionComplete-Fake")>>\
<span class="email-btn-wrapper-span">\
@@.email-listing;<<button "Mark as Read">>\
<<run MindWare.markEmailAsRead("ByteBunker-SuspicionComplete-Fake")>>\
<<goto "ByteBunker-SuspicionComplete-Fake">>\
<</button>>\
@@\
</span>\
<</if>>\
</div>\
</div>\
<p class="back-link">[[Go Back -> EmailClient]]</p><div class="email-message">\
<div class="from-line"><strong>From:</strong> James Chen</div>\
<div class="subject-line"><strong>Subject:</strong> Evidence Received</div>\
<div class="message-content">
I knew it. I KNEW IT!
The evidence you sent is exactly what I needed. She's been lying to my face this whole time.
My lawyer will be putting these materials to good use. I've already transferred the agreed payment and won't require any further surveillance services regarding this situation.
Thank you for your professional handling of this sensitive matter.
James
</div>\
<div class="email-message-buttons">
<<if !MindWare.emailIsRead("ByteBunker-SuspicionComplete")>>\
<span class="email-btn-wrapper-span">\
@@.email-listing;<<button "Mark as Read">>\
<<run MindWare.markEmailAsRead("ByteBunker-SuspicionComplete")>>\
<<goto "ByteBunker-SuspicionComplete">>\
<</button>>\
@@\
</span>\
<</if>>\
</div>\
</div>\
<p class="back-link">[[Go Back -> EmailClient]]</p><div class="email-message">\
<div class="from-line"><strong>From:</strong> ByteBunker Admin</div>\
<div class="subject-line"><strong>Subject:</strong> Job Reassignment Notice</div>\
<div class="message-content">
AUTOMATED MESSAGE - DO NOT REPLY
This notice is to inform you that your accepted contract has been terminated on ConcernedSpouse's request.
PROVIDED REASON: Failure to complete within the specified timeframe.
-- ByteBunker Administration System
</div>\
<div class="email-message-buttons">
<<if !MindWare.emailIsRead("ByteBunker-SuspicionFailEmail")>>\
<span class="email-btn-wrapper-span">\
@@.email-listing;<<button "Mark as Read">>\
<<run MindWare.markEmailAsRead("ByteBunker-SuspicionFailEmail")>>\
<<goto "ByteBunker-SuspicionFailEmail">>\
<</button>>\
@@\
</span>\
<</if>>\
</div>\
</div>\
<p class="back-link">[[Go Back -> EmailClient]]</p><div class="job-description">
<h2>Need Digital Investigation Services</h2>
<div class="job-meta">
<span><b>Posted by:</b> ConcernedSpouse</span>
<span><b>Reward:</b> $3000</span>
</div>
<p>My wife has been acting strangely lately - staying late at work, being secretive with her phone, and changing all her passwords without telling me. I used to know them all, for her own safety of course.</p>
<p>She works as a project manager at Ryushi Motors and has been spending a lot of "overtime" with her boss on their new project. I've tried talking to her about it, but she keeps saying I'm being paranoid. I need someone to check her work computer for any inappropriate communications or evidence.</p>
<p><b>If you accept this job, then I except to see some evidence in 5 days or you won't get paid.</b></p>
<<if setup.byteBunkerJobIsAccepted("ByteBunker-Suspicion")>>
<div class="btn-wrapper">
<button disabled="disabled">Job Taken</button>
</div>
<<else>>
<div class="btn-wrapper">
<<button "Save the target">>
<<run MindWare.addTask("Investigate suspected affair", "Task-Investigate-Wife")>>
<<run setup.changeByteBunkerJobAcceptedStatus("ByteBunker-Suspicion", true)>>
<<run setup.addHackingTarget("Sarah Chen", "186d:4721:9f3c:8e55", "ByteBunker-Suspicion-animation")>>
<<goto "ByteBunker-Suspicion">>
<</button>>
</div>
<</if>>
</div><br>
<p class="back-link">
<<return "Go Back">>
</p>Your task is to infiltrate Sarah Chen's work computer at Ryushi Motors and search for any evidence of an affair with her boss.<div class="desktop-hacking-tool-wrapper">\
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target</span></div>\
<div class="desktop-hacking-tool-animation">\
<div class="msg-container">\
<div class="msg">Connecting</div>\
</div>\
<div class="hacking-animation">\
<div id="console"></div>\
</div>\
</div>\
</div>\
<<if $byteBunkerSuspicionEndEmailSent>>
<<script>>
$(document).one(':passagerender', function() {
setTimeout(function() {
var consoleElement = document.getElementById("console");
const msg = document.querySelector(".msg");
var txt = [
"Initializing Ryushi Motors connection...",
"...",
"Authenticating access credentials...",
"...Authentication failed.",
"...",
"Checking alternate access routes...",
"...All routes blocked.",
"...",
"Target workstation unreachable...",
"...Security protocols updated.",
"Attempting emergency breach...",
"...",
"Access denied: Target system locked.",
"Scanning for security vulnerabilities...",
"...",
"Alert: System administrator notified.",
"Warning: Enhanced security measures detected",
"Mission window has expired...",
"...",
"Final connection attempt...",
"...",
"Connection terminated - Target access revoked."
];
function updateScreen() {
if (txt.length > 0) {
var p = document.createElement("p");
p.textContent = txt.shift();
consoleElement.appendChild(p);
consoleElement.scrollTop = consoleElement.scrollHeight;
} else {
clearInterval(intervalID);
msg.style.background = "red";
msg.innerHTML = "ACCESS DENIED";
msg.style.boxShadow = "0 0 30px red";
}
}
var intervalID = window.setInterval(updateScreen, 200);
}, 0);
});
<</script>>
<<else>>
<<script>>
$(document).one(':passagerender', function() {
setTimeout(function() {
var consoleElement = document.getElementById("console");
const msg = document.querySelector(".msg");
var txt = [
"Initiating Ryushi Motors network scan...",
"...",
"Detecting security protocols...",
"...Multiple layers identified.",
"Bypassing perimeter security...",
"...",
"Access granted to outer network.",
"...Locating target workstation.",
"Target identified: Sarah.Chen@RyushiMotors",
"...",
"Deploying stealth protocols...",
"...Masking digital footprint.",
"Establishing secure connection...",
"...",
"Creating encrypted tunnel...",
"...Connection secured.",
"Implementing counter-detection measures...",
"...",
"Network position secured.",
"...All systems operational.",
"Connection established."
];
function updateScreen() {
if (txt.length > 0) {
var p = document.createElement("p");
p.textContent = txt.shift();
consoleElement.appendChild(p);
consoleElement.scrollTop = consoleElement.scrollHeight;
} else {
clearInterval(intervalID);
msg.style.background = "lime";
msg.innerHTML = "SUCCESS";
msg.style.boxShadow = "0 0 30px lime";
}
}
var intervalID = window.setInterval(updateScreen, 200);
}, 0);
});
<</script>>
<</if>>
<<if $byteBunkerSuspicionEndEmailSent>>
<<timed 8s>><<goto "ByteBunker-Suspicion-404">><</timed>>
<<else>>
<<timed 8s>>
<<set $numberOfPasswordsNeeded to 5>>
<<set $bruteforceBaseTraceTime to 25>>
<<set $nextPassageAfterSuccessBruteforceGame to "ByteBunker-Suspicion-computer-access">>
<<set $nextPassageAfterFailBruteforceGame to "HackTarget">>
<<goto "ByteBunker-Suspicion-computer-login">><</timed>>
<</if>><div class="desktop-hacking-tool-wrapper">\
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Server Connection Error</span></div>\
<div class="server-404-error-message">
<p class="blink-text">404 - Access Revoked</p>
</div>\
</div>\
<p>The target is no longer accessible - your mission window has expired.
Do you want to <<link "delete it">><<run setup.removeHackingTarget("Sarah Chen")>><<goto "HackTarget">><</link>> or just [[go back -> HackTarget]]?</p><div class="desktop-hacking-tool-wrapper">\
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>\
<div class="desktop-hacking-tool-server-password-protected">\
<div class="server-password-lock-icon">🔒</div>\
<input type="text" id="server-password-input" class="server-password-input" placeholder="Enter Password">\
<button class="server-password-submit-button" onclick="checkLandlordPassword()">Submit</button>\
<div class="prank-server-action-selection-box">\
<p class="active-cameras"> Access to the computer is password-protected.</p>\
<button class="server-password-access-button" onclick="SugarCube.Engine.play('ByteBunker-Suspicion-computer-bruteforce')">Manually Bruteforce the Password</button>\
</div>\
</div>\
</div>\
<p class="back-link">[[Go Back -> HackTarget]]</p>
<<nobr>>
<<script>>
setup.showAboutPcWindow = function() {
$('#playerPcName').text("RYUSHI-WEST-45E");
$('#playerNickname').text("sarah.chen");
$('#aboutPcWindow').show();
$('#closeAboutPcWindow').off('click').click(function() {
$('#aboutPcWindow').hide();
});
}
<</script>>
<style>
.prank-server-action-selection-box {
background-color: #111;
color: lime;
font-family: 'Courier New', monospace;
padding: 15px;
margin-top: 80px;
margin-bottom: 0px;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 255, 0, 0.3);
}
.prank-desktop-login {
background-color: #111;
background-size: 100%;
/* padding: 20px; */
height: 500px;
box-shadow: 0px 0px 15px 2px lime;
position: relative;
border: 4px solid lime;
border-radius: 10px;
}
</style>
<script>
function checkLandlordPassword() {
let password = document.getElementById('server-password-input').value;
if (password === "ndsIfyS" || password === "ndsifys") {
SugarCube.Engine.play('ByteBunker-Suspicion-computer-access');
} else {
SugarCube.UI.alert("Incorrect password.");
}
}
</script>
<</nobr>><<include "Minigame-Bruteforce">>\
<<if $minigameSkipButtons === true>>\
@@.minigame-skip-button;<<button "Skip Minigame">><<goto "ByteBunker-Suspicion-computer-access">><</button>>@@
<</if>>\<div class="desktop-hacking-tool-wrapper">
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>
<div class="ryushi-desktop">
<!-- Top bar -->
<div class="top-bar">
@@.about-pc-button;<<button "About This PC">>
<<run setup.showAboutPcWindow()>>
<</button>>@@
<span class="notification-counter"><i class="fas fa-wifi"></i> <i class="fas fa-volume-up"></i></span>
</div>
<div class="ryushi-desktop-content">
<div class="ryushi-desktop-icon" onclick="SugarCube.Engine.display('CheatingWife-RyushiEmail')">📧<span class="desktop-icon-label">Ryushi Mail</span></div>
<div class="ryushi-desktop-icon" onclick="SugarCube.Engine.display('CheatingWife-ProjectTimeline')">📅<span class="desktop-icon-label">Ethereal_Velocity_Timeline.xlsx</span></div>
<div class="ryushi-desktop-icon" onclick="SugarCube.Engine.display('CheatingWife-MeetingNotes')">📝<span class="desktop-icon-label">Team_Meeting_Notes.docx</span></div>
<div class="ryushi-desktop-icon" onclick="SugarCube.Engine.display('CheatingWife-SalesForecast')">📊<span class="desktop-icon-label">Sales_Forecast.pptx</span></div>
</div>
<div class="about-pc" id="aboutPcWindow" style="display: none;">
<div class="about-pc-header">
About This PC
<span id="closeAboutPcWindow" class="about-pc-close-button">⛌</span>
</div>
<div class="about-pc-body">
<p><b>Computer Name:</b> <span id="playerPcName"></span></p>
<p><b>Username:</b> <span id="playerNickname"></span></p>
</div>
</div>
</div>
</div>
<br>
<p class="back-link">[[Leave -> HackTarget]]</p>
<<nobr>>
<<script>>
setup.showAboutPcWindow = function() {
$('#playerPcName').text("RYUSHI-WEST-PM45");
$('#playerNickname').text("sarah.chen");
$('#aboutPcWindow').show();
$('#closeAboutPcWindow').off('click').click(function() {
$('#aboutPcWindow').hide();
});
}
<</script>>
<style>
.ryushi-desktop {
background-color: #111;
background-size: cover;
padding: 20px;
height: 500px;
box-shadow: 0px 0px 15px 2px lime;
position: relative;
border: 2px solid lime;
}
.ryushi-desktop-icon {
width: 120px;
transition: transform 0.2s;
margin: 20px;
text-align: center;
font-size: 48px;
color: white;
cursor: pointer;
position: relative;
}
.desktop-icon-label {
display: block;
color: lime;
font-size: 14px;
margin-top: 5px;
text-shadow: 0 0 5px rgba(0, 255, 0, 0.5);
word-wrap: break-word;
white-space: normal;
text-align: center;
}
.ryushi-desktop-content {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
gap: 10px;
padding: 20px;
overflow: auto;
scrollbar-width: none;
}
.ryushi-desktop-content::-webkit-scrollbar {
display: none;
}
</style>
<</nobr>><div class="desktop-hacking-tool-wrapper">
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>
<div class="ryushi-desktop">
<!-- Top bar -->
<div class="top-bar">
@@.about-pc-button;<<button "About This PC">>
<<run setup.showAboutPcWindow()>>
<</button>>@@
<span class="notification-counter"><i class="fas fa-wifi"></i> <i class="fas fa-volume-up"></i></span>
</div>
<div class="ryushi-desktop-content">
<div class="sales-forecast-container">
<br>
<h3>Sales Forecast</h3>
<div class="sales-forecast-graph">
<div class="bar" style="height: 50%;" title="Q1: $50k"></div>
<div class="bar" style="height: 70%;" title="Q2: $70k"></div>
<div class="bar" style="height: 60%;" title="Q3: $60k"></div>
<div class="bar" style="height: 80%;" title="Q4: $80k"></div>
</div>
<div class="sales-forecast-labels">
<span>Q1</span>
<span>Q2</span>
<span>Q3</span>
<span>Q4</span>
</div>
<br>
<ul>
<li>Q1 shows steady growth from the previous year.</li>
<li>Q2 is projected to see the highest increase due to product launches.</li>
<li>Q3 sees a slight decline due to seasonal demand fluctuations.</li>
<li>Q4 anticipates strong year-end sales, driven by holiday promotions.</li>
</ul>
</div>
</div>
<div class="about-pc" id="aboutPcWindow" style="display: none;">
<div class="about-pc-header">
About This PC
<span id="closeAboutPcWindow" class="about-pc-close-button">⛌</span>
</div>
<div class="about-pc-body">
<p><b>Computer Name:</b> <span id="playerPcName"></span></p>
<p><b>Username:</b> <span id="playerNickname"></span></p>
</div>
</div>
</div>
</div>
<br>
<p class="back-link">[[Go Back -> ByteBunker-Suspicion-computer-access]]</p>
<<script>>
setup.showAboutPcWindow = function() {
$('#playerPcName').text("RYUSHI-WEST-PM45");
$('#playerNickname').text("sarah.chen");
$('#aboutPcWindow').show();
$('#closeAboutPcWindow').off('click').click(function() {
$('#aboutPcWindow').hide();
});
}
<</script>>
<style>
.sales-forecast-description {
margin-top: 20px;
color: lime;
font-size: 14px;
line-height: 1.5;
text-align: left; /* Align text for readability */
background: rgba(0, 255, 0, 0.1); /* Subtle background to distinguish text */
padding: 10px;
border: 1px solid lime;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 255, 0, 0.2);
}
.ryushi-desktop {
background-color: #111;
background-size: cover;
padding: 20px;
height: 500px;
overflow: auto;
box-shadow: 0px 0px 15px 2px lime;
position: relative;
border: 2px solid lime;
}
.ryushi-desktop-icon {
width: 120px;
transition: transform 0.2s;
margin: 20px;
text-align: center;
font-size: 48px;
color: white;
cursor: pointer;
position: relative;
}
.desktop-icon-label {
display: block;
color: lime;
font-size: 14px;
margin-top: 5px;
text-shadow: 0 0 5px rgba(0, 255, 0, 0.5);
word-wrap: break-word;
white-space: normal;
text-align: center;
}
.ryushi-desktop-content {
display: block; /* Change from grid to block to allow dynamic resizing */
padding: 20px;
overflow-x: auto; /* Allow horizontal scrolling if content exceeds the width */
overflow-y: hidden;
white-space: nowrap; /* Prevent line breaks in children */
scrollbar-width: thin; /* Optional: show a thin scrollbar for horizontal scrolling */
}
.ryushi-desktop-content::-webkit-scrollbar {
display: none;
}
.sales-forecast-container {
display: inline-block; /* Ensure the graph acts as a single block within the scrolling area */
width: 100%; /* Allow it to take up as much space as needed */
max-width: 100%; /* Prevent it from breaking the container */
}
.sales-forecast-graph {
display: flex;
justify-content: space-around;
align-items: flex-end;
height: 200px;
margin: 20px auto;
width: 80%;
background: rgba(0, 255, 0, 0.1);
border: 1px solid lime;
padding: 10px;
box-shadow: 0 0 15px rgba(0, 255, 0, 0.3);
}
.bar {
width: 20%;
background: lime;
box-shadow: 0 0 10px rgba(0, 255, 0, 0.5);
transition: all 0.3s ease;
cursor: pointer;
display: inline-block;
}
.bar:hover {
background: white;
color: black;
box-shadow: 0 0 15px white;
}
.sales-forecast-labels {
display: flex;
justify-content: space-around;
margin-top: 10px;
}
.sales-forecast-labels span {
flex: 1;
text-align: center;
color: lime;
font-size: 14px;
}
</style><div class="desktop-hacking-tool-wrapper">
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>
<div class="ryushi-desktop">
<!-- Top bar -->
<div class="top-bar">
@@.about-pc-button;<<button "About This PC">>
<<run setup.showAboutPcWindow()>>
<</button>>@@
<span class="notification-counter"><i class="fas fa-wifi"></i> <i class="fas fa-volume-up"></i></span>
</div>
<div class="ryushi-desktop-content">
<div class="meeting-notes">
<h2>Meeting Notes</h2>
<h4>Project: Concept Car Design</h4>
<p><b>Attendees:</b> Mr. Roderick Hayes (Project Manager), Sarah Chen (Accountant), Alex Grey (Designer), Emily Clarke (Designer)</p>
<h3>Agenda:</h3>
<ol>
<li>Presentation of concept car design by Alex and Emily.</li>
<li>Feedback and budgeting insights from Sarah Chen.</li>
<li>Approval and next steps from Mr. Hayes.</li>
</ol>
<h3>Discussion:</h3>
<p><b>Alex:</b> "Thank you for joining us. The concept car, 'Ethereal Velocity,' represents a significant leap forward in design and sustainability. We've been working hard to refine its aerodynamic profile, and the drag coefficient has been reduced by 15% compared to our last model."</p>
<p><b>Emily:</b> "Additionally, the bodywork will use graphene composites, which reduce weight while maintaining structural integrity. We've also incorporated a modular design that allows for customization without sacrificing efficiency."</p>
<p><b>Mr. Hayes:</b> "Interesting. The use of graphene composites sounds promising, but they tend to be expensive. Have you factored in scalability for production?"</p>
<p><b>Emily:</b> "We have, though it depends on our budget. By prioritizing bulk procurement contracts and simplifying non-critical components, we believe the costs can be kept manageable."</p>
<p><b>Alex:</b> "The LED customization feature is another point of interest. It could be a major selling point for customers who want a personalized touch, but it’s modular to keep production costs flexible."</p>
<p><b>Mr. Hayes:</b> "Good thinking. Sarah, where do we stand financially? Do we have room for these innovations?"</p>
<p><b>Sarah:</b> "The current budget covers most of what Alex and Emily have proposed, but graphene composites might push us to the edge. If we streamline the customization options in the early phases, we could ensure a smoother prototype-to-production transition."</p>
<p><b>Mr. Hayes:</b> "I see. Sarah, you're always on point with balancing ambition and reality. It's why you're indispensable in these discussions."</p>
<p><b>Sarah:</b> "Thank you, Mr. Hayes. I think the modular approach is a smart compromise, and it leaves room for premium upgrades later."</p>
<p><b>Alex:</b> "One question I have is regarding the interior design. Are we leaning toward a minimalist look, or do we want to highlight luxury features upfront?"</p>
<p><b>Mr. Hayes:</b> "Good question. We need to strike a balance. A minimalist interior might attract eco-conscious buyers, but a touch of luxury could broaden the appeal. Emily, any thoughts?"</p>
<p><b>Emily:</b> "I agree. We could go with a dual-tiered approach: a standard minimalist interior and an optional luxury package. This aligns with our modular philosophy."</p>
<p><b>Mr. Hayes:</b> "That sounds workable. Let’s proceed with that idea. Sarah, I’ll need you to prepare a detailed financial projection for both options. Can we discuss this in my office later this week?"</p>
<p><b>Sarah:</b> "Certainly, Mr. Hayes. I’ll have the numbers ready."</p>
<p><b>Mr. Hayes:</b> "All right, great work, everyone. Emily, Alex, refine the design based on today’s feedback. Sarah, let’s touch base later. We’ll finalize the budget and get this project moving."</p>
<h3>Action Items:</h3>
<ul>
<li>Alex and Emily to refine the prototype design, incorporating modular LED customization and dual-tiered interior options.</li>
<li>Sarah to prepare a financial projection for both standard and luxury production models.</li>
<li>Mr. Hayes to approve the final budget and coordinate the prototype development timeline.</li>
</ul>
</div>
</div>
<div class="about-pc" id="aboutPcWindow" style="display: none;">
<div class="about-pc-header">
About This PC
<span id="closeAboutPcWindow" class="about-pc-close-button">⛌</span>
</div>
<div class="about-pc-body">
<p><b>Computer Name:</b> <span id="playerPcName"></span></p>
<p><b>Username:</b> <span id="playerNickname"></span></p>
</div>
</div>
</div>
</div>
<br>
<p class="back-link">[[Go Back -> ByteBunker-Suspicion-computer-access]]</p>
<<script>>
setup.showAboutPcWindow = function() {
$('#playerPcName').text("RYUSHI-WEST-PM45");
$('#playerNickname').text("sarah.chen");
$('#aboutPcWindow').show();
$('#closeAboutPcWindow').off('click').click(function() {
$('#aboutPcWindow').hide();
});
}
<</script>>
<style>
.ryushi-desktop {
background-color: #111;
background-size: cover;
padding: 20px;
height: 500px;
overflow: auto;
box-shadow: 0px 0px 15px 2px lime;
position: relative;
border: 2px solid lime;
}
.ryushi-desktop-icon {
width: 120px;
transition: transform 0.2s;
margin: 20px;
text-align: center;
font-size: 48px;
color: white;
cursor: pointer;
position: relative;
}
.desktop-icon-label {
display: block;
color: lime;
font-size: 14px;
margin-top: 5px;
text-shadow: 0 0 5px rgba(0, 255, 0, 0.5);
word-wrap: break-word;
white-space: normal;
text-align: center;
}
.ryushi-desktop-content {
padding: 20px;
overflow-x: auto; /* Allow horizontal scrolling if content exceeds the width */
scrollbar-width: thin; /* Optional: show a thin scrollbar for horizontal scrolling */
}
.ryushi-desktop-content::-webkit-scrollbar {
display: none;
}
.meeting-notes {
background-color: #222; /* Darker background for contrast */
color: #e0e0e0; /* Light text color for readability */
font-family: 'Arial', sans-serif;
padding: 20px;
margin: 20px auto;
text-align: left;
}
.meeting-notes h2, .meeting-notes h3, .meeting-notes h4 {
color: lime; /* Accent color for headings */
}
.meeting-notes p, .meeting-notes li {
margin-bottom: 10px;
line-height: 1.6;
}
.meeting-notes ul, .meeting-notes ol {
margin-left: 20px;
padding-left: 20px;
}
.meeting-notes i {
color: #b3b3b3; /* Subtle gray for italics */
font-size: 0.9em;
}
</style><div class="desktop-hacking-tool-wrapper">
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>
<div class="ryushi-desktop">
<!-- Top bar -->
<div class="top-bar">
@@.about-pc-button;<<button "About This PC">>
<<run setup.showAboutPcWindow()>>
<</button>>@@
<span class="notification-counter"><i class="fas fa-wifi"></i> <i class="fas fa-volume-up"></i></span>
</div>
<div class="ryushi-desktop-content">
<div class="project-timeline">
<h2>Ethereal Velocity Project Timeline</h2>
<p>This timeline outlines the estimated duration for each stage required to transform the Ethereal Velocity design concept into a production-ready car. Each stage includes key milestones and deliverables.</p>
<ul class="timeline">
<li>
<h3>Stage 1: Concept Development (4 weeks)</h3>
<p>Finalize initial sketches, approve the general design direction, and ensure alignment with sustainability goals and branding.</p>
</li>
<li>
<h3>Stage 2: Engineering Feasibility Study (6 weeks)</h3>
<p>Evaluate structural integrity, aerodynamics, and material selection. Assess feasibility of key features like modular LED systems and graphene composites.</p>
</li>
<li>
<h3>Stage 3: Detailed Design & Prototyping (12 weeks)</h3>
<p>Create CAD models, build a physical prototype, and conduct initial tests for performance, safety, and durability.</p>
</li>
<li>
<h3>Stage 4: Testing & Iteration (16 weeks)</h3>
<p>Conduct rigorous testing, including crash tests, range tests, and environmental simulations. Refine design based on test results.</p>
</li>
<li>
<h3>Stage 5: Pre-Production Setup (8 weeks)</h3>
<p>Establish supply chains, finalize production line configurations, and prepare manufacturing facilities.</p>
</li>
<li>
<h3>Stage 6: Production Launch (4 weeks)</h3>
<p>Begin production of the first batch. Conduct quality assurance checks and prepare for market release.</p>
</li>
</ul>
</div>
</div>
<div class="about-pc" id="aboutPcWindow" style="display: none;">
<div class="about-pc-header">
About This PC
<span id="closeAboutPcWindow" class="about-pc-close-button">⛌</span>
</div>
<div class="about-pc-body">
<p><b>Computer Name:</b> <span id="playerPcName"></span></p>
<p><b>Username:</b> <span id="playerNickname"></span></p>
</div>
</div>
</div>
</div>
<br>
<p class="back-link">[[Go Back -> ByteBunker-Suspicion-computer-access]]</p>
<<script>>
setup.showAboutPcWindow = function() {
$('#playerPcName').text("RYUSHI-WEST-PM45");
$('#playerNickname').text("sarah.chen");
$('#aboutPcWindow').show();
$('#closeAboutPcWindow').off('click').click(function() {
$('#aboutPcWindow').hide();
});
}
<</script>>
<style>
.ryushi-desktop {
background-color: #111;
background-size: cover;
padding: 20px;
height: 500px;
overflow: auto;
box-shadow: 0px 0px 15px 2px lime;
position: relative;
border: 2px solid lime;
}
.ryushi-desktop-icon {
width: 120px;
transition: transform 0.2s;
margin: 20px;
text-align: center;
font-size: 48px;
color: white;
cursor: pointer;
position: relative;
}
.desktop-icon-label {
display: block;
color: lime;
font-size: 14px;
margin-top: 5px;
text-shadow: 0 0 5px rgba(0, 255, 0, 0.5);
word-wrap: break-word;
white-space: normal;
text-align: center;
}
.ryushi-desktop-content {
padding: 20px;
scrollbar-width: thin; /* Optional: show a thin scrollbar for horizontal scrolling */
}
.ryushi-desktop-content::-webkit-scrollbar {
display: none;
}
.project-timeline {
background-color: #222; /* Subtle dark background for better contrast */
color: #e0e0e0; /* Light text for readability */
padding: 20px;
margin: 20px auto;
border: 1px solid #555;
text-align: left;
}
.project-timeline h2 {
color: lime; /* Accent color for the header */
margin-bottom: 10px;
}
.project-timeline p {
margin-bottom: 20px;
line-height: 1.6;
}
.timeline {
list-style-type: none;
padding: 0;
margin: 0;
}
.timeline li {
margin-bottom: 20px;
padding: 10px;
border-left: 3px solid lime;
background: rgba(0, 255, 0, 0.1);
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 255, 0, 0.2);
}
.timeline h3 {
margin: 0 0 5px 0;
color: lime;
font-size: 16px;
}
.timeline p {
margin: 0;
font-size: 14px;
line-height: 1.4;
}
</style><div class="desktop-hacking-tool-wrapper">
<div class="desktop-hacking-tool-header"><span class="desktop-hacking-tool-header-text">Hack-a-Target - Sarah Chen</span></div>
<div class="ryushi-desktop">
<!-- Top bar -->
<div class="top-bar">
@@.about-pc-button;<<button "About This PC">>
<<run setup.showAboutPcWindow()>>
<</button>>@@
<span class="notification-counter"><i class="fas fa-wifi"></i> <i class="fas fa-volume-up"></i></span>
</div>
<div class="ryushi-desktop-content">
<br>
<div id="email-list">
<h2>Inbox</h2>
<table class="email-inbox">
<thead>
<tr>
<th>From</th>
<th>Subject</th>
<th>Date</th>
</tr>
</thead>
<tbody>
<tr>
<td>Roderick Hayes</td>
<td>
<<link "Follow-Up on Financial Projections">>
<<replace "#email-list">><<include "CheatingWife-Email-FinancialProjections">><</replace>>
<<if $cheatingWifeEmail1Read && $cheatingWifeEmail2Read>>
<<replace "#evidence">><<include "CheatingWife-Email-Evidence">><</replace>>
<</if>>
<</link>>
</td>
<td><<set _daysAgo = $totalDays - ($totalDays - 0)>> <<print _daysAgo>> days ago</td>
</tr>
<tr>
<td>Emily Clarke</td>
<td>
<<link "Design Review Feedback">>
<<replace "#email-list">><<include "CheatingWife-Email-DesignFeedback">><</replace>>
<</link>>
</td>
<td><<set _daysAgo = $totalDays - ($totalDays - 3)>> <<print _daysAgo>> days ago</td>
</tr>
<tr>
<td>Accounting Team</td>
<td>
<<link "Monthly Report Submission Reminder">>
<<replace "#email-list">><<include "CheatingWife-Email-ReportReminder">><</replace>