"""Functional tests for the My Sky dashboard feature. My Sky is a dashboard applet linking to /dashboard/sky/ — a full-page natus (natal chart) interface where the user can save their personal sky to their account (stored on the User model, independent of any game room). """ from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.by import By from apps.applets.models import Applet from apps.lyric.models import User from .base import FunctionalTest # Minimal chart fixture — matches the NatusWheel data shape. _CHART_FIXTURE = { "planets": { "Sun": {"sign": "Pisces", "degree": 340.0, "retrograde": False}, "Moon": {"sign": "Gemini", "degree": 72.0, "retrograde": False}, "Mercury": {"sign": "Aquarius", "degree": 310.0, "retrograde": False}, "Venus": {"sign": "Aries", "degree": 10.0, "retrograde": False}, "Mars": {"sign": "Capricorn", "degree": 280.0, "retrograde": False}, "Jupiter": {"sign": "Cancer", "degree": 100.0, "retrograde": False}, "Saturn": {"sign": "Capricorn", "degree": 290.0, "retrograde": True}, "Uranus": {"sign": "Capricorn", "degree": 285.0, "retrograde": False}, "Neptune": {"sign": "Capricorn", "degree": 283.0, "retrograde": False}, "Pluto": {"sign": "Scorpio", "degree": 218.0, "retrograde": False}, }, "houses": { "cusps": [10, 40, 70, 100, 130, 160, 190, 220, 250, 280, 310, 340], "asc": 10.0, "mc": 100.0, }, "elements": {"Fire": 1, "Water": 2, "Stone": 4, "Air": 1, "Time": 0, "Space": 1}, "aspects": [], "distinctions": { "1": 1, "2": 0, "3": 0, "4": 1, "5": 0, "6": 0, "7": 0, "8": 0, "9": 0, "10": 4, "11": 0, "12": 0, }, "house_system": "O", "timezone": "America/New_York", } class MySkyAppletTest(FunctionalTest): """My Sky applet appears on the dashboard and links to the sky page.""" def setUp(self): super().setUp() Applet.objects.get_or_create( slug="my-sky", defaults={"name": "My Sky", "grid_cols": 6, "grid_rows": 6, "context": "dashboard"}, ) self.gamer = User.objects.create(email="stargazer@test.io") # ── T1 ─────────────────────────────────────────────────────────────────── def test_my_sky_applet_links_to_sky_page_with_form(self): """Applet is visible on dashboard; link leads to /dashboard/sky/ with all natus form fields present.""" self.create_pre_authenticated_session("stargazer@test.io") self.browser.get(self.live_server_url) # 1. Applet is on the dashboard applet = self.wait_for( lambda: self.browser.find_element(By.ID, "id_applet_my_sky") ) # 2. Heading contains a link whose text is "My Sky" link = applet.find_element(By.CSS_SELECTOR, "h2 a") self.assertIn("MY SKY", link.text.upper()) # 3. Clicking the link navigates to /dashboard/sky/ link.click() self.wait_for( lambda: self.assertRegex(self.browser.current_url, r"/dashboard/sky/$") ) # 4. All natus form fields are present self.browser.find_element(By.ID, "id_nf_date") self.browser.find_element(By.ID, "id_nf_time") self.browser.find_element(By.ID, "id_nf_place") self.browser.find_element(By.ID, "id_nf_lat") self.browser.find_element(By.ID, "id_nf_lon") self.browser.find_element(By.ID, "id_nf_tz") self.browser.find_element(By.ID, "id_natus_confirm") class MySkyLocalStorageTest(FunctionalTest): """My Sky form fields persist to localStorage across visits.""" def setUp(self): super().setUp() Applet.objects.get_or_create( slug="my-sky", defaults={"name": "My Sky", "grid_cols": 6, "grid_rows": 6, "context": "dashboard"}, ) self.gamer = User.objects.create(email="stargazer@test.io") self.sky_url = self.live_server_url + "/dashboard/sky/" def _fill_form(self): """Set date, lat, lon directly — bypasses Nominatim network call.""" self.browser.execute_script( "document.getElementById('id_nf_date').value = '1990-06-15';" "document.getElementById('id_nf_lat').value = '51.5074';" "document.getElementById('id_nf_lon').value = '-0.1278';" "document.getElementById('id_nf_place').value = 'London, UK';" "document.getElementById('id_nf_tz').value = 'Europe/London';" ) # Fire input events so the localStorage save listener triggers self.browser.execute_script(""" ['id_nf_date','id_nf_lat','id_nf_lon','id_nf_place','id_nf_tz'].forEach(id => { document.getElementById(id) .dispatchEvent(new Event('input', {bubbles: true})); }); """) def _field_values(self): return self.browser.execute_script(""" return { date: document.getElementById('id_nf_date').value, lat: document.getElementById('id_nf_lat').value, lon: document.getElementById('id_nf_lon').value, place: document.getElementById('id_nf_place').value, tz: document.getElementById('id_nf_tz').value, }; """) # ── T2 ─────────────────────────────────────────────────────────────────── def test_sky_form_fields_repopulated_after_page_refresh(self): """Form values survive a full page refresh via localStorage.""" self.create_pre_authenticated_session("stargazer@test.io") self.browser.get(self.sky_url) self.wait_for(lambda: self.browser.find_element(By.ID, "id_nf_date")) self._fill_form() self.browser.refresh() self.wait_for(lambda: self.browser.find_element(By.ID, "id_nf_date")) values = self._field_values() self.assertEqual(values["date"], "1990-06-15") self.assertEqual(values["lat"], "51.5074") self.assertEqual(values["lon"], "-0.1278") self.assertEqual(values["place"], "London, UK") self.assertEqual(values["tz"], "Europe/London") class MySkyAppletWheelTest(FunctionalTest): """Saved natal chart renders as an interactive wheel inside the My Sky applet.""" def setUp(self): super().setUp() Applet.objects.get_or_create( slug="my-sky", defaults={"name": "My Sky", "grid_cols": 6, "grid_rows": 6, "context": "dashboard"}, ) self.gamer = User.objects.create(email="stargazer@test.io") self.gamer.sky_chart_data = _CHART_FIXTURE self.gamer.sky_birth_place = "Lindenwold, NJ, US" self.gamer.save() # ── T3 ─────────────────────────────────────────────────────────────────── def test_saved_sky_wheel_renders_with_tooltips_in_applet(self): """When the user has saved sky data, the natal wheel appears in the My Sky applet with working element-ring and planet tooltips.""" self.create_pre_authenticated_session("stargazer@test.io") self.browser.get(self.live_server_url) # 1. Wheel SVG is drawn inside the applet self.wait_for(lambda: self.assertTrue( self.browser.find_element( By.CSS_SELECTOR, "#id_applet_my_sky .nw-root" ) )) # 2. Hovering an element-ring slice shows the tooltip slice_el = self.browser.find_element( By.CSS_SELECTOR, "#id_applet_my_sky .nw-element-group" ) ActionChains(self.browser).move_to_element(slice_el).perform() self.wait_for(lambda: self.assertEqual( self.browser.find_element(By.ID, "id_natus_tooltip") .value_of_css_property("display"), "block", )) # 3. Hovering a planet also shows the tooltip planet_el = self.browser.find_element( By.CSS_SELECTOR, "#id_applet_my_sky .nw-planet-group" ) ActionChains(self.browser).move_to_element(planet_el).perform() self.wait_for(lambda: self.assertEqual( self.browser.find_element(By.ID, "id_natus_tooltip") .value_of_css_property("display"), "block", ))