From e515144252e417d3aa95cfa92607095d5ea4f4f4 Mon Sep 17 00:00:00 2001 From: Pherring04 <158035107+Pherring04@users.noreply.github.com> Date: Thu, 18 Apr 2024 11:24:02 -0500 Subject: [PATCH] Water Clock Sim v3 (#1668) * Added Water Clock Sim * Updated graphics and documentation * Added images for README * Typos in README --------- Co-authored-by: Pherring04 --- .../Modified_data/my_waterclock.dr | 37 + trick_sims/SIM_waterclock/README.md | 62 ++ trick_sims/SIM_waterclock/RUN_test/input.py | 44 + trick_sims/SIM_waterclock/S_define | 36 + trick_sims/SIM_waterclock/S_overrides.mk | 2 + .../SIM_waterclock/WaterClockRunning.png | Bin 0 -> 13638 bytes trick_sims/SIM_waterclock/WaterClockStart.png | Bin 0 -> 13024 bytes .../SIM_waterclock/models/graphics/Makefile | 6 + .../SIM_waterclock/models/graphics/pom.xml | 121 +++ .../waterClockDisplay/WaterClockDisplay.java | 779 ++++++++++++++++++ .../models/waterclock/include/waterclock.h | 76 ++ .../waterclock/include/waterclock_numeric.h | 21 + .../models/waterclock/src/waterclock_init.c | 59 ++ .../waterclock/src/waterclock_numeric.c | 124 +++ .../waterclock/src/waterclock_shutdown.c | 13 + trick_sims/SIM_waterclock/waterclock.tv | 96 +++ trick_sims/SIM_waterclock/waterclock_debug.tv | 141 ++++ 17 files changed, 1617 insertions(+) create mode 100644 trick_sims/SIM_waterclock/Modified_data/my_waterclock.dr create mode 100644 trick_sims/SIM_waterclock/README.md create mode 100644 trick_sims/SIM_waterclock/RUN_test/input.py create mode 100644 trick_sims/SIM_waterclock/S_define create mode 100644 trick_sims/SIM_waterclock/S_overrides.mk create mode 100644 trick_sims/SIM_waterclock/WaterClockRunning.png create mode 100644 trick_sims/SIM_waterclock/WaterClockStart.png create mode 100644 trick_sims/SIM_waterclock/models/graphics/Makefile create mode 100644 trick_sims/SIM_waterclock/models/graphics/pom.xml create mode 100644 trick_sims/SIM_waterclock/models/graphics/src/main/java/trick/waterClockDisplay/WaterClockDisplay.java create mode 100644 trick_sims/SIM_waterclock/models/waterclock/include/waterclock.h create mode 100644 trick_sims/SIM_waterclock/models/waterclock/include/waterclock_numeric.h create mode 100644 trick_sims/SIM_waterclock/models/waterclock/src/waterclock_init.c create mode 100644 trick_sims/SIM_waterclock/models/waterclock/src/waterclock_numeric.c create mode 100644 trick_sims/SIM_waterclock/models/waterclock/src/waterclock_shutdown.c create mode 100644 trick_sims/SIM_waterclock/waterclock.tv create mode 100644 trick_sims/SIM_waterclock/waterclock_debug.tv diff --git a/trick_sims/SIM_waterclock/Modified_data/my_waterclock.dr b/trick_sims/SIM_waterclock/Modified_data/my_waterclock.dr new file mode 100644 index 00000000..b55e06ca --- /dev/null +++ b/trick_sims/SIM_waterclock/Modified_data/my_waterclock.dr @@ -0,0 +1,37 @@ +global DR_GROUP_ID +global drg +try: + if DR_GROUP_ID >= 0: + DR_GROUP_ID += 1 +except NameError: + DR_GROUP_ID = 0 + drg = [] + +drg.append(trick.DRAscii("WaterClock")) +drg[DR_GROUP_ID].set_freq(trick.DR_Always) +drg[DR_GROUP_ID].set_cycle(0.01) +drg[DR_GROUP_ID].set_single_prec_only(False) +drg[DR_GROUP_ID].add_variable("dyn.waterclock.time") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.input_flow") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_clock_spout_flowrate") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_overflow_flowrate") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_bucket_net_flow") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_bucket_depth") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_bucket_diam") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_overflow_height") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_overflow_diameter") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_clock_spout_height") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_clock_spout_diameter") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.timer_bucket_depth") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.timer_bucket_diam") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_bucket_vol") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.intake_water_level") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.timer_bucket_vol") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.timer_water_level") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.tick_gap") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.total_ticks") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.current_tick") +drg[DR_GROUP_ID].add_variable("dyn.waterclock.gravity") +drg[DR_GROUP_ID].set_max_file_size(1 * 1073741824) # multiply converts GiB to B --Dr. Dre +trick.add_data_record_group(drg[DR_GROUP_ID], trick.DR_Buffer) +drg[DR_GROUP_ID].enable() diff --git a/trick_sims/SIM_waterclock/README.md b/trick_sims/SIM_waterclock/README.md new file mode 100644 index 00000000..6093628d --- /dev/null +++ b/trick_sims/SIM_waterclock/README.md @@ -0,0 +1,62 @@ +# SIM_WaterClock + +--- + +SIM_WaterClock simulates a simple water clock. A water clock is a clock that measures time utilizing the flow of water. There exists different designs for water clocks, but this one uses a basic two bucket design. + +The first bucket is called the intake bucket. The intake bucket receives water from an external source of variable flow rate. In the real world this might be piped in from a stream for example. In this sim, the water source (input_flow) is a variable range from 0 to 10 liters/s that can be arbitrarily adjusted via the GUI. The intake bucket contains two spouts, a clock spout and an overflow spout. Both spouts are cylinders defined by their width and height on the intake bucket. The clock spout pours water into the second bucket (timer bucket), while the overflow spout empties water into the void. + +In an ideal water clock, by balancing the dimensions of the intake bucket, the clock spout, overflow spout, and the input flow rate, the flow rate of the clock spout (and therefore the input flow rate of the timer bucket) will be constant. Given a constant flow rate into the timer bucket, we can correlate water level with time passed. + +Assumptions: +* For the purpose of determining if a pipe is submerged, the pipe will be treated as a single point at its center. +* When the timing bucket is filled, a magical siphon will drain it instaneously. This prevents the need to simulate the recalibration of the clock. +* Buckets and spouts are perfectly cylindrical. + +### A Brief History +The water clock (or clepsydra to the ancient Greek) is a device used to measure time through the displacement of water. It is not known precisely when the earliest water clocks were developed, but they have been discovered to exist in various parts of the world as early as the 16th century BC. These early designs were relatively simple compared to later designs, consiting of a bowl with markings to indicate time passing as the bowl either filled or drained. + +Water clocks remained a prominent timekeeping device until the 17th century when better methods of timekeeping were developed. Over those many centruies, different cultures innovated on the design of the early water clocks. Of particular note were the contributions of the Greek inventor Ctesibius in the 3rd century BC. Ctesibius innovated on the water clock by making it largely automatic. Prior to Ctesibius, water clocks needed to be consistently refilled/emptied by hand. Furthermore, if a water clock had indicators marking the position of the sun, they would need to be manually adjusted for different seasons. Ctesibius automated both these processes. + +Ctesibius automated the draining process with a siphon akin to a Pythagorean cup. This siphon would empty over a water wheel, which powered a series of gears that would turn a cylinder. The cylinder had a row of irregular rings drawn around it, meant to correlate with the position of the sun throughout the seasons. Each day, as the water clock would fill it would accurately indicate the position of the sun, drain itself at the end of the day, and adjust the cylinder to be accurate for the next day. + +### Building the Simulation +After building trick, in the SIM\_waterclock directory run the command **trick-CP** to build the sim. When it's complete, you should see: + +``` +Trick Build Process Complete +``` +To build the graphics client **cd** into models/graphics/ and run **mvn package**. This isn't necessary for the sim to run, but will provide a visual display of the water clock. + +### Running the Simulation +In the SIM_waterclock directory: + +``` +% S_main_*.exe RUN_test/input.py +``` +The Sim Control Panel, and a GUI called "Water Clock" should appear. + +![Water Clock graphics client at the start of a run.](WaterClockStart.png) + +Click the Start on the Trick Sim Control Panel. The Water Clock should begin to fill up. + +![Water Clock graphics client mid-run.](WaterClockRunning.png) + +The only control on the GUI client is a slider which changes the flow rate of water into the water clock. It can be set anywhere from no flow up to 10 liters per second (10,000 cm^3^/s). + +By default, the water clock will count out 60 "ticks" at a little over 1 second a tick. See if you can clibrate the water clock to count out a proper minute (1 second a tick). + +### Configurable Parameters +The following parameters are meant to be configured by the user. They consist of the dimensions of the buckets, the dimensions and placement of the intake bucket spouts, and the number of time ticks. + +Variable | Type | Units +----------------------------------------------------------------------|----------------|------- +dyn.waterclock.intake_bucket_depth | double | m +dyn.waterclock.intake_bucket_diam | double | m +dyn.waterclock.intake_overflow_height | double | m +dyn.waterclock.intake_overflow_diameter | double | m +dyn.waterclock.intake_clock_spout_height | double | m +dyn.waterclock.intake_clock_spout_diameter | double | m +dyn.waterclock.timer_bucket_depth | double | m +dyn.waterclock.timer_bucket_diam | double | m +dyn.waterclock.total_ticks | int | -- diff --git a/trick_sims/SIM_waterclock/RUN_test/input.py b/trick_sims/SIM_waterclock/RUN_test/input.py new file mode 100644 index 00000000..0648aca5 --- /dev/null +++ b/trick_sims/SIM_waterclock/RUN_test/input.py @@ -0,0 +1,44 @@ +exec(open("Modified_data/my_waterclock.dr").read()) + +trick.frame_log_on() +trick.real_time_enable() +trick.exec_set_software_frame(0.1) +trick.itimer_enable() +trick.exec_set_enable_freeze(True) +trick.exec_set_freeze_command(True) +trick.sim_control_panel_set_enabled(True) + +dyn.waterclock.input_flow = 1000.0 + +dyn.waterclock.intake_bucket_depth = 70.0 +dyn.waterclock.intake_bucket_diam = 40.0 + +dyn.waterclock.intake_overflow_height = 60.0 +dyn.waterclock.intake_overflow_diameter = 25.0 + +dyn.waterclock.intake_clock_spout_height = 15.0 +dyn.waterclock.intake_clock_spout_diameter = 20.0 + +dyn.waterclock.timer_bucket_depth = 100.0 +dyn.waterclock.timer_bucket_diam = 45.0 + +dyn.waterclock.total_ticks = 60 + +trick.message_unsubscribe(trick_message.mcout) + +# ========================================== +# Start the Satellite Graphics Client +# ========================================== +varServerPort = trick.var_server_get_port(); +WaterClockDisplay_path = "models/graphics/build/WaterClockDisplay.jar" + +if (os.path.isfile(WaterClockDisplay_path)) : + WaterClockDisplay_cmd = "java -jar " \ + + WaterClockDisplay_path \ + + " " + str(varServerPort) + " &" ; + print(WaterClockDisplay_cmd) + os.system( WaterClockDisplay_cmd); +else : + print('==================================================================================') + print('Display needs to be built. Please \"cd\" into ../models/graphics and type \"mvn package\".') + print('==================================================================================') diff --git a/trick_sims/SIM_waterclock/S_define b/trick_sims/SIM_waterclock/S_define new file mode 100644 index 00000000..c6f593fd --- /dev/null +++ b/trick_sims/SIM_waterclock/S_define @@ -0,0 +1,36 @@ +/************************TRICK HEADER************************* +PURPOSE: + (S_define file for SIM_waterclock_numeric) +LIBRARY DEPENDENCIES: + ( + (waterclock/src/waterclock_init.c) + (waterclock/src/waterclock_numeric.c) + (waterclock/src/waterclock_shutdown.c) + ) +*************************************************************/ + +#include "sim_objects/default_trick_sys.sm" +##include "waterclock/include/waterclock_numeric.h" + +class WaterClockSimObject : public Trick::SimObject { + + public: + WATERCLOCK waterclock; + + WaterClockSimObject() { + ("default_data") waterclock_default_data( &waterclock ) ; + ("initialization") waterclock_init( &waterclock ) ; + ("derivative") waterclock_deriv( &waterclock ) ; + ("integration") trick_ret= waterclock_integ( & waterclock ) ; + ("shutdown") waterclock_shutdown( &waterclock ) ; + ("dynamic_event") waterclock_tick_change( &waterclock ) ; + ("dynamic_event") waterclock_overflow_timer( &waterclock ) ; + } +} ; + +WaterClockSimObject dyn ; + +IntegLoop dyn_integloop (0.01) dyn ; +void create_connections() { + dyn_integloop.getIntegrator(Runge_Kutta_4, 4); +} diff --git a/trick_sims/SIM_waterclock/S_overrides.mk b/trick_sims/SIM_waterclock/S_overrides.mk new file mode 100644 index 00000000..e1f6cccd --- /dev/null +++ b/trick_sims/SIM_waterclock/S_overrides.mk @@ -0,0 +1,2 @@ +TRICK_CFLAGS += -Imodels +TRICK_CXXFLAGS += -Imodels diff --git a/trick_sims/SIM_waterclock/WaterClockRunning.png b/trick_sims/SIM_waterclock/WaterClockRunning.png new file mode 100644 index 0000000000000000000000000000000000000000..b73ace9f1a9afe829fcfef6916a9b43e4151614b GIT binary patch literal 13638 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 literal 0 HcmV?d00001 diff --git a/trick_sims/SIM_waterclock/WaterClockStart.png b/trick_sims/SIM_waterclock/WaterClockStart.png new file mode 100644 index 0000000000000000000000000000000000000000..43f924b51ae108b47c3d0fa8532236816f1c6986 GIT binary patch literal 13024 zcmcI~XIxX;x9vs{ER=(dCLmw~>C&r6lS2~$5$QtcMWlodM--6ikuFtGf>Z(NfuJ<$ zAiX3AF?5LZ5|X?X&;P#Pz3<)fzC1nv*?VQ}Ip>;ljIrk0ukPz2Z!q!enUUA;bjm9{TX(o^MSF0(YRMq-A%0iC*(SDYxX!KL;n*1>ZT zR5WZ0B(a@A4}BlE;D*Kq!nS|5zjZ^6AtC^?AGDfjSpCVSHA0sPf?f|+YOw%w(-fZr zXI^LKf@@cAID#u(r-6@mp8dzmxADp4vSN&8o_;gGhI51ugLdUff|QTQT_%0gtwW*O znREI9Q{8@Zcx0XT?D*_c)#F7tilr9gYxZ4yzX2OBKi`VTVMYfV$hnKm^?&SNTFm-B zxtMwWMz;5}8J}^BMBy_nYNy#-dIM*8BGuH|uBDbuZ%taj(HB`Pj^m=D>UO4AQ%XX= zsIzo-#BpGMUDW4C?<179C;blb&2r6Nhq%+)K?yl9)61b>i+>w9PW;*6cc9A6PQce% znFp>{Z`N`jSi4S*#KX%%R;nC2VYBrL?i>a`*nJcc-q<+l_Ryf!HwTNMMAW}C=3Pd3 zN2vL;u9N}c#^bFzC*o1vKBIiwKxl~V@zRt_tsr4)9L2@Vd@FnF=jJ7e&Upe!s_bx% z346f8&5=VGxq%%)1vwnA&j-$Xd-E_Z!BWu3NO#;Zn@zZF3IP)|$qNSW`_U zukY>Yb+#ddm!3I z%)LF=>N~kSf2eYugQI;+kT^Gw1oH95{*zXr+O~2GKE!j!E1R>EYdg=c)ht9=7L0LRwv|?L1T9-Q&?5vgu5~YUw5h6&M;se0~<&9|!N?N$}Un+?hIWn`+_v3fRK; zPyw61S!9A#5H_t99t$Q^`hB;FZ08UCRdWs3Jcm$@le9W|c03i%eYrfKy~ozZ#-^uo zy>{ANJFMGl+6@)8pX2*DGxW>L&20{kSUSaDVSc-}^rJMD=9!r*{3nLe$-iD5d^Z{E z02#z0zm7D#?BB+`8I857v{TNIa*M7d4)J?#jQ7V#=L9`az}F8L--vgsObV77Npw4n+~?zx&L_k2lT5ZKCvvtB)Q%iZb>H@-p|t zC0F2;2oh5WE(YrbA+s7mM*rS?EZw+y6;7}3kj3)C{<`=4OVsZ1qZ(IJ^9J|rUmvxF zEm%0DT%!;ebVh~iWaWetA`82;s34E={=S-DwOY2>ydQd{U!HvIM;aP<)v0te#LT?; ze36sWLt%M$6X~EV&&!M1A$Y`u9->_4zwu==M=Im^H$Rk@#KDoi&Z3G?(BRdc{ks0UCSd1 zAA*mLiaJPCCR=UJ4G%;SW*ip%CO1W1Ih2{RHLfAI9}*5Hl!JKtiBtZYJFPiyImCH+ z$Fd346c-f^wNM6!2h7Y?j~k^@M9jEQK0996GTn@MRc=K^xTBGQqNdu6_pq+#u<}U-T<>i$S6%!dBF_t~9Bi{6PW1VmyWc^L4k)geR(x{h+3?Eg^Evo= zLy@1Cw>bdgKPt@H@}h}QPU79etdcxOM`sg$FH(avQmm|Qt9!yLL2;f^+xfvOL(sxk zETinz=4Trw*f&0_^EPyx7fj6kVG66?96I{t*Do;5gp!4}r5jocvX7DiH}2+a?bW&; zlAHD#LNgDJ60vWz$J<|(lh6la*HSxBq1zdoKW`8_QaXMC9E|a>2yjCFI2?#EL2bPN z@%zFQ%DQOfXHbGVT&Nr0o7Plov*jWd9;~7ctf+f}h`ZyOYHAMS7A{$Ka>ag#i zB~tkuLE#70=9o0f|Hmf9?>cgJU|aA8qX4Yaib&^?5ET{9c)a^aVb=nb{YZANmjGlSgoQFj;VrDmE?u9V${ zZ)9CIu^J$wIRox=d_|M4Xv@who#z2h({T=!v(sY6IgC;X6A&T zpICm_wGfjmGs$bespPW;GVuA4EDAmB{*@?Zzf<>1#pT5e(D`KD+zU zZ*Yhjb5o(>^GiT{s@mU%e`%Iw@Nc(C0Fb0dwaZ{6x&VEM~E25*Iu= zVk{4yPlPnC*OY{!2pH)y)Lg{7emd*S8npfN1Zg9~hGKD}n=1;vchc;_ZSON_XB+V~ zW^1+h=FL)V;ka!PQl1P>f0h}J z@pqsAJ$AQTv0~?uJRWJNruI8W_lbRtsmJ~;)O%&-3&H#78<#F!nlD}qR@^{t_jgbG zaI1~x@Lb}3a=3*v%QiCJX)EHkf?JeWp_FQ7LkKdfw{u?YO3!9CVx*>B&-bJV#mGxz z44W&li>6U8KX6EoG`VNwk^Tw7T1Dr^=H_iBc1_O1-fxvTodjj?$H=wh^5|&~RB%rN z>KG@IO2ZT;MKDcie+A#SUuzstqJU~!C~--1lTyhcDX_Rben-d)3sUZ!iAAuF`7xf% z1v0XA-XsLJjihkR99Y}UDkU~%X8rYaBq5Hx-xn1nGlvt2n+;lE_WXrj_4VQCubp|; zlQPf%-}k7SLu?j?Gg)pef)%=7kE8&#>3{-nh&K=+W&6i{Z*H)<&D4nrTkNl2gH3$0 z@nYlhYauZ+d(NrPPE?Y93zMex8owV@bZzB|2VcPRJ*-U44$2}^n|{{$byc9wQwU7y z*%4d)s-RV~f?Spb_GlGf?`7qCe0^)BUT3oU)&tyS-&wa>ziE8aHPT^bXl$!VNffvF zc5B_w4VOw!Y{vnJ3dsG3hiOpaPRDGZ%khu5E>C8I_g?KcFOK-(1NGv9_a|@sRx=D9 zheQp61V|W;1o@w@WmboCUQsfs=hMfiC=1`oYkSl1oP#ft%CS~WvqTF1eTGfC?>Abh zsb%i{J&bLfZRG7o{rGx~Ftfcer;8^KMq;~*%i?u2q}or51Ue}!zlv@Xn4OuE&x+L4N184E}f;UyKuiLY6>+Qi5T*`5lX8K^>@P66?L@ z*SwUtMf^zHPw-89Q$?{%iobGY_Fb%EC>cEyI-rPciBx$1>p=JC=(ct|G3%>acbUcC zm0mJrAeWG5g`?=!s!fjPSx`|xdcpa4pNv~p;v9 zKYs+o1@Ccyb(Xu_S7hOD^o2$D&y7vH$AAhYM-A&K^N~p0O3c%4wQk76pNaEdm;&wt zKs&tU!pEDU@ak-CU%wRrCxiXWRV z!z?1VS67EkMa2$;%yZM~UCv=qCN2*E{pIBf_%X1P4G~h=A+&6&Q|WdK`j>8 zJlFP;X6JwhlAkf~E4IgkHq9nN5A5l>*yK+Bw>vv9#H|X{wN@$^ohCLt<2&t^C`=7O zqP8{lhYK0kEx`WJy2EvQfPAd~uN#*t1*avJ??UU(c1N?cSZuB`%3R6tW-fS$xIKke zOR1NFsx$^6-Y34T>dasAn+;Afs!&_+onjYVToAL~IWP0QnByJ|L)c_h-}|$WPDGez zg1g13%h2oiTq-Y(7FL}fP?$H8?eap@YlFCe6Jc}9Lt(PHbLKf128b1mxvkT}n%zPL z^?@0~X<~P17@sZR=(KLF-+jqaQitI8#?KBK&RP0TO?=V!>qAJt{nH?K; z42I;Gn3#&Xy0lLc&>+`yS{*T-e9pJ#k(e+ZS4cEQhc=S`s`Y36EUo!rE?44$CeAs;svd_s}e#sy!wb?W^t%ceK3rl$4)1ev}%0K z?z33bDON}V7Ek**Gfb?h;1Cz`^(55#J)cVGx?OLCr)8&6LT_KCV5RHPiF45UJ56e9 z+F0ia;}lqcG~}o@7?$!qL(D8*&e!RXiOOAd##7BKGigr2{oI_2(dzChitAOAGCs zm&m=gW8cft#18c~ycFz_(rx%d6c0ukOIX>C78#obKdhGK^O*K5lGkii^9q}^6_?RT zdo4B_eLZqNB3g@%4l2}6pw)T(TE%4LhJmhJ+R_lza+x9R51w#K{zX?EIXmdi3rbrd zHY}Iwe6Do3Lblq;7C|w#%hnM_oKnVnI|8lc5Niq-1MBT3whKeWR4pZg`Bb8+msnV| z1NkC?tNhS6{-hGk*>P-PUGB?`wh5F3x9B2H^H7I*GD|@AMxlj&h(oM5RH!@DJZOF1 zzbzhOU}=33tB2sTgYIYKQrTXy8&q5QH1r~j>W<;{aOzKOMaIQ6^iV{BCbfo+2W@NG z2-dwyFh;QF`ZHI^K0KGIBn8+)l4k33mlhh~ z`Q2Orvdn75g(ksMqaVvK%03-=Z zIlq*y5g>7er?r4J(m*k|=yV)6L6%t2W(SID}8QXaYHc;^J>-IuGe1Pk`Bw2+VA$^|Idn1KVs$QTAJgs#Xc0vH>~ zLKxVCWaAS)4Gn7~&_b%k?DwEF?&q|S)+r8X=bQv|PXo-ZJqTUlx1kO@1)kETLr-_* zt0BVLahO!u%)l|G*Kq25H8%d%b&D72ymrvw2Tke-?-nXd3OzHhRvWal=yktAxjA5U z@4ujjw$k#cTJHGsU}0`Dj-lEELbkp9X=2ub;`n=mjVV1f-c;f^AcB> z6Y5}@9h;1CMLW*W>rO)i#5y%UpC|oF|6Nu5tx&lMF^7g%;`ktaM^M)qPuim|YLl9M z*QY96&qAL;GN0yrdgqdfmEtU(X^H7HF^A=4giTG>jeK^g(P=&d2brX75RFspcf^}i z-YH-3PAz>@!7D9FUnR6`e46vbZ;P(fBSk-fDgRQu^s(kcecs@e`_SMEgr3TiY^C;_ z)yqzX(0yRatUJxIz&pA`KP{`!i9cijIt@a(=RnfP z(J-)9_^*sEb+4uu9%ySN7bKh*Xe%)`>T!jlSzC_eF1B zre{Br=&hk2;}E3`TyA26UOxv=6%HnP)adKxeuZ_%zsSvD=u;D$e-o=upcKt{=q*5Y z+h;9Qrsl!B$H)xpG;7g>cX_j-Jo$n;-l=Dqq1ZRM0N_@39;Cnj?1~G zyHHn=4&X|{qlsPOwvI=~{QQ&kPgGF8H7M-g%~@M1{qwCKMcT2h4*sX8p@QG`H;Z@D)|cWEKx)9iyMzIFt$+i&e({TlI_>JDdw zo{BcBhNM-&@@c~pw7s?H=86XIEZctX*4WbJA;>YkD6y$?sF(*-Us5iW(34Z6L1__~ ze}Gu@f+`{xM*r@u+ecN6&~l9;%~|`&g~3~{|BSur+FQ`4_%J`1PGbdND;=Z@AW{g% zI_kP?IZ%V*vwbbtQ}Y5a{9Q_g0t5-T0|A;4DloB?s*8tu`4rg)pP@C1WI2WrPAzF; zbH>mhg_lml+Y#DByN|1#lSQdode2K(N1x&tez)S&eU{_IDX8i$AZk9>&t~Y=d^mNO zDd1=wo^7cqhY9E=5CJYA_bv=K4l$NdL$11Dx_X;g-ev$Y-!IxTo(8#~pFrCS#i6_I z;~~*BAW81#!;vG>5M_64AEh;)f5RHOBl(=R_sU;oSj^j{t)iSpKdhJxI%Sj5MsaOGJ~M!eNF@Eu&7Y8 zvduxzLaKB)P=x>K$-sYo+5ff-R1H{0i-jtTKNpZk_di}kg=zqf83C%^ylM^51keHP zIdM?TV1dn;MSn{IsOAmK)Xr3D`3WQnG#nW4yV{}L*BA!qEg+ymH9!lDiC?G+OPnV} z(=S7SPX4b`i~q(p8*V7+k-2%MH?s!z_2`FUL1EzzAPzLFE?F-C&i)O+6x65BI+2^A zhq%4W3yGej5PmvX8e>4tH?LVk_fr9<+@Sy$AQ@Nn7RdV1f+O@PS+Iw<2OO_Zhw)!` zghXF|`n`J4%N=*%El}@9Z(x%%{pd&4&ZF;RM9(>JnuI7O$Qpj5c?%dC_%qOfq&)|1 z*eqT6e>j=>uYGj8y?g=^wUEo||IO5(XzsxNDR0OK@Z|l-cZSe~bE1$VrLw3&QcD2l z3@ZhSl^zrjWYz_QsbGe;7?k$pIb{F`Btg*x_wc~qQ@{&eS1?D{56I}Dq-&baztOr1 zNR)eezbVY@Tczvp`?FLnt{*8Ha=S}rRmRKd9yaYZk;C2v#PA{x-2VGS+q)dnVw&7@vzOREX?|)C)h_g3iuD+bdkZKuj!RKGqvPa?noWc z5o^0DGNc4L4ICuKKd0XPmZk_&hveb)xTB3Adt0t*N0I90llYEP30eCw4P7GmJU5Al zH$3i(wCB#5_)n`$OJo(Fge-2eNlfLrmZ|Jl2@q_gSmoM3Gs^(IyC$->N`x8j#kh>1 zCZfB)bDw)Kz}zKoKf1dzGIujcw?w!xF(=^g{YYsbgJgjzTo~+tB>(1$Q)$)Qib$b% zwqqikEpNmdFu|4?ZtC`XnB#hHsapCMxl6yoVN>A5=zwzitS6U@h?tu7*;^2{_BCv> zz>~8?2%cx^e5p)(y8oRU&PG9Jt4R}`E2dAj4AKwt7u$Ph_<2}jHrUm6a`EF34NyUL zPYaq26xMNcP@1~!`q1DN8Yt}0`hPB!@uS?Kj?{|2bGv7umYej%ugyTC91>Z5#&m`7 z6UNHL4X6v(O*Q$w!?p3UJCW&?+qcX@lX|whnFMjPCn4j{&6<3mu>TB|)eGc932m*2 z_EbLhF{>dJrZkf7>i|S6dKV}W;KQ-fgoE_Gbq7A6LRM0$ z3l-EAZ|9ffixXk1+2maDq*CP-obkZN@m#z@FFOxVD{pAyqK>Z&HU%oI{raH7lV8hh z3STlDwKY~ut=y7Wmvjm!^QjrmH?$a%G*0Mcr=kwK{)Q>5Xu5^nH$;<+UaqudIC_~5 zM>M{w45lB3I6vb_+X8Yd4A!}fVETeL`;HP~y)&;2NatKD@iobae&4&N1_79fmb{{i z%j3Ccp{|;#F)CB`piCj(wFI#Qqbhtdcv%E0Eb`IMV&Bnr8~lVI4RnX_??t8*CH_@R zsw_3jmW$p)D<%j~eRMo{^V4R}{(rrd;deYsv7DWnBw%rB9cHQEXeW&2gAH82?McF{I zWByQH=v^kc#|gOh4x{S*53(V(&p;$qIlSdq*_gOnA*X+a#=7}5`&G$Os(hEJ9wun; zQfUS~BLwLv|92qs4`8_j5B~a&2p9ne<8GomX9+9FlL2o$V*BNNM+b^FJTAa4<2I)X zINk&dR4L0OkE%=_<(p@lkc_(bENZhMNTYODk^Q5c>BE59P%iduw^S3dY07AWFl;r! z`LM8A9*xK4;-BKT@^&Gp!)fVA6Q&74=jRqk;30;l8J_#9rh91(5>W}VfHX&DJcZ(h@!a6U+vq`I5 z3KRI9LK>cUi!qxB5eDa{`%*NkjKM_#B}I!59%kDF>PE;eDyYMH&Rz?TFP1KBDDuIe z+=m=Qg(>uKja$yRmRVx$I9?9!8xxWAzCl|dUSt0KhJ9lU#vT3Z#*UZ(80T{pKjio% zt^R!ear?O4Y-%Px+=ML}WEQ|7i|IeY(%i$d_2?jDAv@C=r|&PrqV{(Fptrr$|JOplTp7|J zq&IzdpUA2?k%kDBt~yerS{iS7XV_co)Ep?+2aDuIcLq@3;XI#1&uB4>d7ZwurJ8ON zVRb?@6zCLz;_VGSXdweo>;Ire%vZxZJ}dK93@^LNMmi~jlq+;z+IIwc>G6QEN2_*U z0LO8@;6UAgw8*y+Su(DLgWpAfZ}j~-yYt5 zkNE`4|59LBOf~7)a4-&h-aW=Wx@FKLRKGukv*;|!p@MYoe$O3p`}uR1b%%-DIbdH- zL2Cv>`&s2xMr&})!X-5OiBV0}nFWN45w9_QZODq0+I z8AK%(Ehg3Wz2QxcuMedv8rJkeR&a?AYBvz?G#Q~ZPFwuY zQjU41bY?yU*~}bd?hViJ>BdX;6O%a+@UL)krDb^8z}1ME_wM%+0uJkN7N|VTEg*iw zJRkp~kG=%CXfu5go$_ULyETB6S*_a=4d_kZ_Ow``-S$6fKFWr+^L$hhY5H!fLWTKp zupER22Q;WhNx)&u&FsA`YA6G<*sJELizE>r1i!>6|3iF8KFCzUc2_ckR6XXIGzsQ z?@;iEt%W?Sb|`Ws8VC(kx?XK^Xc{kD+4p!{CJo*B`@Ks3^qm#APrK4<5l!3Z*;5=f zk$H{)#BMd-QYsvJj~8bfVx+?3<(D3x zxeDDeyM`b7=5hgMD)q_6WHEjtm)69?R=gekLxsSx;HS%8;~%|JTo~WeuN_|fHqYAh zn%etjQW+;S_^>OyLsapMI;~{k)&?sJXYpEp$Rugya{0$fTPd;lXT#>hmS=dVA;;o} zy+b!$c2XGXKiaZ)w5BpFI-fh&eS$SnvM}|@;ICrjIs{GG&8y@)7V4V}eabiWa7^x# zTnc#zQ|_guEUD9A!@lDMMWMoOG%-&Do=oeHmxFZ{e?NcQ**L+#NI(Ck19`Vd{MBzK zbCMc-Jic*2qCy>*{2^sAS&y6dOedCRYgX>R5%7`-=$Swv*Zhq8KY*Qlz(8Y{*vgpU zpC38C=UXbdtIu{JMeTs<5f%DZGyb!>*~T;deAB|lnk?1nZq<6wB4VoS%#`IG_ntWP zs_NaRNnUSzgG$SIw#>${ovZr#%TkgESv^vJKIOL&ED8Eio$b)4|GRA9( zte2E!^$%f&K+tzq0mmKbsUr^8MCTUc)3U>uZyqs7IZ7tceV}9#K13fF;Zc}tP*5vo zHTJDI@J0iiaN#eT!Oe!8@uom!SGJgL!;#yDwVFI1NR`Pan?a`jW*KsijD$QM?cY?K`g9{UXT^srS&iDLX#W z!DMY{B~^4a69$Tx%q7C8ptPilJ;xETjf!pEcQ*WqP$p|!G`M!RWqmSZFp$hcQ|}CQBAqsb(iSSIq0bh#BK5>L3y6$AKc{j- zxqV;@`=9L?9mHi?GM%vJ$2&}~Gd+h`nf}9eTC9Ca<`{JHc_T9O4MqmRi3QkHs&#x_ z;H+J7Fy5Dr7LVWZa~mdZ>|{HT_dk{}0@lH8A-50@YVZxaG?3_N)su0g2$RrohX)L~ z0z3}?Qb*{4qP;QTUl9(Cf8^XW+cWYP8)TiSaWwLgq+wW2ig=7}>}r}uINCADS?+H) zz<=~~%iN^j4Sl$~;WOF%V2pfoClw_qnDQ?XZIUb}B42O^a9m7<5~ik*?rOQg3}HWS zw+C#WwrlvE-?N;2o&>vWv{7k0@^=x5vhi}x+AlZTVS`#EC~08BVXdxe_5gj2OOwy@ z0lW&W?osQ0aQ+Ait8rdGOXwh~eYoDqQ4 zv0VxWyeh~Y9D7*0MC$EI?_}(7?d-D}Weu9`KH~3rBoX?)ixj!;#xGZEup_n00oFvcv9V$p-w3{mq(*egjaC0?-ti=xrkf`_%eKu-SVk#Q-S zD^OU-S<@7^x86Ky%5KX%tQ~@x)J|DG=Ki9J3%y;T4pWzMJKesM-4VFJ6QU(!j9E|i zbiHgH-u5s;qKu`N<=WRWb~xK@bSbs3(KpB+rdCA%>MsG%^|1fzf7URWxnki+9Y!cj zOMCBYGp56h!^(ba+51e7@rcYoWr;W8!qoi;fd<@~kap`xa5+hH*1#|Lsq&5hpmlp{TaV7#T(U5MP!*1 zV56ZGAaTU;R$c#!^O0KKhHopf-1Z#a>z=`e0D<%Kfajj8j`lBV$a?2rxGie&mWHM2 zG6ZdZxTv4?geGoPR16bXWgJ82&HN`Txu1Y5VUdp)`kECS|cvQIalWI(1hQ z$4g_PCQB3iWi-RUVNu%SMA!EwioPDLmE%`A(m!@f6dieBcYBMgNIoWI=BAJLz}M(Y zLc>h?9HCeh4!q@liW9CoIhnn?s<3u0$Am6iX_aX12Ti=qrRt3pfUm5U++M08ML3Zg zS_K*dF`>PcD1@N4Woy9TCas&9cc#F&^Ptjd*(|zXt*U7Ekx>eX*yr_$dI2=2T67*) z+3U`s7dE!8!`~A);_^?iwI+$cOlgzzOBjy^Y*~qf!T`78k{pv7o1P3f!>%8+eJ_Dl z)N9HwwXPpcEFhEyMj}gebbB6+v<`K+7>Z03OQ4w$e!k;WHov(#e^60-aEPI|v$TGP;MGcOi?mLfd^d?N@ zX1~b`aNQoxSV{Tl8jRPYS^%#CSPL5m3)~+izIT>jl=HU{E#XI5YZ&dF4hC}@ z%C-+bXUn&U{@XcxY)oU!_VR=?Z_5$&`Y#d=>45O!}($-baReQKv30J zyI*Y&!F!X4tYG)>^=#0?gG6uB#4%6!N8%EN!3dqCT(J}KXmEG}Q8vEa_9Uvz1?*UT1? zAHT;C^!!MM^HD5K&h*y+9(;MAPs=Ufc;6MI*CuM<=z9yUTz2Cdx&t!v2sZLSzC{L%p0>!t1HPA(8`-k#wx6Hj{sba2tp3Ws9rXSm?_B^66tN8Q!Qv999@bA)dEg!h zi3<>bv*MQvWY0&#|J&ZiPHp@f&+b_g-*7$IxoGusmSIL~*5< zYu7m8s7Nz-G{1Spz-shE*{y$w8$)S%Tt$Gj`?qH7KYi62r-r%i=h}pAze>=LvrlWWH=d7T>V(d{TBSjAQE=6TkEbPbsJSwcSgqTVYr9? zTEYrX{ioUX*vw-zDv0vNf$kZ_a@#X>k~A?7d@YIejX{quNM>#~#1NGAtApPD98JfY zNQQ@q6_)oMA3GPg_9q=RXeM{0A2meg?C%<-qhu0xM^kdv{d+j*0Q`rkus3~Tf}q@Q bWV)}mKQ2CeyAciO1kzI1Q!Be`{rrCcH`TNs literal 0 HcmV?d00001 diff --git a/trick_sims/SIM_waterclock/models/graphics/Makefile b/trick_sims/SIM_waterclock/models/graphics/Makefile new file mode 100644 index 00000000..d3e0cbc0 --- /dev/null +++ b/trick_sims/SIM_waterclock/models/graphics/Makefile @@ -0,0 +1,6 @@ + +all: + mvn package + +clean: + rm -rf build diff --git a/trick_sims/SIM_waterclock/models/graphics/pom.xml b/trick_sims/SIM_waterclock/models/graphics/pom.xml new file mode 100644 index 00000000..2651af39 --- /dev/null +++ b/trick_sims/SIM_waterclock/models/graphics/pom.xml @@ -0,0 +1,121 @@ + + + + 4.0.0 + + trick-java + trick-java + 23.0.0-beta + + trick-java + + https://github.com/nasa/trick + + + UTF-8 + 1.8 + 1.8 + + + + + junit + junit + 4.13.1 + test + + + + + + WaterClockDisplay + + build + + + org.apache.maven.plugins + maven-javadoc-plugin + 3.1.1 + + ${java.home}/bin/javadoc + ../../share/doc/trick/java + + + + + + + + + + maven-clean-plugin + 3.1.0 + + + + + maven-resources-plugin + 3.0.2 + + + + maven-compiler-plugin + 3.8.0 + + + -g + -Xlint:unchecked + -Xlint:deprecation + + + + + + + org.apache.maven.plugins + maven-jar-plugin + 3.1.0 + + + + true + lib/ + WaterClockDisplay + + + + + + + maven-surefire-plugin + 2.22.1 + + + + maven-install-plugin + 2.5.2 + + + + maven-deploy-plugin + 2.8.2 + + + + + maven-site-plugin + 3.7.1 + + + + + + + + diff --git a/trick_sims/SIM_waterclock/models/graphics/src/main/java/trick/waterClockDisplay/WaterClockDisplay.java b/trick_sims/SIM_waterclock/models/graphics/src/main/java/trick/waterClockDisplay/WaterClockDisplay.java new file mode 100644 index 00000000..c3cbcace --- /dev/null +++ b/trick_sims/SIM_waterclock/models/graphics/src/main/java/trick/waterClockDisplay/WaterClockDisplay.java @@ -0,0 +1,779 @@ +import java.awt.Color; +import java.awt.Graphics2D; +import java.awt.Graphics; +import java.awt.RenderingHints; +import java.awt.event.ActionEvent; +import java.awt.event.ActionListener; +import java.io.BufferedOutputStream; +import java.io.BufferedReader; +import java.io.DataOutputStream; +import java.io.IOException; +import java.io.InputStreamReader; +import java.net.Socket; +import java.util.*; +import javax.swing.BoxLayout; +import javax.swing.JButton; +import javax.swing.JFrame; +import javax.swing.JLabel; +import javax.swing.JPanel; +import javax.swing.JSlider; +import javax.swing.event.ChangeEvent; +import javax.swing.event.ChangeListener; +import javax.sound.sampled.*; +import java.net.URL; +import java.awt.event.KeyEvent; +import java.awt.event.KeyListener; +import java.awt.geom.AffineTransform; +import java.awt.geom.Area; +import java.awt.geom.Rectangle2D; +import java.awt.Font; + +import javax.swing.text.NumberFormatter; +import java.text.NumberFormat; +import javax.swing.JFormattedTextField; +import java.awt.Dimension; + +import javax.swing.BorderFactory; +import javax.swing.border.EtchedBorder; +import java.awt.Component; + +class ScenePoly { + public Color color; + public int n; + public double[] x; + public double[] y; +} + +class RangeView extends JPanel { + + private int scale; + private Color bucket_color; + private Color spout_color; + private Color water_color; + private Color float_color; + private Color clock_color; + private Color clock_display_color; + private Color tick_color; + private Color gear_color; + + // Origin of world coordinates in jpanel coordinates. + private int worldOriginX; + private int worldOriginY; + + private double spout_rate; + private double overflow_rate; + private double intake_depth; + private double timer_depth; + private double intake_lvl; + private double timer_lvl; + private double intake_vol; + private double timer_vol; + private double spout_height; + private double overflow_height; + private double input_flow; + private int current_tick; + + private double clock_float_height; + + // Controls + private double input_flow_select; + public boolean new_input_flow; + + /** + * Class constructor. + */ + public RangeView( int mapScale) { + + setScale(mapScale); + + spout_rate = 0.0; + overflow_rate = 0.0; + intake_depth = 0.0; + timer_depth = 0.0; + intake_lvl = 0.0; + timer_lvl = 0.0; + intake_vol = 0.0; + timer_vol = 0.0; + spout_height = 0.0; + overflow_height = 0.0; + input_flow = 0.0; + input_flow_select = 0.0; + current_tick = 0; + + new_input_flow = false; + + bucket_color = new Color(150,75,0); + spout_color = new Color(128,128,128); + water_color = new Color(0,0,128); + float_color = new Color(0,0,0); + clock_color = new Color(204,204,0); + clock_display_color = new Color(192,192,192); + tick_color = new Color(0,0,0); + gear_color = new Color(128,128,128); + + workPolyX = new int[150]; + workPolyY = new int[150]; + + tooth = new ScenePoly(); + tooth.color = float_color; + + tooth.x = new double[] { 0, -0.15, -0.25 , -0.15 , 0 }; + tooth.y = new double[] { 0, 0 , 0.075 , 0.15 , 0.15 }; + + tooth.n = 5; + + clock_float_height = 8; + } + + private ScenePoly gear; + private ScenePoly tooth; + + private int[] workPolyX, workPolyY; + + public void fillSceneRect(Graphics2D g2d, Color color, double x, double y, double w, double h) { + g2d.setPaint(color); + g2d.fillRect( (int)(worldOriginX+scale*(x)), (int)(worldOriginY-scale*(y)), (int)(scale*w), (int)(scale*h)); + } + + public void fillScenePoly(Graphics2D g, ScenePoly p, double angle_r , double x, double y) { + for (int ii = 0; ii < p.n; ii++) { + workPolyX[ii] = (int)(worldOriginX + scale * + ( Math.cos(angle_r) * p.x[ii] - Math.sin(angle_r) * p.y[ii] + x)); + workPolyY[ii] = (int)(worldOriginY - scale * + ( Math.sin(angle_r) * p.x[ii] + Math.cos(angle_r) * p.y[ii] + y)); + } + g.setPaint(p.color); + g.fillPolygon(workPolyX, workPolyY, p.n); + } + + public void fillSceneOval(Graphics2D g2d, Color color, double x, double y, double w, double h) { + g2d.setPaint(color); + g2d.fillOval( (int)(worldOriginX+scale*(x-w/2)), (int)(worldOriginY-scale*(y+h/2)), (int)(scale*w), (int)(scale*h)); + } + + public void setSpoutRate(double x) { + spout_rate = x; + } + + public void setOverflowRate(double x) { + overflow_rate = x; + } + + public void setIntakeDepth(double x) { + intake_depth = x; + } + + public void setTimerDepth(double x) { + timer_depth = x; + } + + public void setIntakeWarerLevel(double x) { + intake_lvl = x; + } + + public void setTimerWarerLevel(double x) { + timer_lvl = x; + } + + public void setIntakeWaterVol(double x) { + intake_vol = x; + } + + public void setTimerWaterLevel(double x) { + timer_vol = x; + } + + public void setSpoutHeight(double x) { + spout_height = x; + } + + public void setOverflowHeight(double x) { + overflow_height = x; + } + + public void setInputFlow(double x) { + input_flow = x; + } + + public void setInputFlowSelect(double x) { + input_flow_select = x; + } + + public double getInputFlowSelect() { + return input_flow_select; + } + + public void setCurrentTick(int x) { + current_tick = x; + } + + public void setScale (int mapScale) { + if (mapScale < 2) { + scale = 2; + } else if (mapScale > 128) { + scale = 128; + } else { + scale = mapScale; + } + repaint(); + } + + private void doDrawing(Graphics g) { + Graphics2D g2d = (Graphics2D) g; + + RenderingHints rh = new RenderingHints( + RenderingHints.KEY_ANTIALIASING, + RenderingHints.VALUE_ANTIALIAS_ON); + + rh.put(RenderingHints.KEY_RENDERING, + RenderingHints.VALUE_RENDER_QUALITY); + + int width = getWidth(); + int height = getHeight(); + + worldOriginX = (width/2) - (int)(scale * 0); + worldOriginY = (height/2) + (int)(scale * 0); + + + // Draw intake Bucket + double bucket_length = 2.0; + double bucket_width = 0.15; + double intake_bucket_origin_x = 4; + double intake_bucket_origin_y = 1.7; + double intake_bucket_left_wall_x = intake_bucket_origin_x - bucket_length - bucket_width; + double intake_bucket_bottom_x = intake_bucket_origin_x - bucket_length - bucket_width; + double intake_bucket_bottom_y = intake_bucket_origin_y - bucket_length; + double spout_width = 0.175; + double spout_length = 1.0; + double spout_pipe_x = intake_bucket_left_wall_x - spout_length + bucket_width; + double spout_pipe_y = intake_bucket_origin_y - bucket_length + spout_width + ((spout_height / intake_depth) * bucket_length) - spout_width/2; + double overflow_pipe_x = intake_bucket_origin_x; + double overflow_pipe_y = intake_bucket_origin_y - bucket_length + spout_width + ((overflow_height / intake_depth) * bucket_length) - spout_width/2; + fillSceneRect( g2d, + bucket_color, + intake_bucket_origin_x, + intake_bucket_origin_y, + bucket_width, + bucket_length); + fillSceneRect( g2d, + bucket_color, + intake_bucket_bottom_x, + intake_bucket_bottom_y, + bucket_length + (2 * bucket_width), + bucket_width); + fillSceneRect( g2d, + bucket_color, + intake_bucket_left_wall_x, + intake_bucket_origin_y, + bucket_width, + bucket_length); + fillSceneRect( g2d, + spout_color, + overflow_pipe_x, + overflow_pipe_y, + spout_length, + spout_width); + fillSceneRect( g2d, + spout_color, + spout_pipe_x, + spout_pipe_y, + spout_length, + spout_width); + + //Draw timer bucket + double timer_bucket_origin_x = 1.3; + double timer_bucket_origin_y = -0.5; + double timer_bucket_left_wall_x = timer_bucket_origin_x - bucket_length - bucket_width; + double timer_bucket_bottom_x = timer_bucket_origin_x - bucket_length - bucket_width; + double timer_bucket_bottom_y = timer_bucket_origin_y - bucket_length; + fillSceneRect( g2d, + bucket_color, + timer_bucket_origin_x, + timer_bucket_origin_y, + bucket_width, + bucket_length); + fillSceneRect( g2d, + bucket_color, + timer_bucket_bottom_x, + timer_bucket_bottom_y, + bucket_length + (2 * bucket_width), + bucket_width); + fillSceneRect( g2d, + bucket_color, + timer_bucket_left_wall_x, + timer_bucket_origin_y, + bucket_width, + bucket_length); + + + //Draw water flow + double flow_width = 0.15; + double spout_flow_height = intake_bucket_origin_y - timer_bucket_origin_y + (bucket_length - (intake_bucket_origin_y - spout_pipe_y)) - spout_width/2; + double overflow_height = 20; + if (spout_rate != 0.0) { + fillSceneRect( g2d, + water_color, + intake_bucket_left_wall_x - spout_length + bucket_width - flow_width + 0.015, + spout_pipe_y - spout_width/2, + flow_width, + spout_flow_height); + } + + if (overflow_rate != 0.0) { + fillSceneRect( g2d, + water_color, + intake_bucket_origin_x + spout_length, + overflow_pipe_y - spout_width/2, + flow_width, + overflow_height); + } + + + //Draw water level + double intake_capacity = intake_lvl / intake_depth; + double intake_offset = ((1 - intake_capacity) * bucket_length) ; + double timer_capacity = timer_lvl / timer_depth; + double timer_offset = ((1 - timer_capacity) * bucket_length) ; + fillSceneRect( g2d, + water_color, + intake_bucket_origin_x - bucket_length - 0.01, + intake_bucket_origin_y - intake_offset, + bucket_length + 0.03, + intake_capacity * bucket_length); + //fill in pixel gaps due to rounding error + if(intake_lvl > 0.0) { + fillSceneRect( g2d, + water_color, + intake_bucket_origin_x - bucket_length - 0.01, + intake_bucket_origin_y - bucket_length + 0.05, + bucket_length + 0.03, + 0.05); + } + + + fillSceneRect( g2d, + water_color, + timer_bucket_origin_x - bucket_length - 0.01, + timer_bucket_origin_y - timer_offset, + bucket_length + 0.03, + timer_capacity * bucket_length); + //fill in pixel gaps due to rounding error + if(timer_lvl > 0.0) { + fillSceneRect( g2d, + water_color, + timer_bucket_origin_x - bucket_length - 0.01, + timer_bucket_origin_y - bucket_length + 0.05, + bucket_length + 0.03, + 0.075); + } + + //Draw external source pipe + fillSceneRect( g2d, + spout_color, + intake_bucket_origin_x - bucket_length/4, + intake_bucket_origin_y + 0.8, + 7, + spout_width); + + if(input_flow > 0) { + fillSceneRect( g2d, + water_color, + intake_bucket_origin_x - bucket_length/4 - flow_width + 0.03, + intake_bucket_origin_y + 0.8 - spout_width/2, + flow_width, + 0.815 + bucket_length - spout_width/2); + } + + //Draw float + double float_base_height = 0.25; + fillSceneRect( g2d, + float_color, + timer_bucket_origin_x - bucket_length + bucket_length/4, + timer_bucket_origin_y - timer_offset + float_base_height, + bucket_length/2, + 0.25); + + double float_arm_origin_x = timer_bucket_origin_x - bucket_length + bucket_length * 0.45; + double float_arm_origin_y = timer_bucket_origin_y - timer_offset + clock_float_height + float_base_height; + fillSceneRect( g2d, + float_color, + float_arm_origin_x, + float_arm_origin_y, + bucket_length*0.1, + clock_float_height); + + double tooth_ratio = 2.25 * tooth.y[3]; + for(double ii = bucket_length/2.0; ii < (clock_float_height) ; ii += tooth_ratio) { + fillScenePoly(g2d, tooth, 0.0, float_arm_origin_x, ii + float_arm_origin_y - clock_float_height); + } + + //Draw Gear + double clock_origin_x = float_arm_origin_x - 1.85; + double clock_origin_y = 2.0; + double clock_diam = 2.5; + double gear_x = clock_origin_x + 0.85; + double gear_y = clock_origin_y; + double gear_width = 1.5; + double gear_noise = 0.3; //initial gear rotational offset + gear = new ScenePoly(); + gear.x = new double[] { 0, 0, 0, 0, 0}; + gear.y = new double[] { 0, 0, 0, 0, 0}; + gear.n = 5; + for(int ii = 0; ii < gear.n; ++ii) { + gear.x[ii] = tooth.x[ii]-(gear_width*0.5); + gear.y[ii] = tooth.y[ii]; + } + gear.n = 5; + double gear_circ = (Math.PI * (gear_width - (2*tooth.x[2]))); + int gear_count = (int)(gear_circ / tooth_ratio); + double gear_offset = ((timer_capacity * bucket_length) / gear_circ) * 2 * Math.PI; + gear_offset *= 3; //I don't know why, but this magic number makes the gear animation sync with the rack + gear.color = gear_color; + for(int ii = 0; ii < gear_count; ++ii) { + fillScenePoly(g2d, gear, (ii + gear_noise + gear_offset) * (360/gear_count) * (Math.PI/180), gear_x, gear_y); + } + fillSceneOval ( g2d, + gear_color, + gear_x, + gear_y, + gear_width*1.05, + gear_width*1.05); + + //Draw Clock + fillSceneOval ( g2d, + clock_color, + clock_origin_x, + clock_origin_y, + clock_diam, + clock_diam); + + double clock_display_length = 1.0; + double clock_display_origin_x = clock_origin_x - clock_display_length/2.0; + double clock_display_origin_y = clock_origin_y; + fillSceneRect( g2d, + clock_display_color, + clock_display_origin_x, + clock_display_origin_y, + clock_display_length, + clock_display_length); + + g2d.setFont(new Font("", Font.PLAIN, 24)); + g2d.setPaint(tick_color); + g2d.drawString ( String.format("%02d",current_tick), (int)(worldOriginX+scale*(clock_display_origin_x + clock_display_length*0.25)), (int)(worldOriginY-scale*(clock_display_origin_y - clock_display_length*0.6))); + + // Draw Information + g2d.setPaint(Color.BLACK); + g2d.setFont(new Font("", Font.PLAIN, 12)); + int string_height = 220; + g2d.drawString ( String.format("Input Flow Rate: [%.4f] cm^3/s",input_flow), 20,string_height+=20); + g2d.drawString ( String.format("Intake Water Level: [%.4f] cm",intake_lvl), 20,string_height+=20); + g2d.drawString ( String.format("Intake Water Volume: [%.4f] cm^3",intake_vol), 20,string_height+=20); + g2d.drawString ( String.format("Intake Overflow Rate: [%.4f] cm^3/s",overflow_rate), 20,string_height+=20); + g2d.drawString ( String.format("Intake Spout Rate: [%.4f] cm^3/s",spout_rate), 20,string_height+=20); + g2d.drawString ( String.format("Timer Water Level: [%.4f] cm",timer_lvl), 20,string_height+=20); + g2d.drawString ( String.format("Timer Water Volume: [%.4f] cm^3",timer_vol), 20,string_height+=20); + g2d.drawString ( String.format("Current Time Tick: [%d]",current_tick), 20,string_height+=20); + } + + @Override + public void paintComponent(Graphics g) { + super.paintComponent(g); + doDrawing(g); + } +} + +class TrickSimMode { + public static final int INIT = 0; + public static final int FREEZE = 1; + public static final int RUN = 5; +} + +class InputFlowCtrlPanel extends JPanel implements ChangeListener { + + static final int FLOW_MIN = 0; + static final int FLOW_MAX = 10000; + static final int FLOW_INIT = 1000; + + private RangeView rangeView; + private JSlider inputFlowSlider; + + public InputFlowCtrlPanel (RangeView view) { + rangeView = view; + setLayout(new BoxLayout(this, BoxLayout.X_AXIS)); + + JLabel sliderLabel = new JLabel("", JLabel.CENTER); + sliderLabel.setAlignmentX(CENTER_ALIGNMENT); + + inputFlowSlider = new JSlider(JSlider.HORIZONTAL, FLOW_MIN, FLOW_MAX, FLOW_INIT); + inputFlowSlider.addChangeListener(this); + inputFlowSlider.setToolTipText("Input Flow Rate (cm^3/s)."); + inputFlowSlider.setMajorTickSpacing(1000); + inputFlowSlider.setMinorTickSpacing(100); + inputFlowSlider.setPaintTicks(true); + inputFlowSlider.setPaintLabels(true); + + add(sliderLabel); + add(inputFlowSlider); + } + + @Override + public void stateChanged(ChangeEvent e) { + JSlider source = (JSlider)e.getSource(); + if (!source.getValueIsAdjusting()) { + rangeView.setInputFlowSelect( source.getValue()); + rangeView.new_input_flow = true; + } + } + + public void setValue(int value) { + if (value > FLOW_MAX) value = FLOW_MAX; + if (value < FLOW_MIN) value = FLOW_MIN; + inputFlowSlider.setValue(value); + } +} + +class ControlPanel extends JPanel implements ActionListener { + + private RangeView rangeView; + private InputFlowCtrlPanel inputFlowCtrlPanel; + + public ControlPanel(RangeView view) { + + rangeView = view; + setLayout(new BoxLayout(this, BoxLayout.X_AXIS)); + + JPanel labeledInputFlowCtrlPanel = new JPanel(); + labeledInputFlowCtrlPanel.setLayout(new BoxLayout(labeledInputFlowCtrlPanel, BoxLayout.Y_AXIS)); + JLabel inputFlowCtrlLabel = new JLabel("Input Flow Rate (cm^3/s)"); + inputFlowCtrlLabel.setAlignmentX(Component.CENTER_ALIGNMENT); + labeledInputFlowCtrlPanel.add(inputFlowCtrlLabel); + inputFlowCtrlPanel = new InputFlowCtrlPanel(rangeView); + labeledInputFlowCtrlPanel.add( inputFlowCtrlPanel ); + add(labeledInputFlowCtrlPanel); + + } + + public void actionPerformed(ActionEvent e) { + String s = e.getActionCommand(); + switch (s) { + default: + System.out.println("Unknown Action Command:" + s); + break; + } + } +} // class ControlPanel + +public class WaterClockDisplay extends JFrame { + + private RangeView rangeView; + private BufferedReader in; + private DataOutputStream out; + private JPanel panelGroup0; + private JPanel panelGroup1; + private ControlPanel controlPanel; + + public WaterClockDisplay(RangeView arena) { + setTitle("Water Clock"); + + rangeView = arena; + + panelGroup1 = new JPanel(); + panelGroup1.setLayout(new BoxLayout(panelGroup1, BoxLayout.X_AXIS)); + panelGroup1.add(rangeView); + + controlPanel = new ControlPanel(rangeView); + + panelGroup0 = new JPanel(); + panelGroup0.setLayout(new BoxLayout(panelGroup0, BoxLayout.Y_AXIS)); + panelGroup0.add(panelGroup1); + panelGroup0.add(controlPanel); + + add(panelGroup0); + + rangeView.setScale(64); + + setSize(800, 500); + setLocationRelativeTo(null); + setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); + setFocusable(true); + + } + + public void connectToServer(String host, int port ) throws IOException { + Socket socket = new Socket(host, port); + in = new BufferedReader( new InputStreamReader( socket.getInputStream())); + out = new DataOutputStream(new BufferedOutputStream(socket.getOutputStream())); + } + + public void drawRangeView() { + rangeView.repaint(); + } + + private static void printHelpText() { + System.out.println( + "----------------------------------------------------------------------\n" + + "usage: java jar WaterClockDisplay.jar \n" + + "----------------------------------------------------------------------\n" + ); + } + + public enum ModelState { INACTIVE, READY, ACTIVE } + + public static void main(String[] args) throws IOException, InterruptedException { + + String host = "localHost"; + int port = 0; + boolean boom = false; + + int ii = 0; + while (ii < args.length) { + switch (args[ii]) { + case "-help" : + case "--help" : { + printHelpText(); + System.exit(0); + } break; + default : { + port = (Integer.parseInt(args[ii])); + } break; + } + ++ii; + } + + boolean go = true; + double dt = 0.100; // Time between updates (seconds). + double spout_rate = 0.0; + double overflow_rate = 0.0; + double intake_depth = 0.0; + double timer_depth = 0.0; + double intake_lvl = 0.0; + double timer_lvl = 0.0; + double intake_vol = 0.0; + double timer_vol = 0.0; + double spout_height = 0.0; + double overflow_height = 0.0; + double input_flow = 0.0; + double flow_select = 0.0; + int tick = 0; + + // Outbound command variables + int simMode = 0; + boolean standalone = false; + + int mapScale = 32 ; // pixels per meter. + + RangeView rangeView = new RangeView( mapScale); + WaterClockDisplay waterClockDisplay = new WaterClockDisplay( rangeView); + waterClockDisplay.setVisible(true); + waterClockDisplay.drawRangeView(); + + if (go) { + if (port == 0) { + System.out.println("No variable server port specified."); + printHelpText(); + System.exit(0); + } + + // Connect to the Trick simulation's variable server + System.out.println("Connecting to: " + host + ":" + port); + waterClockDisplay.connectToServer(host, port); + + waterClockDisplay.out.writeBytes("trick.var_set_client_tag(\"WaterClockDisplay\") \n"); + waterClockDisplay.out.flush(); + + // Have the Variable Server send us the simulation mode ONCE. + waterClockDisplay.out.writeBytes( "trick.var_add(\"trick_sys.sched.mode\")\n" + + "trick.var_send() \n" + + "trick.var_clear() \n"); + waterClockDisplay.out.flush(); + + // Read the response and extract the simulation mode. + try { + String line; + String field[]; + line = waterClockDisplay.in.readLine(); + field = line.split("\t"); + simMode = Integer.parseInt( field[1]); + } catch (IOException | NullPointerException e ) { + go = false; + } + + // Configure the Variable Server to cyclically send us the following varibales. + // Tell the variable server: + // 1) We want the values of the following variables: + waterClockDisplay.out.writeBytes( "trick.var_pause() \n" + + "trick.var_add(\"dyn.waterclock.intake_clock_spout_flowrate\")\n" + + "trick.var_add(\"dyn.waterclock.intake_overflow_flowrate\")\n" + + "trick.var_add(\"dyn.waterclock.intake_bucket_depth\")\n" + + "trick.var_add(\"dyn.waterclock.timer_bucket_depth\")\n" + + "trick.var_add(\"dyn.waterclock.intake_water_level\")\n" + + "trick.var_add(\"dyn.waterclock.timer_water_level\")\n" + + "trick.var_add(\"dyn.waterclock.intake_bucket_vol\")\n" + + "trick.var_add(\"dyn.waterclock.timer_bucket_vol\")\n" + + "trick.var_add(\"dyn.waterclock.intake_clock_spout_height\")\n" + + "trick.var_add(\"dyn.waterclock.intake_overflow_height\")\n" + + "trick.var_add(\"dyn.waterclock.input_flow\")\n" + + "trick.var_add(\"dyn.waterclock.current_tick\")\n" + + "trick.var_add(\"trick_sys.sched.mode\")\n" + + // 2) We want the responses in ASCII: + "trick.var_ascii() \n" + + // 3) We want values to be updated at the specified rate: + String.format("trick.var_cycle(%.3f)\n", dt) + + // 4) Start sending values as specified. + "trick.var_unpause() \n" ); + waterClockDisplay.out.flush(); + } // if go + + while (go) { + + // Recieve and parse periodic data response from the variable server. + try { + String line; + String field[]; + line = waterClockDisplay.in.readLine(); + field = line.split("\t"); + spout_rate = Double.parseDouble( field[1]); + overflow_rate = Double.parseDouble( field[2]); + intake_depth = Double.parseDouble( field[3]); + timer_depth = Double.parseDouble( field[4]); + intake_lvl = Double.parseDouble( field[5]); + timer_lvl = Double.parseDouble( field[6]); + intake_vol = Double.parseDouble( field[7]); + timer_vol = Double.parseDouble( field[8]); + spout_height = Double.parseDouble( field[9]); + overflow_height = Double.parseDouble( field[10]); + input_flow = Double.parseDouble( field[11]); + tick = Integer.parseInt( field[12]); + simMode = Integer.parseInt( field[13]); + } catch (IOException | NullPointerException e ) { + go = false; + } + + // Update the display data. + rangeView.setSpoutRate(spout_rate); + rangeView.setOverflowRate(overflow_rate); + rangeView.setIntakeDepth(intake_depth); + rangeView.setTimerDepth(timer_depth); + rangeView.setIntakeWarerLevel(intake_lvl); + rangeView.setTimerWarerLevel(timer_lvl); + rangeView.setIntakeWaterVol(intake_vol); + rangeView.setTimerWaterLevel(timer_vol); + rangeView.setSpoutHeight(spout_height); + rangeView.setOverflowHeight(overflow_height); + rangeView.setInputFlow(input_flow); + rangeView.setCurrentTick(tick); + + if(rangeView.new_input_flow) { + flow_select = rangeView.getInputFlowSelect(); + waterClockDisplay.out.writeBytes( String.format("dyn.waterclock.input_flow = %f ;\n", flow_select )); + rangeView.new_input_flow = false; + } + + waterClockDisplay.out.flush(); + + // Update the scene. + waterClockDisplay.drawRangeView(); + + } // while + } // main +} // class diff --git a/trick_sims/SIM_waterclock/models/waterclock/include/waterclock.h b/trick_sims/SIM_waterclock/models/waterclock/include/waterclock.h new file mode 100644 index 00000000..15d1377b --- /dev/null +++ b/trick_sims/SIM_waterclock/models/waterclock/include/waterclock.h @@ -0,0 +1,76 @@ +/************************************************************************* +PURPOSE: (Represent the state and initial conditions of the water clock) +**************************************************************************/ +#ifndef WATERCLOCK_H +#define WATERCLOCK_H + +#include "trick/regula_falsi.h" + +typedef struct { + + + ////////////////////////// + + /* ASSUMPTIONS + * 1) For the purpose of determining if a pipe is submerged, it will be treated as a single point at the center of the pipe + * 2) When the timing bucket is filled, a siphon will drain it instaneously + * 3) Buckets are cylindrical + */ + + /* DESCRIPTION + * This is a simulation of a simple water clock. The clock consists of two cylinders (buckets). + * The first bucket is known as the intake bucket. It receives water from an unspecified external source (like a stream). + * The intake bucket has two cylindrical spouts extending from its sides, one known as an overflow spout and the other the clock spout. + * The overflow spout is simply for maintaing an equilibrium of the water level. It keeps the intake bucket from exceeding a certain water level, thus ensuring it's water level is consistent. + * The clock spout pours water into the second bucket (timer bucket). In an ideal clock, the water level of the intake bucket will remain consistent, meaning the clock spout will output at a constant rate and operate as an accurate clock. + * The timer bucket receives water from the clock spout. The timer bucket has no spouts, it fills until it is full, upon which a siphon instantaneously drains it. + * There exists a float in the timer bucket, connected to an arrow which will point to vertical tick marks. As the timer bucket fills, the arrow will move vertically along these ticks, indicating how much time has passed. + * Depending on the configuration of the clock (size of the buckets, size of the spouts, external water source rate, and the number/spacing of ticks) you can create a clock that tracks a variety of times. + */ + + double time; /* s Model time */ + + double input_flow; /* cm^3/s Flow rate of the external water source to the water clock */ + double intake_clock_spout_flowrate; /* cm^3/s Intake bucket output spout pipe flow rate */ + double intake_overflow_flowrate; /* cm^3/s Intake bucket overflow pipe flow rate */ + double intake_bucket_net_flow; /* cm^3/s Intake bucket net flow rate */ + + double intake_bucket_depth; /* cm Intake bucket depth */ + double intake_bucket_diam; /* cm Intake bucket diameter */ + + double intake_overflow_height; /* cm Intake bucket overflow pipe height of center */ + double intake_overflow_diameter; /* cm Intake bucket overflow pipe diameter */ + + double intake_clock_spout_height; /* cm Intake bucket output spout pipe height of center */ + double intake_clock_spout_diameter; /* cm Intake bucket output spout pipe diameter */ + + double timer_bucket_depth; /* cm Timer bucket depth */ + double timer_bucket_diam; /* cm Timer bucket diameter */ + + double intake_bucket_vol; /* cm^3 Intake bucket water volume */ + double intake_water_level; /* cm Intake bucket water level */ + double timer_bucket_vol; /* cm^3 Timer bucket water volume */ + double timer_water_level; /* cm Timer bucket water level */ + + double tick_gap; /* cm Distance between tick marks */ + int total_ticks; /* -- Total number of ticks on the timer */ + int current_tick; /* -- The current timer tick to have crossed the threshold*/ + + double gravity; /* m/s^2 Gravity constant */ + + REGULA_FALSI rf1 ; + REGULA_FALSI rf2 ; + +} WATERCLOCK ; + +#ifdef __cplusplus +extern "C" { +#endif + int waterclock_default_data(WATERCLOCK*) ; + int waterclock_init(WATERCLOCK*) ; + int waterclock_shutdown(WATERCLOCK*) ; +#ifdef __cplusplus +} +#endif + +#endif diff --git a/trick_sims/SIM_waterclock/models/waterclock/include/waterclock_numeric.h b/trick_sims/SIM_waterclock/models/waterclock/include/waterclock_numeric.h new file mode 100644 index 00000000..eff39022 --- /dev/null +++ b/trick_sims/SIM_waterclock/models/waterclock/include/waterclock_numeric.h @@ -0,0 +1,21 @@ +/************************************************************************* +PURPOSE: ( Water Clock Numeric Model ) +**************************************************************************/ + +#ifndef WATRERCLOCK_NUMERIC_H +#define WATERCLOCK_NUMERIC_H + +#include "waterclock.h" + +#ifdef __cplusplus +extern "C" { +#endif +int waterclock_integ(WATERCLOCK*); +int waterclock_deriv(WATERCLOCK*); +double waterclock_tick_change(WATERCLOCK*); +double waterclock_overflow_timer(WATERCLOCK*); +void waterclock_update_water_level(WATERCLOCK* WC); +#ifdef __cplusplus +} +#endif +#endif diff --git a/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_init.c b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_init.c new file mode 100644 index 00000000..26437d17 --- /dev/null +++ b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_init.c @@ -0,0 +1,59 @@ +/******************************* TRICK HEADER **************************** +PURPOSE: (Set the initial data values) +*************************************************************************/ + +/* Model Include files */ +#include +#include "../include/waterclock.h" + +/* default data job */ +int waterclock_default_data( WATERCLOCK* WC ) { + + WC->time = 0.0 ; + + WC->input_flow = 1000; + WC->intake_clock_spout_flowrate = 0.0; + WC->intake_overflow_flowrate = 0.0; + + WC->intake_bucket_depth = 100.0; + WC->intake_bucket_diam = 500.0; + + WC->intake_overflow_height = 080.0 * WC->intake_bucket_depth; + WC->intake_overflow_diameter = 100.0; + + WC->intake_clock_spout_height = 010.0; + WC->intake_clock_spout_diameter = 005.0; + + WC->timer_bucket_depth = 120.0; + WC->timer_bucket_diam = 050.0; + + WC->intake_bucket_vol = 0.0; + WC->timer_bucket_vol = 0.0; + + WC->total_ticks = 5; + WC->current_tick = 0; + + WC->gravity = 9.81; + + return 0 ; +} + +/* initialization job */ +int waterclock_init( WATERCLOCK* WC) { + + if (WC->intake_overflow_height > WC->intake_bucket_depth) { + WC->intake_overflow_height = WC->intake_bucket_depth; + } + if (WC->intake_clock_spout_height > WC->intake_bucket_depth) { + WC->intake_clock_spout_height = WC->intake_bucket_depth; + } + if (WC->intake_overflow_height < 0) { + WC->intake_overflow_height = 0; + } + if (WC->intake_clock_spout_height < 0) { + WC->intake_clock_spout_height = 0; + } + WC->tick_gap = WC->timer_bucket_depth / WC->total_ticks ; + + return 0 ; +} diff --git a/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_numeric.c b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_numeric.c new file mode 100644 index 00000000..f7a2ec7b --- /dev/null +++ b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_numeric.c @@ -0,0 +1,124 @@ +/********************************************************************* + PURPOSE: ( Trick numeric ) +*********************************************************************/ +#include +#include +#include +#include "trick/integrator_c_intf.h" +#include "../include/waterclock_numeric.h" +#include "trick/exec_proto.h" +#include "trick/message_proto.h" +#include + +int waterclock_deriv(WATERCLOCK* WC) { + + //Make sure water level is up to date + waterclock_update_water_level(WC); + + //Clock spout area + double spout_radius = WC->intake_clock_spout_diameter / 2; + double spout_area = M_PI * spout_radius * spout_radius; + + //Overflow spout area + double overflow_radius = WC->intake_overflow_diameter / 2; + double overflow_area = M_PI * overflow_radius * overflow_radius; + + //Calculate flow rate using Torricelli's equation ( V = sqrt(2gh) ) to find water velocity. Multiple velocity by spout area to find flow rate. + //Calculate input bucket spout flow rate. + if(WC->intake_water_level > WC->intake_clock_spout_height) + WC->intake_clock_spout_flowrate = spout_area * sqrt(2 * WC->gravity * (WC->intake_water_level - WC->intake_clock_spout_height) ); + else + WC->intake_clock_spout_flowrate = 0.0; + + //Calculate input bucket over flow rate + if(WC->intake_water_level > WC->intake_overflow_height) + WC->intake_overflow_flowrate = overflow_area * sqrt(2 * WC->gravity * (WC->intake_water_level - WC->intake_overflow_height) ); + else + WC->intake_overflow_flowrate = 0.0; + + //Need to know net flow of the source, overflow spout, and clock spout for integration + WC->intake_bucket_net_flow = WC->input_flow - WC->intake_clock_spout_flowrate - WC->intake_overflow_flowrate; + + return(0); +} + +int waterclock_integ(WATERCLOCK* WC) { + int ipass; + + load_state( + &WC->intake_bucket_vol, + &WC->timer_bucket_vol, + NULL); + + load_deriv( + &WC->intake_bucket_net_flow, + &WC->intake_clock_spout_flowrate, + NULL); + + ipass = integrate(); + + unload_state( + &WC->intake_bucket_vol, + &WC->timer_bucket_vol, + NULL ); + + waterclock_update_water_level(WC); + + return(ipass); +} + +//Since we are inetgrating over water volume, we must convert volume to water level in order to track how quickly the timer bucket is filling up. +void waterclock_update_water_level(WATERCLOCK* WC) +{ + //Calculate Input bucket water level + double intake_bucket_radius = WC->intake_bucket_diam / 2; + double intake_bucket_base = M_PI * intake_bucket_radius * intake_bucket_radius; + WC->intake_water_level = WC->intake_bucket_vol / intake_bucket_base; + + //Calculate Timer bucket water level + double timer_bucket_radius = WC->timer_bucket_diam / 2; + double timer_bucket_base = M_PI * timer_bucket_radius * timer_bucket_radius; + WC->timer_water_level = WC->timer_bucket_vol / timer_bucket_base; + +} + +//When timer bucket fills, drain it instantly (we have a magic siphon) +double waterclock_overflow_timer( WATERCLOCK* WC ) { + double tgo; + double now; + + WC->rf1.error = WC->timer_bucket_depth - WC->timer_water_level; + now = get_integ_time(); + tgo = regula_falsi( now, &(WC->rf1) ); + if (tgo == 0.0) { + now = get_integ_time() ; + reset_regula_falsi( now, &(WC->rf1) ) ; + WC->timer_water_level = 0; //Instantly drains + WC->timer_bucket_vol = 0; //Instantly drains + WC->current_tick = 0; + message_publish(MSG_NORMAL, "WATER CLOCK RESET\n" ) ; + } + return (tgo) ; +} + +//Detect when the arrow has passed a new tick, print some diagnostics. +double waterclock_tick_change( WATERCLOCK* WC ) { + double tgo; + double now; + + WC->rf2.error = WC->tick_gap - (WC->timer_water_level - ( WC->current_tick * WC->tick_gap )); + now = get_integ_time(); + tgo = regula_falsi( now, &(WC->rf2) ); + if (tgo == 0.0) { + now = get_integ_time() ; + reset_regula_falsi( now, &(WC->rf2) ) ; + if( (WC->current_tick < WC->total_ticks) && (WC->current_tick >= 0) ) + { + WC->current_tick += 1; + message_publish(MSG_NORMAL, "Tick %d, Sim Time %f, Water Level %f\n", WC->current_tick, exec_get_sim_time(), WC->timer_water_level) ; + } + else + fprintf(stderr, "ERROR, SOMETHING WENT VERY WRONG!\n" ) ; + } + return (tgo) ; +} diff --git a/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_shutdown.c b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_shutdown.c new file mode 100644 index 00000000..17486af8 --- /dev/null +++ b/trick_sims/SIM_waterclock/models/waterclock/src/waterclock_shutdown.c @@ -0,0 +1,13 @@ +/************************************************************************ +PURPOSE: (Shutdown the sim) +*************************************************************************/ +#include +#include "../include/waterclock.h" +#include "trick/exec_proto.h" + +int waterclock_shutdown( WATERCLOCK* WC) { + printf( "========================================\n"); + printf( " Water Clock Shutdown \n"); + printf( "========================================\n"); + return 0 ; +} diff --git a/trick_sims/SIM_waterclock/waterclock.tv b/trick_sims/SIM_waterclock/waterclock.tv new file mode 100644 index 00000000..44dff879 --- /dev/null +++ b/trick_sims/SIM_waterclock/waterclock.tv @@ -0,0 +1,96 @@ + + + 0.5 + + dyn.waterclock.intake_overflow_flowrate + + 0.0 + Decimal + + Valid + m^3/s + + + dyn.waterclock.intake_clock_spout_flowrate + + 0.0 + Decimal + + Valid + m^3/s + + + dyn.waterclock.intake_bucket_depth + + 5.0 + Decimal + + Valid + m + + + dyn.waterclock.intake_overflow_height + + 4.0 + Decimal + + Valid + m + + + dyn.waterclock.intake_clock_spout_height + + 0.25 + Decimal + + Valid + m + + + dyn.waterclock.intake_water_level + + 0.0 + Decimal + + Valid + m + + + dyn.waterclock.total_ticks + + 60 + false + Decimal + + Valid + -- + + + dyn.waterclock.current_tick + + 0 + false + Decimal + + Valid + -- + + + dyn.waterclock.timer_bucket_depth + + 5.133689839572193 + Decimal + + Valid + m + + + dyn.waterclock.timer_water_level + + 0.0 + Decimal + + Valid + m + + diff --git a/trick_sims/SIM_waterclock/waterclock_debug.tv b/trick_sims/SIM_waterclock/waterclock_debug.tv new file mode 100644 index 00000000..228c6d81 --- /dev/null +++ b/trick_sims/SIM_waterclock/waterclock_debug.tv @@ -0,0 +1,141 @@ + + + 0.5 + + dyn.waterclock.intake_overflow_flowrate + + 0.0 + Decimal + + Valid + m^3/s + + + dyn.waterclock.intake_clock_spout_flowrate + + 0.0 + Decimal + + Valid + m^3/s + + + dyn.waterclock.intake_bucket_depth + + 1.0 + Decimal + + Valid + m + + + dyn.waterclock.intake_overflow_height + + 0.8 + Decimal + + Valid + m + + + dyn.waterclock.intake_clock_spout_height + + 0.1 + Decimal + + Valid + m + + + dyn.waterclock.intake_water_level + + 0.0 + Decimal + + Valid + m + + + dyn.waterclock.intake_bucket_vol + + 0.0 + Decimal + + Valid + m^3 + + + dyn.waterclock.timer_bucket_depth + + 2.3077 + Decimal + + Valid + m + + + dyn.waterclock.timer_water_level + + 0.0 + Decimal + + Valid + m + + + dyn.waterclock.timer_bucket_vol + + 0.0 + Decimal + + Valid + m^3 + + + dyn.waterclock.rf1.error + + 0.0 + Decimal + + Valid + -- + + + dyn.waterclock.rf2.error + + 0.0 + Decimal + + Valid + -- + + + dyn.waterclock.tick_gap + + 0.03846166666666666 + Decimal + + Valid + m + + + dyn.waterclock.total_ticks + + 60 + false + Decimal + + Valid + -- + + + dyn.waterclock.current_tick + + 0 + false + Decimal + + Valid + -- + +