-
Notifications
You must be signed in to change notification settings - Fork 815
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: add month calendar #3667
base: main
Are you sure you want to change the base?
feat: add month calendar #3667
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Early review with some comments and general out loud thinking. Good progress!
from textual.widgets import MonthCalendar | ||
|
||
|
||
class MonthCalendarApp(App): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding a CSS
variable containing the following CSS to this app yields better results. I reckon we should add the equivalent CSS to the widget itself to override the default DataTable behaviour (unless you can think of a reason not to).
MonthCalendar { height: auto; width: auto; }
MonthCalendar > DataTable { height: auto; width: auto; }
We don't do this in the DataTable
itself because it's intended to scroll, but this widget has a very constrained height compared to the DataTable
which can be arbitrarily large.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry I should have mentioned, I haven't started looking at the DEFAULT_CSS
yet as I wanted to focus on the core functionality first. Do you think the table/columns widths should just be static, or allow the developer to specify the column width (or perhaps even grow dynamically according to the widget width)?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I haven't started looking at the DEFAULT_CSS yet as I wanted to focus on the core functionality first.
Seems sensible.
Do you think the table/columns widths should just be static, or allow the developer to specify the column width (or perhaps even grow dynamically according to the widget width)?
I think static for now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've now added some basic DEFAULT_CSS
in cfcac14. I've flagged min-width
as TODO as this might depend on locale?
class MonthCalendar(Widget): | ||
year: Reactive[int | None] = Reactive[int | None](None) | ||
month: Reactive[int | None] = Reactive[int | None](None) | ||
first_weekday: Reactive[int] = Reactive(0) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm wondering about using Literal["monday", "tuesday", ...]
here to remove any second-guessing about how days are indexed. Will leave it as an open thought here if you or anyone else wants to chip in.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps this could be an IntEnum
similar to the calendar module in 3.12?
self.first_weekday = first_weekday | ||
|
||
def compose(self) -> ComposeResult: | ||
yield DataTable() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Open thoughts: I wonder if we should display the month name above the DataTable. I wonder if it should be something that can be toggled.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm assuming this would simply display "November 2023" for example, and then if developers wanted something different they would be expected to create a custom widget and set MonthCalendar.month_header
to False?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure - I guess the month name would be defined by locale too? @willmcgugan any thoughts?
I'm struggling with handling the reactives for this widget, due to a class of problem explained this issue: I'm afraid I'm blocked on this, unless there's a workaround that would include a cursor reactive. My current |
Hi @TomJGooding, could you try adding a boolean Let me know if I've misunderstood the issue! |
Thanks Darren for your help, it looks like this works: def watch_month(self) -> None:
if not self.is_mounted:
return
self._update_calendar_days() Would it be better to add this new |
No problem! I think separate PR would be great for that. |
My latest commit (334e771) will update the calendar cursor to the same highlighted day when the month changes. Here's a quick demo recording, notice this accounts for when the new month doesn't have 31 days: This requires the The problem with updating the cursor when the month changes is I'm not sure how to handle when the highlighted date is outside the month. For example in the demo recording, on the initial month the cursor could be highlighting the first Monday (31st). Where should the cursor highlight when the month changes? If changed to just the previous month, it probably should just highlight the same date, but what if the updated calendar is months or years different? Perhaps if you don't mind (I appreciate how busy you are), this would a good time for a pre-review review? Tagging @darrenburns and @davep but any feedback would be welcome! Thanks again for your help so far. (Also just to note, the failing test is a keyline snapshot mismatch and not related to this PR!) |
Thanks @willmcgugan for your feedback. I've now removed the Hopefully the component classes issue makes sense, but just to clarify I haven't managed to find any workaround. I'm happy to look at adding some mechanism for this, but I'm afraid I would need a little guidance on where to start. |
Just to give an example of the issue: class CalendarGrid(DataTable, inherit_bindings=False):
# TODO: Ideally we want to hide that there's a DataTable underneath the
# `MonthCalendar` widget. Is there any mechanism so component classes could be
# defined in the parent and substitute the component styles in the child?
# For example, allow styling the header using `.month-calendar--header`
# rather than `.datatable--header`?
DEFAULT_CSS = """
CalendarGrid {
height: auto;
width: auto;
.datatable--header {
background: $surface;
}
}
""" We need to use the |
It is a bit of a wart. We discussed it in the office and didn't come up with a conclusion. Suggest we go with this for now. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi Tom, sorry for the delay again. Getting caught up.
I know this is marked as draft, just wanted to add a few comments to avoid blocking while waiting for my input.
|
||
from rich.text import Text | ||
|
||
from textual import on |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you change these to relative imports, which is the convention we use elsewhere.
I think we are going to switch to absolute imports eventually, but best to be consistent for now...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've reverted back to absolute imports following #4997!
"month-calendar--outside-month", | ||
} | ||
|
||
date: Reactive[datetime.date] = Reactive(datetime.date.today()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The today()
is evaluated at import time. We probably want it to be evaluated when the widget is instantiated.
Easy fix: reactives take a callable. So we can just drop the ()
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in 172fbbe.
show_cursor: Reactive[bool] = Reactive(True) | ||
show_other_months: Reactive[bool] = Reactive(True) | ||
|
||
class DateHighlighted(Message): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Docstring pls
def control(self) -> MonthCalendar: | ||
return self.month_calendar | ||
|
||
class DateSelected(Message): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Docstring pls
event.stop() | ||
pass | ||
|
||
def previous_month(self) -> None: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Docstrings on all methods pls. With the exception of inherited methods / handlers.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry just to confirm, do all watch
and action
methods need docstings? Also should these methods be private (i.e. include a leading underscore)? I did have a look at other widgets for guidance but there seems to be a mix.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actions, yes. Just a note to say what they do. Watch methods, the docstrings seems rendundant, so we often skip those unless you want to document for future devs why they are needed.
With regards to marking them private, if there is no earthly reason the user would want to override those, then its best to make them private...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, I've added those docstrings but wasn't sure if they should be private so left as-is for now.
This reverts commit 694a24d.
This PR is approaching its one year anniversary, which is perhaps fitting for a calendar widget! I think I under-estimated some of the intricacies of implementing this, and that's before starting work on the Hopefully there is still appetite for this! Since this PR has been open for a while, I wanted to ensure I was keeping track of any outstanding features/issues. I've started a list below, I'd really appreciate some feedback when you have some time.
|
I definitely want to see this in Textual. I don't consider the outstanding issues to be blockers personally... I think it'd be far more useful to the community to get this merged and then if those turn out to be major issues then we can revisit them separately? |
Description
Add a
MonthCalendar
widget, intended to form part of aDatePicker
widget.Note
Currently very much a work in progress wanting feedback from Textual maintainers and users!
Related Issue
Checklist: