rjw-sexperience-ideology/Source/IdeologyAddon/Rituals/JobGiver_DrugOrgy.cs

252 lines
6 KiB
C#
Raw Normal View History

2023-04-23 12:15:07 +00:00
using RimWorld;
2022-07-26 03:55:56 +00:00
using rjw;
2023-04-23 12:15:07 +00:00
using System;
using System.Collections.Generic;
2022-07-26 03:55:56 +00:00
using Verse;
using Verse.AI;
namespace RJWSexperience.Ideology
{
2023-04-23 12:15:07 +00:00
public class JobGiver_DrugOrgy : ThinkNode_JobGiver
{
protected override Job TryGiveJob(Pawn pawn)
{
if (pawn.Drafted || pawn.mindState == null)
{
return null;
}
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
PawnDuty duty = pawn.mindState.duty;
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
if (duty.def == DutyDefOf.TravelOrLeave || !xxx.can_do_loving(pawn))
{
return null;
}
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
Pawn target = FindPartner(pawn, duty);
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
if (target == null || !pawn.CanReserveAndReach(target, PathEndMode.ClosestTouch, Danger.None, 1))
return JobMaker.MakeJob(RsiDefOf.Job.DrugMasturbate);
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
return JobMaker.MakeJob(RsiDefOf.Job.DrugSex, target);
}
2022-07-26 03:55:56 +00:00
2023-04-23 12:15:07 +00:00
protected Pawn FindPartner(Pawn pawn, PawnDuty duty)
{
if (duty != null)
{
List<Pawn> pawns = pawn.Map.mapPawns.AllPawnsSpawned.FindAll(x => x.mindState?.duty?.def == duty.def);
return pawns.RandomElementByWeightWithDefault(x => SexAppraiser.would_fuck(pawn, x), 0.1f);
}
return null;
}
}
2022-07-26 03:55:56 +00:00
/// <summary>
/// copied from rjw
/// </summary>
public class JobDriver_SexDrugOrgy : JobDriver_SexBaseInitiator
{
public override bool TryMakePreToilReservations(bool errorOnFailed)
{
return true;
}
protected override IEnumerable<Toil> MakeNewToils()
{
setup_ticks();
2023-04-23 11:41:27 +00:00
var PartnerJob = RsiDefOf.Job.GettinDrugSex;
2022-07-26 03:55:56 +00:00
this.FailOnDespawnedNullOrForbidden(iTarget);
this.FailOn(() => !Partner.health.capacities.CanBeAwake);
this.FailOn(() => pawn.Drafted);
this.FailOn(() => Partner == null);
yield return Toils_Goto.GotoThing(iTarget, PathEndMode.OnCell);
2023-04-23 12:15:07 +00:00
Toil WaitForPartner = new Toil
2022-07-26 03:55:56 +00:00
{
2023-04-23 12:15:07 +00:00
defaultCompleteMode = ToilCompleteMode.Delay,
initAction = delegate
2022-07-26 03:55:56 +00:00
{
2023-04-23 12:15:07 +00:00
ticksLeftThisToil = 5000;
},
tickAction = delegate
{
pawn.GainComfortFromCellIfPossible();
if (pawn.Position.DistanceTo(Partner.Position) <= 1f)
{
ReadyForNextToil();
}
2022-07-26 03:55:56 +00:00
}
};
yield return WaitForPartner;
2023-04-23 12:15:07 +00:00
Toil StartPartnerJob = new Toil
2022-07-26 03:55:56 +00:00
{
2023-04-23 12:15:07 +00:00
defaultCompleteMode = ToilCompleteMode.Instant,
socialMode = RandomSocialMode.Off,
initAction = delegate
{
if (!(Partner.jobs.curDriver is JobDriver_DrugSexReceiver))
{
Job gettingQuickie = JobMaker.MakeJob(PartnerJob, pawn, Partner);
Partner.jobs.StartJob(gettingQuickie, JobCondition.InterruptForced);
}
2022-07-26 03:55:56 +00:00
}
};
yield return StartPartnerJob;
2023-04-23 12:15:07 +00:00
Toil SexToil = new Toil
{
defaultCompleteMode = ToilCompleteMode.Never,
socialMode = RandomSocialMode.Off,
defaultDuration = duration,
handlingFacing = true
};
2022-07-26 03:55:56 +00:00
SexToil.FailOn(() => Partner.CurJob.def != PartnerJob);
SexToil.initAction = delegate
{
Partner.pather.StopDead();
Partner.jobs.curDriver.asleep = false;
2023-04-23 12:15:07 +00:00
2022-07-26 03:55:56 +00:00
Start();
Sexprops.usedCondom = CondomUtility.TryUseCondom(pawn) || CondomUtility.TryUseCondom(Partner);
};
SexToil.AddPreTickAction(delegate
{
SexTick(pawn, Partner);
SexUtility.reduce_rest(Partner, 1);
SexUtility.reduce_rest(pawn, 1);
if (ticks_left <= 0)
ReadyForNextToil();
});
SexToil.AddFinishAction(delegate
{
End();
});
yield return SexToil;
yield return new Toil
{
initAction = delegate
{
//// Trying to add some interactions and social logs
SexUtility.ProcessSex(Sexprops);
},
defaultCompleteMode = ToilCompleteMode.Instant
};
}
}
/// <summary>
/// copied from rjw
/// </summary>
public class JobDriver_DrugSexReceiver : JobDriver_SexBaseRecieverLoved
2023-04-23 12:15:07 +00:00
{
2022-07-26 03:55:56 +00:00
protected override IEnumerable<Toil> MakeNewToils()
{
setup_ticks();
parteners.Add(Partner);// add job starter, so this wont fail, before Initiator starts his job
// More/less hearts based on opinion.
if (pawn.relations.OpinionOf(Partner) < 0)
ticks_between_hearts += 50;
else if (pawn.relations.OpinionOf(Partner) > 60)
ticks_between_hearts -= 25;
this.FailOnDespawnedOrNull(iTarget);
this.FailOn(() => !Partner.health.capacities.CanBeAwake);
this.FailOn(() => pawn.Drafted);
this.FailOn(() => Partner.Drafted);
this.FailOn(() => Partner == null);
yield return Toils_Reserve.Reserve(iTarget, 1, 0);
2023-04-23 12:15:07 +00:00
var get_loved = MakeSexToil();
get_loved.handlingFacing = false;
yield return get_loved;
2022-07-26 03:55:56 +00:00
}
protected Toil MakeSexToil()
{
2023-04-23 12:15:07 +00:00
Toil get_loved = new Toil
2022-07-26 03:55:56 +00:00
{
2023-04-23 12:15:07 +00:00
defaultCompleteMode = ToilCompleteMode.Never,
socialMode = RandomSocialMode.Off,
handlingFacing = true,
tickAction = delegate
{
}
2022-07-26 03:55:56 +00:00
};
get_loved.AddEndCondition(new Func<JobCondition>(() =>
{
2023-04-23 12:15:07 +00:00
if (parteners.Count == 0)
2022-07-26 03:55:56 +00:00
{
return JobCondition.Succeeded;
}
return JobCondition.Ongoing;
}));
get_loved.AddFinishAction(delegate
{
if (xxx.is_human(pawn))
pawn.Drawer.renderer.graphics.ResolveApparelGraphics();
});
get_loved.socialMode = RandomSocialMode.Off;
return get_loved;
}
}
/// <summary>
/// copied from rjw
/// </summary>
public class JobDriver_DrugMasturabate : JobDriver_Masturbate
2023-04-23 12:15:07 +00:00
{
2022-07-26 03:55:56 +00:00
protected override IEnumerable<Toil> MakeNewToils()
{
setup_ticks();
this.FailOn(() => pawn.health.Downed);
this.FailOn(() => pawn.IsBurning());
this.FailOn(() => pawn.IsFighting());
this.FailOn(() => pawn.Drafted);
Toil SexToil = Toils_General.Wait(duration);
SexToil.handlingFacing = true;
SexToil.initAction = delegate
{
Start();
};
SexToil.tickAction = delegate
{
SexTick(pawn, null, true);
SexUtility.reduce_rest(pawn, 1);
if (ticks_left <= 0)
ReadyForNextToil();
};
SexToil.AddFinishAction(delegate
{
End();
});
yield return SexToil;
yield return new Toil
{
initAction = delegate
{
SexUtility.Aftersex(Sexprops);
if (!SexUtility.ConsiderCleaning(pawn)) return;
LocalTargetInfo own_cum = pawn.PositionHeld.GetFirstThing<Filth>(pawn.Map);
Job clean = JobMaker.MakeJob(JobDefOf.Clean);
clean.AddQueuedTarget(TargetIndex.A, own_cum);
pawn.jobs.jobQueue.EnqueueFirst(clean);
},
defaultCompleteMode = ToilCompleteMode.Instant
};
}
}
}