Show events before and after gaps
This commit is contained in:
		
							parent
							
								
									45e4a04fb9
								
							
						
					
					
						commit
						7c54fdfe09
					
				| 
						 | 
					@ -2,6 +2,7 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import asyncio
 | 
					import asyncio
 | 
				
			||||||
import collections
 | 
					import collections
 | 
				
			||||||
 | 
					import itertools
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
import typing
 | 
					import typing
 | 
				
			||||||
from datetime import date, datetime, timedelta
 | 
					from datetime import date, datetime, timedelta
 | 
				
			||||||
| 
						 | 
					@ -37,6 +38,8 @@ from . import (
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
from .types import Event, Holiday
 | 
					from .types import Event, Holiday
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					StrDict = dict[str, typing.Any]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
here = dateutil.tz.tzlocal()
 | 
					here = dateutil.tz.tzlocal()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# deadline to file tax return
 | 
					# deadline to file tax return
 | 
				
			||||||
| 
						 | 
					@ -272,13 +275,23 @@ def find_markets_during_stay(
 | 
				
			||||||
    return overlapping_markets
 | 
					    return overlapping_markets
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def find_gaps(events: list[Event], min_gap_days: int = 3) -> list[tuple[date, date]]:
 | 
					def find_gaps(events: list[Event], min_gap_days: int = 3) -> list[StrDict]:
 | 
				
			||||||
    """Gaps of at least `min_gap_days` between events in a list of events."""
 | 
					    """Gaps of at least `min_gap_days` between events in a list of events."""
 | 
				
			||||||
    # Sort events by start date
 | 
					    # Sort events by start date
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    gaps: list[tuple[date, date]] = []
 | 
					    gaps: list[tuple[date, date]] = []
 | 
				
			||||||
    previous_event_end = None
 | 
					    previous_event_end = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    by_start_date = {
 | 
				
			||||||
 | 
					        d: list(on_day)
 | 
				
			||||||
 | 
					        for d, on_day in itertools.groupby(events, key=lambda e: e.as_date)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    by_end_date = {
 | 
				
			||||||
 | 
					        d: list(on_day)
 | 
				
			||||||
 | 
					        for d, on_day in itertools.groupby(events, key=lambda e: e.end_as_date)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for event in events:
 | 
					    for event in events:
 | 
				
			||||||
        # Use start date for current event
 | 
					        # Use start date for current event
 | 
				
			||||||
        start_date = event.as_date
 | 
					        start_date = event.as_date
 | 
				
			||||||
| 
						 | 
					@ -294,11 +307,19 @@ def find_gaps(events: list[Event], min_gap_days: int = 3) -> list[tuple[date, da
 | 
				
			||||||
                gaps.append(start_end)
 | 
					                gaps.append(start_end)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # Update previous event end date
 | 
					        # Update previous event end date
 | 
				
			||||||
        end = event.end_as_date if event.end_date else start_date
 | 
					        end = event.end_as_date
 | 
				
			||||||
        if not previous_event_end or end > previous_event_end:
 | 
					        if not previous_event_end or end > previous_event_end:
 | 
				
			||||||
            previous_event_end = end
 | 
					            previous_event_end = end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return gaps
 | 
					    return [
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            "start": gap_start,
 | 
				
			||||||
 | 
					            "end": gap_end,
 | 
				
			||||||
 | 
					            "after": by_start_date[gap_end + timedelta(days=1)],
 | 
				
			||||||
 | 
					            "before": by_end_date[gap_start - timedelta(days=1)],
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        for gap_start, gap_end in gaps
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def busy_event(e: Event) -> bool:
 | 
					def busy_event(e: Event) -> bool:
 | 
				
			||||||
| 
						 | 
					@ -455,7 +476,9 @@ async def get_data(
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    gaps = find_gaps(busy_events)
 | 
					    gaps = find_gaps(busy_events)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    events += [Event(name="gap", date=start, end_date=end) for start, end in gaps]
 | 
					    events += [
 | 
				
			||||||
 | 
					        Event(name="gap", date=gap["start"], end_date=gap["end"]) for gap in gaps
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # Sort events by their datetime; the "today" event is prioritised
 | 
					    # Sort events by their datetime; the "today" event is prioritised
 | 
				
			||||||
    # at the top of the list for today. This is achieved by sorting first by
 | 
					    # at the top of the list for today. This is achieved by sorting first by
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -50,12 +50,15 @@ class Event:
 | 
				
			||||||
    @property
 | 
					    @property
 | 
				
			||||||
    def end_as_date(self) -> datetime.date:
 | 
					    def end_as_date(self) -> datetime.date:
 | 
				
			||||||
        """Date of event."""
 | 
					        """Date of event."""
 | 
				
			||||||
        assert self.end_date
 | 
					 | 
				
			||||||
        return (
 | 
					        return (
 | 
				
			||||||
 | 
					            (
 | 
				
			||||||
                self.end_date.date()
 | 
					                self.end_date.date()
 | 
				
			||||||
                if isinstance(self.end_date, datetime.datetime)
 | 
					                if isinstance(self.end_date, datetime.datetime)
 | 
				
			||||||
                else self.end_date
 | 
					                else self.end_date
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
 | 
					            if self.end_date
 | 
				
			||||||
 | 
					            else self.as_date
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @property
 | 
					    @property
 | 
				
			||||||
    def display_time(self) -> str | None:
 | 
					    def display_time(self) -> str | None:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -17,13 +17,26 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <table class="table table-hover w-auto">
 | 
					  <table class="table table-hover w-auto">
 | 
				
			||||||
  {% for start, end in gaps %}
 | 
					    <thead>
 | 
				
			||||||
      <tr>
 | 
					      <tr>
 | 
				
			||||||
    <td class="text-end">{{ start.strftime("%A, %-d %b %Y") }}</td>
 | 
					        <th>before</th>
 | 
				
			||||||
    <td class="text-end">{{ end.strftime("%A, %-d %b %Y") }}</td>
 | 
					        <th class="text-end">start</th>
 | 
				
			||||||
    <td>{{ (end - start).days }} days</td>
 | 
					        <th class="text-end">end</th>
 | 
				
			||||||
 | 
					        <th class="text-end">days</th>
 | 
				
			||||||
 | 
					        <th>after</th>
 | 
				
			||||||
 | 
					      </tr>
 | 
				
			||||||
 | 
					    </thead>
 | 
				
			||||||
 | 
					    <tbody>
 | 
				
			||||||
 | 
					    {% for gap in gaps %}
 | 
				
			||||||
 | 
					    <tr>
 | 
				
			||||||
 | 
					      <td>{% for event in gap.before %}{% if not loop.first %}, {% endif %}{{ event.title or event.name }}{% endfor %}</td>
 | 
				
			||||||
 | 
					      <td class="text-end">{{ gap.start.strftime("%A, %-d %b %Y") }}</td>
 | 
				
			||||||
 | 
					      <td class="text-end">{{ gap.end.strftime("%A, %-d %b %Y") }}</td>
 | 
				
			||||||
 | 
					      <td class="text-end">{{ (gap.end - gap.start).days }} days</td>
 | 
				
			||||||
 | 
					      <td>{% for event in gap.after %}{% if not loop.first %}, {% endif %}{{ event.title or event.name }}{% endfor %}</td>
 | 
				
			||||||
    </tr>
 | 
					    </tr>
 | 
				
			||||||
    {% endfor %}
 | 
					    {% endfor %}
 | 
				
			||||||
 | 
					    </tbody>
 | 
				
			||||||
  </table>
 | 
					  </table>
 | 
				
			||||||
</div>
 | 
					</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue