This project aims to build a non-conventional alarm clock with smart features such as auto-dismissal based on phone activity, weather and more! It also includes challenges to ensure you completely wake up and an option to set shared alarms! This is the ULTIMATE alarm clock :)
- What's new (GSOC'24)
- GetX Pattern
- Database Schema
- Installation & Setup
- User Interface & Features
- Contribution Guidelines
- Future Plans
- Community
- Flutter
During the GSOC'24 period, the following features were implemented by Contributor Aryan Saraf:
The previous alarm scheduling logic and smart controls would launch the app when working in the background due to reliance on Flutter method channels for database access and functions. To resolve this, we have migrated the Alarm Scheduling and Smart Controls logic from Flutter to Kotlin. Pull-Requests: #563 #574 #580
Switching to SQLite databases will enhance cross-language data accessibility and streamline native feature integration. Earlier, we used Method Channels to extract data from the ISAR database for scheduling alarms upon booting, which triggers MainActivity and launches the app from the background. Transitioning to SQLite will improve flexibility and adaptability for future developments as it can be directly used by Kotlin. Issue: #562, Pull-Request: #563
Effortlessly manage and share custom alarm profiles for different days and occasions, ensuring only the active profile’s alarms are prioritized. Alarms and profiles can now be shared with other users using their emails. In-app notifications for received alarms and profiles with the option to either accept or reject them. Issue: #591, Pull-Request: #584
Integrate Google Calendar to import reminders, events and aggregate alarms. Users can now create alarms that can be triggered on a specific date Issue: #590, Pull-Request: #584
Set a Guardian Angel to send a call or SMS to if you fail to wake up for important events, ensuring you never miss crucial moments. Issue: #592, Pull-Request: #584
Automatically dismiss alarms if the user’s screen is on for more than X minutes, indicating they are busy and don’t need the alarm.” Issue: #572, Pull-Request: #574
Switch to Open-Meteo for weather API for seamless weather integration without the need for an API key, reducing setup friction and streamlining the user experience. Add Notification for user location being accessed in the background. Weather and Location-based smart controls logic shifted to Kotlin. Issue: #579, Pull-Request: #580
Revamped the timer with multiple timer support, an easy-to-use UI, notification display when minimized, and direct Kotlin database access for faster performance without relying on Flutter for data. Issue: #564, Pull-Request: #565
“Added 5 new royalty-free ringtones and fixed related bugs, including erratic ringtone preview behaviour.” Issue: #595, Pull-Request: #596
- Fixed Scheduling logic for weekday scheduling.
- Fixed Several Controller errors throughout different PRs.
- New UI for alarm and profile setting screen.
- Fixed existing Firebase Auth implementation.
The "Ultimate Alarm Clock" project employs the GetX pattern for state management. The GetX pattern is a popular state management solution in the Flutter ecosystem, known for its simplicity, efficiency, and developer-friendly approach. It simplifies the process of managing the state of a Flutter application and helps in building reactive and performant user interfaces.
The "Ultimate Alarm Clock" project uses get_cli
, a powerful Flutter package that simplifies various aspects of project development, such as generating new pages, routes, and more. Below, we'll briefly explain how to use get_cli
for creating new pages and touch upon the purpose of different files in a GetX-based Flutter project.
To get started with get_cli
, you need to install it. Run the following command in your project directory:
flutter pub global activate get_cli
With get_cli
, you can quickly generate the necessary files for a new page. Here's how you can do it:
get create page:/your_page_name
Replace /your_page_name
with the desired name for your new page. This command will create several files and folders for your page, including a controller, view, and binding.
- Controller: The controller is responsible for handling the business logic and state management of a page. It connects the UI (View) with the underlying data and functions.
- View: The view represents the UI of the page. It defines how the page should look and interact with users. It focuses on the presentation of data.
- Binding: The binding connects the controller and view, ensuring that they work together seamlessly. It sets up dependencies, routes, and other configurations required for the page.
To learn more about the GetX, you can read the documentation here.
The "Ultimate Alarm Clock" project utilizes multiple databases for different purposes. These databases include Firebase Firestore, ISAR, and Flutter Secure Storage. Each database serves a distinct role in managing various aspects of the application. Below we provide an overview of the schema for each of these databases.
Firebase Firestore is used for real-time data synchronization and storage of user-related data. It plays a pivotal role in enabling features like shared alarms and seamless collaboration among users. Here's why Firestore is a necessary component:
-
Shared Alarms: Firestore allows users to share alarms with one another in real-time. Users can collaborate on alarm settings, making it convenient for multiple individuals to set alarms for a common purpose.
-
Real-time Updates: Firestore enables instant updates and synchronization of alarm data across devices and users. This means that any changes made to shared alarms are immediately reflected on all connected devices.
The schema for Firebase Firestore consists of collections and documents, and here is a summary of its structure.
- Attributes:
fullName
(String): User's fullnamefirstName
(String): User's first namelastName
(String): User's last nameemail
(String): User's email addressid
(String): A unique identifier for each user
- Attributes:
-
isarId
(Auto-incremented Integer): A unique identifier generated automatically by ISAR. -
firestoreId
(String): An optional identifier associated with Firestore. -
alarmTime
(String): The time at which the alarm is set to go off. -
alarmID
(String): A unique identifier for each alarm. -
isEnabled
(Boolean): Indicates whether the alarm is enabled. -
isLocationEnabled
(Boolean): Indicates whether location-based features are enabled for the alarm. -
isSharedAlarmEnabled
(Boolean): Indicates whether the alarm is shared with other users. -
isWeatherEnabled
(Boolean): Indicates whether weather-related features are enabled for the alarm. -
isMathsEnabled
(Boolean): Indicates whether math-related features are enabled for the alarm. -
isShakeEnabled
(Boolean): Indicates whether the alarm can be turned off by shaking the device. -
isQrEnabled
(Boolean): Indicates whether QR code scanning is enabled for the alarm. -
intervalToAlarm
(Integer): Time interval in minutes for the alarm to go off. -
isActivityEnabled
(Boolean): Indicates whether activity tracking is enabled for the alarm. -
location
(String): The location associated with the alarm. -
minutesSinceMidnight
(Integer): The number of minutes since midnight when the alarm is set. -
days
(List of Booleans): A list representing the days on which the alarm should repeat. -
weatherTypes
(List of Integers): List of weather conditions for the alarm. -
shakeTimes
(Integer): Number of times the device must be shaken to turn off the alarm. -
numMathsQuestions
(Integer): The number of math questions for the alarm. -
mathsDifficulty
(Integer): The difficulty level of math questions. -
qrValue
(String): The QR code value associated with the alarm. -
sharedUserIds
(List of Strings): User IDs with whom the alarm is shared. -
ownerId
(String): The user ID of the alarm owner. -
ownerName
(String): The name of the alarm owner. -
lastEditedUserId
(String): User ID of the last user who edited the alarm. -
mutexLock
(Boolean): A flag indicating whether a mutex lock is applied to the alarm. -
mainAlarmTime
(String): The main time at which the alarm is set. -
label
(String): A label or description associated with the alarm. -
isOneTime
(Boolean): Indicates whether the alarm is a one-time alarm. -
snoozeDuration
(Integer): The snooze duration for the alarm, in minutes. -
offsetDetails
(Map, Ignored): A map containing additional offset details. -
deleteAfterGoesOff
(Boolean): Deletes the alarm once it rings. -
showMotivationalQuote
(Boolean): Toggle motivational quote pop on alarm dismiss. -
activityMonitor
(Integer): Checks whether the activity monitor is on as 0 and 1. -
alarmDate
(String): Stores the date on which the alarm should ring. -
ringOn
(Boolean): Toggle specific ring date for alarms. -
profile
(String): Stores the name of the profile the alarm belongs to. -
isGuardian
(Boolean): Toggle the Guardian Angel feature for the alarm. -
guardianTimer
(Integer): Stores time in seconds for Guardian Angel to activate after inactivity during an alarm trigger. -
guardian
(String): Contact number of the person to contact for Guardian Angel feature. -
isCall
(Boolean): Toggles Call or SMS for Guardian Angel.
-
- Attributes:
-
timerValue
(Integer): The total duration of the timer in seconds. -
startedOn
(String): The timestamp when the timer was started. -
ringtoneName
(String): The name of the ringtone to be played when the timer ends. -
timerName
(String): A unique name for the timer. -
isPaused
(Integer): Indicates whether the timer is paused (0 for not paused, 1 for paused). -
timeElapsed
(Integer): The amount of time that has elapsed since the timer started, in seconds.
-
ISAR is the go-to solution for local storage of alarm-related data. It facilitates the efficient and structured management of alarm settings and preferences, ensuring that alarms function smoothly even in offline scenarios. ISAR optimizes data retrieval, enabling quick access to triggered alarms and user-specific configurations. Its performance efficiency and data integrity make it an essential component, ensuring that alarms trigger accurately and promptly. ISAR complements Firestore's real-time data synchronization, offering a responsive local data store to enhance the overall user experience.
The key reasons for utilizing IsarDb are as follows:
-
Offline Functionality: IsarDb ensures that alarms function seamlessly even in offline scenarios. Users can trust that their alarms will trigger as expected, regardless of their internet connection status.
-
Performance Optimization: IsarDb optimizes data retrieval, allowing for quick access to triggered alarms and user-specific configurations. It enhances the overall performance and responsiveness of the application.
The schema for this data is already described above.
SQFLite is a Flutter plugin that allows you to use SQLite databases in your Flutter applications. Using SQLite database we can share alarm data between dart and kotlin or other native languages without the need for calling method channels.
-
Cross Communication Between Dart and Native: SqLite database can be accessed by any native language with ease, allowing for seamless flow of data between the two.
-
Performance Optimization: Performance and data retrieval times are improved as need for method channels for fetching data is eliminated.
The schema is same as described above.
The Flutter Secure Storage library is utilized in the "Ultimate Alarm Clock" project for securely storing various settings and preferences. This storage solution employs key-value pairs to manage and access data. Below are the keys and their associated purposes:
-
userModel
: Stores user-related data in a JSON-encoded format, including user settings and preferences. -
weather_state
: Stores the current weather state information. -
API keys
: Several keys are used for storing API keys securely, allowing the application to access external services. -
Haptic Feedback
: Key-value pairs are used to store and manage user preferences related to haptic feedback settings. -
Sorted Alarm List
: Key-value pairs are used to store and manage user preferences for sorting the alarm list. -
theme_value
: Stores the selected theme (e.g., dark or light) for the application.
Flutter Secure Storage is instrumental in ensuring the security and privacy of sensitive user data and preferences, contributing to a seamless and secure user experience in the "Ultimate Alarm Clock" project.
Before getting started, ensure you have the following prerequisites installed on your system:
-
Flutter: Install the latest version of Flutter, including Dart SDK, by following the official installation guide.
-
Git: Version control tool to clone the project's repository.
-
A code editor such as Visual Studio Code or Android Studio with Flutter and Dart plugins.
-
Clone the Repository:
Open your terminal or command prompt and navigate to the directory where you want to store the project. Then, run the following command to clone the repository:
git clone https://github.com/CCExtractor/ultimate_alarm_clock.git
-
Navigate to the Project Directory:
Change your working directory to the project folder:
cd ultimate-alarm-clock
-
Install Dependencies:
Use the
flutter pub get
command to install the project's dependencies:flutter pub get
-
Run the Application:
You can run the application on a connected device (emulator or physical device) using the following command:
flutter run
This command will compile and launch the "Ultimate Alarm Clock" app on your device.
The "Ultimate Alarm Clock" offers a user-friendly and versatile interface designed to meet your alarm management needs. This section provides an overview of the app's user interface, highlighting key features and functionalities.
- The app opens with a welcoming splash screen, providing a brief introduction to the application.
- The "Ultimate Alarm Clock" offers both light and dark modes to suit your preferred theme and enhance readability in different lighting conditions.
- After the splash screen, you are directed to the Home View, where you can manage your alarms.
- The Home View allows you to:
- View all your alarms.
- Toggle alarms on/off.
- Preview alarms.
- Delete alarms.
- Edit alarms.
- Create new alarms using the floating action button.
- A settings icon in the app bar provides access to the Settings View.
- In the Settings View, you can customize various aspects of the app:
- Set or edit the weather API key for weather-related features.
- Log in with your Google account.
- Toggle haptic feedback on/off.
- Toggle the sorting of the alarm list based on time.
- Toggle between light and dark modes.
- The floating action button in the Home View allows you to create alarms, directing you to the Add or Update Alarm View.
- In this view, you can:
- Set the alarm time using a time picker or manually entering time using keyboard.
- Configure alarm repetition.
- Set the snooze duration.
- Add a label to the alarm.
- Define automatic cancellation conditions based on screen activity, weather, and location.
- Choose from challenges, including shake to dismiss, QR code, and math challenges.
- Manage shared alarms.
The "Ultimate Alarm Clock" project introduces the feature of shared alarms, allowing users to collaborate with friends, family members, or colleagues to ensure they wake up on time.
-
Google Sign-In: To create or join a shared alarm, users must first sign in with their Google account. This authentication step ensures that only registered users can access this feature and maintains the security of shared alarms.
-
Creating a Shared Alarm:
- From the Home View, users can create a new alarm by tapping the "Create Alarm" button.
- During the alarm creation process, users have the option to make it a "shared alarm."
-
Joining an Existing Shared Alarm:
- Users can join an existing shared alarm by entering a unique shared alarm ID or accepting an invitation from the owner.
- The shared alarm ID is a unique code that the owner shares with potential collaborators.
The addUserToAlarmSharedUsers
function enables users to add new collaborators to a shared alarm. Here's how it works:
- When a user initiates the process to join a shared alarm, this function is called.
- It begins by checking whether the provided alarm ID exists in the database.
- If the alarm ID is not found, the function returns
false
, indicating that the addition process failed. - If the alarm ID is found, the function proceeds to add the user as a collaborator.
- If the user is the owner of the alarm, the function returns
null
, preventing the owner from being added as a collaborator. - The function then updates the list of shared user IDs and offset details for the alarm. It sets
isOffsetBefore
totrue
,offsetDuration
to0
, andoffsettedTime
to the alarm's original time. - The user is added to the list of shared user IDs, and the changes are instantly synchronized with all collaborators.
The removeUserFromAlarmSharedUsers
function allows the owner of a shared alarm, to remove a collaborator from the shared alarm. Here's how it works:
- When the owner initiates the removal process, this function is called.
- Similar to the previous function, it begins by checking whether the provided alarm ID exists in the database.
- If the alarm ID is not found, the function returns an empty list, indicating that the removal process failed due to the alarm's non-existence.
- If the alarm ID is found, the function proceeds to remove the specified user from the list of shared user IDs.
- After the removal, the function updates the shared user IDs for the alarm, and the changes are immediately synchronized with all collaborators.
-
Data Sources: It combines data from three primary data sources:
- Firestore: Alarms stored in the Firestore database.
- Shared Alarms: Alarms shared with the user through collaboration.
- IsarDB: Alarms stored locally for offline use.
-
Data Transformation: The function processes the data retrieved from these sources. This transformation includes converting the raw document snapshots into
AlarmModel
objects, a data structure that represents alarm records. -
Sorting: Depending on user preferences, the alarms are sorted. Users have the option to enable or disable sorting, and two different sorting methods are applied based on this preference:
- Time-Based Sorting: If the sorting preference is enabled, the alarms are sorted in ascending order based on their scheduled time.
- Priority Sorting: If the sorting preference is disabled, the alarms are organized based on priority:
- First, the enabled alarms are listed ahead of disabled ones.
- Then, alarms are sorted by their upcoming time, factoring in repetition patterns and immediate execution for one-time alarms.
-
Returned Stream: After these processes, the function returns a stream (
streamAlarms
) that represents the list of alarms. This stream reflects the real-time status of alarms and is used to update the user interface whenever alarms are added, modified, or removed.
The "Ultimate Alarm Clock" offers an intuitive and feature-rich user interface, making it easy to set, manage, and customize your alarms while providing a seamless experience.
Thank you for your interest in contributing to the "Ultimate Alarm Clock" project. Contributions from the open-source community are highly valued and help improve the application. Please read the following guidelines to understand how you can contribute effectively.
-
Fork the Repository: Start by forking the project's repository to your own GitHub account.
-
Clone the Repository: Clone the forked repository to your local development environment using the
git clone
command.git clone https://github.com/your-username/ultimate-alarm-clock.git
-
Create a Branch: Create a new branch for your contributions, giving it a descriptive name.
git checkout -b your-feature-name
-
Make Changes: Make your desired changes, improvements, or bug fixes in your local environment.
-
Test: Ensure that your changes do not introduce new issues and do not break existing features. Test your code thoroughly.
-
Documentation: If your changes impact the user interface, configuration, or functionality, update the documentation to reflect the changes.
-
Commit: Commit your changes with a clear and concise message.
git commit -m "Add feature/fix: Describe your changes here"
-
Push Changes: Push your changes to your GitHub fork.
git push origin your-feature-name
-
Pull Request: Create a Pull Request (PR) from your fork to the original repository. Ensure your PR has a clear title and description outlining the changes.
-
Code Review: Your PR will undergo code review. Make any necessary adjustments based on feedback.
-
Merge: Once your PR is approved, it will be merged into the main project repository.
- Be respectful and considerate when contributing and interacting with the community.
- Follow the project's coding style, conventions, and best practices.
- Keep your PR focused on a single issue or feature. If you wish to contribute multiple changes, create separate branches and PRs for each.
- Provide a detailed and clear description of your PR, including the purpose of the changes and any related issues.
- Ensure that your code is well-documented and that any new features or changes are reflected in the project's documentation.
- Make sure your contributions do not introduce security vulnerabilities or cause regressions.
If you find a bug or have a suggestion for improvement, please create an issue on the project's GitHub repository. Be sure to include a clear and detailed description of the problem or enhancement.
We appreciate your contributions to the "Ultimate Alarm Clock" project, and your help is invaluable in making it even better.
If you have any questions regarding something in the project, do not hesitate to ask :)
- We want to add Google Assistant commands to set alarms directly without the need to open the app.
- We are working on implementing Google Cloud functions to make shared alarms more interactive by allowing users to disable or reconfigure alarms for others.
- We have plans to make architectural and data flow changes within the application to enhance its overall performance and maintainability. These changes will optimize resource utilization and streamline the user experience.
We would love to hear from you! You may join the CCExtractor community through Slack:
For help in getting started with Flutter, view online documentation, which offers tutorials, samples, guidance on mobile development, and a full API reference.