Skip to content
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: hour chime #66

Merged
merged 6 commits into from
Nov 12, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 1 addition & 4 deletions MANIFEST.in
Original file line number Diff line number Diff line change
@@ -1,8 +1,5 @@
recursive-include dialog *
recursive-include vocab *
recursive-include locale *
recursive-include res *
recursive-include qt5 *
recursive-include skill *
recursive-include gui *
include *.json
include *.txt
28 changes: 23 additions & 5 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,7 @@
Get the time, date, day of the week

## About
Get the local time or time for major cities around the world. Times
are given in 12-hour (2:30 pm) or 24-hour format (14:30) based on the
Time Format setting in your `mycroft.conf`

Get the local time or time for major cities around the world. Times are given in 12-hour (2:30 pm) or 24-hour format (14:30) based on the Time Format setting in your `mycroft.conf`

## Examples
* "What time is it?"
Expand All @@ -16,8 +13,29 @@ Time Format setting in your `mycroft.conf`
* "How many days until July 4th"
* "What day is Memorial Day 2020?"

## Configuration

You can adjust certain aspects of this skill's behavior by configuring the `settings.json` file.

2 sound files are included with the skill, `"casio-watch.wav"` and `"clock-chime.mp3"`, to audibly signal when the hour changes

Below is an example configuration file with explanations for each option.

```json
{
"play_hour_chime": true,
"hour_sound": "clock-chime.mp3"
}
```

- **`play_hour_chime`**: (boolean) Enables or disables the hourly chime notification. If `true`, the skill will play an audio chime at the start of every hour. Default is `false`.
- **`hour_sound`**: (string) Specifies the file path to the audio file used for the hourly chime. By default, it points to `casio-watch.wav` in the `res` folder. You can customize this with the path to any audio file you prefer.

## Credits
Mycroft AI (@MycroftAI)

- [casio-watch.wav by @Pablobd](https://freesound.org/people/Pablobd/sounds/492481/) under the [CC0 1.0 Universal License](https://creativecommons.org/publicdomain/zero/1.0/)
- [clock-chime.mp3 by @ecfike](https://pixabay.com/sound-effects/clock-chime-88027/) under the [Pixabay Content License](https://pixabay.com/service/license-summary/)
- Original skill by Mycroft AI (@MycroftAI)

## Category
**Daily**
Expand Down
70 changes: 63 additions & 7 deletions __init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,8 @@
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import datetime
import os
import re
from typing import Optional

Expand Down Expand Up @@ -67,8 +67,63 @@ def runtime_requirements(self):
no_gui_fallback=True)

def initialize(self):
"""Initialize the skill by pre-loading lingua-franca."""
"""Initialize the skill by pre-loading language settings and scheduling
the hourly chime.

This method is called automatically when the skill starts, preloading
language-related formatting for date and time and setting up the initial
scheduling for the hourly chime event.
"""
date_time_format.cache(self.lang)
self._schedule_hour_chime()

def _handle_play_hour_chime(self, message: Message):
"""Play the hourly chime audio and re-schedule the next chime event.

Args:
message (Message): message object

This method checks if the hourly chime setting is enabled. If it is, it
plays the specified chime audio. Then, it re-schedules the next hourly
chime event.
"""
if self.play_hour_chime:
self.play_audio(self.hour_chime, instant=True)
self._schedule_hour_chime()

def _schedule_hour_chime(self):
"""Schedule the next hourly chime event for the start of the next hour.

This method calculates the time for the upcoming hour, setting it as
the scheduled time for the next chime event.
"""
n = now_local() + datetime.timedelta(hours=1)
self.schedule_event(self._handle_play_hour_chime,
when=datetime.datetime(year=n.year, month=n.month, day=n.day,
hour=n.hour, minute=0, second=0))

@property
def play_hour_chime(self) -> bool:
"""Check if the hourly chime setting is enabled.

Returns:
bool: True if the chime should be played on the hour, False otherwise.
"""
return self.settings.get("play_hour_chime", False)

@property
def hour_chime(self) -> str:
"""Get the file path for the hourly chime sound.

Returns:
str: The file path to the chime audio file. If not set in settings,
defaults to 'casio-watch.wav' in the 'res' folder.
"""
snd = self.settings.get("hour_sound", "casio-watch.wav")
if not os.path.isfile(snd):
snd2 = f"{os.path.dirname(__file__)}/res/{snd}"
snd = snd2 if os.path.isfile(snd2) else snd
return snd
JarbasAl marked this conversation as resolved.
Show resolved Hide resolved
JarbasAl marked this conversation as resolved.
Show resolved Hide resolved

@property
def use_24hour(self):
Expand All @@ -95,7 +150,8 @@ def _extract_location(self, utt: str) -> str:
pass
return None

def _get_timezone_from_builtins(self, location_string: str) -> datetime.tzinfo:
@staticmethod
def _get_timezone_from_builtins(location_string: str) -> Optional[datetime.tzinfo]:
"""Get timezone from built-in resources."""
if "/" not in location_string:
try:
Expand Down Expand Up @@ -130,7 +186,7 @@ def _get_timezone_from_table(self, location_string: str) -> Optional[datetime.tz
return pytz.timezone(timezones[timezone].strip())
return None

def _get_timezone_from_fuzzymatch(self, location_string: str) -> datetime.tzinfo:
def _get_timezone_from_fuzzymatch(self, location_string: str) -> Optional[datetime.tzinfo]:
"""Fuzzymatch a location against the pytz timezones.

The pytz timezones consists of
Expand Down Expand Up @@ -187,7 +243,7 @@ def get_timezone_in_location(self, location_string: str) -> datetime.tzinfo:
######################################################################
# utils
def get_datetime(self, location: str = None,
anchor_date: datetime.datetime = None) -> datetime.datetime:
anchor_date: datetime.datetime = None) -> Optional[datetime.datetime]:
"""return anchor_date/now_local at location/session_tz"""
if location:
JarbasAl marked this conversation as resolved.
Show resolved Hide resolved
tz = self.get_timezone_in_location(location)
Expand Down Expand Up @@ -430,7 +486,7 @@ def show_date_mark1(self, dt: datetime.datetime):
show = self.get_display_date(anchor_date=dt)
LOG.debug(f"sending date to mk1 {show}")
self.bus.emit(Message("ovos.mk1.display_date",
{"text": show}))
{"text": show}))

def show_date_gui(self, dt: datetime.datetime, location: str):
self.gui.clear()
Expand All @@ -454,7 +510,7 @@ def show_time(self, display_time: str):
def show_time_mark1(self, display_time: str):
LOG.debug(f"Emitting ovos.mk1.display_time with time: {display_time}")
self.bus.emit(Message("ovos.mk1.display_time",
{"text": display_time}))
{"text": display_time}))

def show_time_gui(self, display_time):
""" Display time on the GUI. """
Expand Down
Binary file added res/casio-watch.wav
Binary file not shown.
Binary file added res/clock-chime.mp3
Binary file not shown.
4 changes: 2 additions & 2 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,9 +30,9 @@ def get_requirements(requirements_filename: str):


def find_resource_files():
resource_base_dirs = ("locale", "gui")
resource_base_dirs = ("locale", "gui", "res")
base_dir = path.dirname(__file__)
package_data = ["*.json"]
package_data = ["*.json", "*.mp3", "*.wav"]
for res in resource_base_dirs:
if path.isdir(path.join(base_dir, res)):
for (directory, _, files) in walk(path.join(base_dir, res)):
Expand Down
Loading