├── .github └── FUNDING.yml ├── AUTHORS ├── COPYING ├── ISSUE_TEMPLATE.md ├── README.md ├── data ├── icon.png ├── icons │ ├── hicolor │ │ ├── meson.build │ │ └── scalable │ │ │ └── actions │ │ │ └── color-picker-symbolic.svg │ ├── io.github.lainsce.Countdown-symbolic.svg │ ├── io.github.lainsce.Countdown.Devel.svg │ ├── io.github.lainsce.Countdown.svg │ └── meson.build ├── io.github.lainsce.Countdown.Source.svg ├── io.github.lainsce.Countdown.desktop.in ├── io.github.lainsce.Countdown.gschema.xml ├── io.github.lainsce.Countdown.metainfo.xml.in ├── meson.build ├── shot-dark.png └── shot.png ├── io.github.lainsce.Countdown.doap ├── io.github.lainsce.Countdown.json ├── meson.build ├── meson_options.txt ├── po ├── LINGUAS ├── POTFILES ├── README.md ├── es.po ├── io.github.lainsce.Countdown.pot ├── it.po ├── meson.build ├── nl.po ├── ru.po └── tr.po └── src ├── Models └── event.vala ├── Repositories ├── eventrepository.vala └── pasteventrepository.vala ├── Utils ├── file.vala ├── misc.vala ├── obs_list.vala ├── sorter.vala └── thread.vala ├── ViewModels ├── eventviewmodel.vala └── pasteventviewmodel.vala ├── Views ├── pastlistview.ui ├── pastlistview.ui.vala ├── upcominglistview.ui ├── upcominglistview.ui.vala └── view.vala ├── Widgets ├── dialog.ui ├── dialog.ui.vala ├── eventrow.ui ├── eventrowcontent.ui ├── eventrowcontent.ui.vala ├── keys.ui ├── mainwindow.ui ├── mainwindow.ui.vala └── menu.ui ├── app.vala ├── config.vapi ├── io.github.lainsce.Countdown.gresource.xml ├── style-dark.css ├── style-hc.css └── style.css /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | # These are supported funding model platforms 2 | 3 | patreon: lainsce 4 | github: lainsce 5 | -------------------------------------------------------------------------------- /AUTHORS: -------------------------------------------------------------------------------- 1 | Lains 2 | -------------------------------------------------------------------------------- /COPYING: -------------------------------------------------------------------------------- 1 | GNU GENERAL PUBLIC LICENSE 2 | Version 3, 29 June 2007 3 | 4 | Copyright (C) 2007 Free Software Foundation, Inc. 5 | Everyone is permitted to copy and distribute verbatim copies 6 | of this license document, but changing it is not allowed. 7 | 8 | Preamble 9 | 10 | The GNU General Public License is a free, copyleft license for 11 | software and other kinds of works. 12 | 13 | The licenses for most software and other practical works are designed 14 | to take away your freedom to share and change the works. By contrast, 15 | the GNU General Public License is intended to guarantee your freedom to 16 | share and change all versions of a program--to make sure it remains free 17 | software for all its users. We, the Free Software Foundation, use the 18 | GNU General Public License for most of our software; it applies also to 19 | any other work released this way by its authors. You can apply it to 20 | your programs, too. 21 | 22 | When we speak of free software, we are referring to freedom, not 23 | price. Our General Public Licenses are designed to make sure that you 24 | have the freedom to distribute copies of free software (and charge for 25 | them if you wish), that you receive source code or can get it if you 26 | want it, that you can change the software or use pieces of it in new 27 | free programs, and that you know you can do these things. 28 | 29 | To protect your rights, we need to prevent others from denying you 30 | these rights or asking you to surrender the rights. Therefore, you have 31 | certain responsibilities if you distribute copies of the software, or if 32 | you modify it: responsibilities to respect the freedom of others. 33 | 34 | For example, if you distribute copies of such a program, whether 35 | gratis or for a fee, you must pass on to the recipients the same 36 | freedoms that you received. You must make sure that they, too, receive 37 | or can get the source code. And you must show them these terms so they 38 | know their rights. 39 | 40 | Developers that use the GNU GPL protect your rights with two steps: 41 | (1) assert copyright on the software, and (2) offer you this License 42 | giving you legal permission to copy, distribute and/or modify it. 43 | 44 | For the developers' and authors' protection, the GPL clearly explains 45 | that there is no warranty for this free software. For both users' and 46 | authors' sake, the GPL requires that modified versions be marked as 47 | changed, so that their problems will not be attributed erroneously to 48 | authors of previous versions. 49 | 50 | Some devices are designed to deny users access to install or run 51 | modified versions of the software inside them, although the manufacturer 52 | can do so. This is fundamentally incompatible with the aim of 53 | protecting users' freedom to change the software. The systematic 54 | pattern of such abuse occurs in the area of products for individuals to 55 | use, which is precisely where it is most unacceptable. Therefore, we 56 | have designed this version of the GPL to prohibit the practice for those 57 | products. If such problems arise substantially in other domains, we 58 | stand ready to extend this provision to those domains in future versions 59 | of the GPL, as needed to protect the freedom of users. 60 | 61 | Finally, every program is threatened constantly by software patents. 62 | States should not allow patents to restrict development and use of 63 | software on general-purpose computers, but in those that do, we wish to 64 | avoid the special danger that patents applied to a free program could 65 | make it effectively proprietary. To prevent this, the GPL assures that 66 | patents cannot be used to render the program non-free. 67 | 68 | The precise terms and conditions for copying, distribution and 69 | modification follow. 70 | 71 | TERMS AND CONDITIONS 72 | 73 | 0. Definitions. 74 | 75 | "This License" refers to version 3 of the GNU General Public License. 76 | 77 | "Copyright" also means copyright-like laws that apply to other kinds of 78 | works, such as semiconductor masks. 79 | 80 | "The Program" refers to any copyrightable work licensed under this 81 | License. Each licensee is addressed as "you". "Licensees" and 82 | "recipients" may be individuals or organizations. 83 | 84 | To "modify" a work means to copy from or adapt all or part of the work 85 | in a fashion requiring copyright permission, other than the making of an 86 | exact copy. The resulting work is called a "modified version" of the 87 | earlier work or a work "based on" the earlier work. 88 | 89 | A "covered work" means either the unmodified Program or a work based 90 | on the Program. 91 | 92 | To "propagate" a work means to do anything with it that, without 93 | permission, would make you directly or secondarily liable for 94 | infringement under applicable copyright law, except executing it on a 95 | computer or modifying a private copy. Propagation includes copying, 96 | distribution (with or without modification), making available to the 97 | public, and in some countries other activities as well. 98 | 99 | To "convey" a work means any kind of propagation that enables other 100 | parties to make or receive copies. Mere interaction with a user through 101 | a computer network, with no transfer of a copy, is not conveying. 102 | 103 | An interactive user interface displays "Appropriate Legal Notices" 104 | to the extent that it includes a convenient and prominently visible 105 | feature that (1) displays an appropriate copyright notice, and (2) 106 | tells the user that there is no warranty for the work (except to the 107 | extent that warranties are provided), that licensees may convey the 108 | work under this License, and how to view a copy of this License. If 109 | the interface presents a list of user commands or options, such as a 110 | menu, a prominent item in the list meets this criterion. 111 | 112 | 1. Source Code. 113 | 114 | The "source code" for a work means the preferred form of the work 115 | for making modifications to it. "Object code" means any non-source 116 | form of a work. 117 | 118 | A "Standard Interface" means an interface that either is an official 119 | standard defined by a recognized standards body, or, in the case of 120 | interfaces specified for a particular programming language, one that 121 | is widely used among developers working in that language. 122 | 123 | The "System Libraries" of an executable work include anything, other 124 | than the work as a whole, that (a) is included in the normal form of 125 | packaging a Major Component, but which is not part of that Major 126 | Component, and (b) serves only to enable use of the work with that 127 | Major Component, or to implement a Standard Interface for which an 128 | implementation is available to the public in source code form. A 129 | "Major Component", in this context, means a major essential component 130 | (kernel, window system, and so on) of the specific operating system 131 | (if any) on which the executable work runs, or a compiler used to 132 | produce the work, or an object code interpreter used to run it. 133 | 134 | The "Corresponding Source" for a work in object code form means all 135 | the source code needed to generate, install, and (for an executable 136 | work) run the object code and to modify the work, including scripts to 137 | control those activities. However, it does not include the work's 138 | System Libraries, or general-purpose tools or generally available free 139 | programs which are used unmodified in performing those activities but 140 | which are not part of the work. For example, Corresponding Source 141 | includes interface definition files associated with source files for 142 | the work, and the source code for shared libraries and dynamically 143 | linked subprograms that the work is specifically designed to require, 144 | such as by intimate data communication or control flow between those 145 | subprograms and other parts of the work. 146 | 147 | The Corresponding Source need not include anything that users 148 | can regenerate automatically from other parts of the Corresponding 149 | Source. 150 | 151 | The Corresponding Source for a work in source code form is that 152 | same work. 153 | 154 | 2. Basic Permissions. 155 | 156 | All rights granted under this License are granted for the term of 157 | copyright on the Program, and are irrevocable provided the stated 158 | conditions are met. This License explicitly affirms your unlimited 159 | permission to run the unmodified Program. The output from running a 160 | covered work is covered by this License only if the output, given its 161 | content, constitutes a covered work. This License acknowledges your 162 | rights of fair use or other equivalent, as provided by copyright law. 163 | 164 | You may make, run and propagate covered works that you do not 165 | convey, without conditions so long as your license otherwise remains 166 | in force. You may convey covered works to others for the sole purpose 167 | of having them make modifications exclusively for you, or provide you 168 | with facilities for running those works, provided that you comply with 169 | the terms of this License in conveying all material for which you do 170 | not control copyright. Those thus making or running the covered works 171 | for you must do so exclusively on your behalf, under your direction 172 | and control, on terms that prohibit them from making any copies of 173 | your copyrighted material outside their relationship with you. 174 | 175 | Conveying under any other circumstances is permitted solely under 176 | the conditions stated below. Sublicensing is not allowed; section 10 177 | makes it unnecessary. 178 | 179 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law. 180 | 181 | No covered work shall be deemed part of an effective technological 182 | measure under any applicable law fulfilling obligations under article 183 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or 184 | similar laws prohibiting or restricting circumvention of such 185 | measures. 186 | 187 | When you convey a covered work, you waive any legal power to forbid 188 | circumvention of technological measures to the extent such circumvention 189 | is effected by exercising rights under this License with respect to 190 | the covered work, and you disclaim any intention to limit operation or 191 | modification of the work as a means of enforcing, against the work's 192 | users, your or third parties' legal rights to forbid circumvention of 193 | technological measures. 194 | 195 | 4. Conveying Verbatim Copies. 196 | 197 | You may convey verbatim copies of the Program's source code as you 198 | receive it, in any medium, provided that you conspicuously and 199 | appropriately publish on each copy an appropriate copyright notice; 200 | keep intact all notices stating that this License and any 201 | non-permissive terms added in accord with section 7 apply to the code; 202 | keep intact all notices of the absence of any warranty; and give all 203 | recipients a copy of this License along with the Program. 204 | 205 | You may charge any price or no price for each copy that you convey, 206 | and you may offer support or warranty protection for a fee. 207 | 208 | 5. Conveying Modified Source Versions. 209 | 210 | You may convey a work based on the Program, or the modifications to 211 | produce it from the Program, in the form of source code under the 212 | terms of section 4, provided that you also meet all of these conditions: 213 | 214 | a) The work must carry prominent notices stating that you modified 215 | it, and giving a relevant date. 216 | 217 | b) The work must carry prominent notices stating that it is 218 | released under this License and any conditions added under section 219 | 7. This requirement modifies the requirement in section 4 to 220 | "keep intact all notices". 221 | 222 | c) You must license the entire work, as a whole, under this 223 | License to anyone who comes into possession of a copy. This 224 | License will therefore apply, along with any applicable section 7 225 | additional terms, to the whole of the work, and all its parts, 226 | regardless of how they are packaged. This License gives no 227 | permission to license the work in any other way, but it does not 228 | invalidate such permission if you have separately received it. 229 | 230 | d) If the work has interactive user interfaces, each must display 231 | Appropriate Legal Notices; however, if the Program has interactive 232 | interfaces that do not display Appropriate Legal Notices, your 233 | work need not make them do so. 234 | 235 | A compilation of a covered work with other separate and independent 236 | works, which are not by their nature extensions of the covered work, 237 | and which are not combined with it such as to form a larger program, 238 | in or on a volume of a storage or distribution medium, is called an 239 | "aggregate" if the compilation and its resulting copyright are not 240 | used to limit the access or legal rights of the compilation's users 241 | beyond what the individual works permit. Inclusion of a covered work 242 | in an aggregate does not cause this License to apply to the other 243 | parts of the aggregate. 244 | 245 | 6. Conveying Non-Source Forms. 246 | 247 | You may convey a covered work in object code form under the terms 248 | of sections 4 and 5, provided that you also convey the 249 | machine-readable Corresponding Source under the terms of this License, 250 | in one of these ways: 251 | 252 | a) Convey the object code in, or embodied in, a physical product 253 | (including a physical distribution medium), accompanied by the 254 | Corresponding Source fixed on a durable physical medium 255 | customarily used for software interchange. 256 | 257 | b) Convey the object code in, or embodied in, a physical product 258 | (including a physical distribution medium), accompanied by a 259 | written offer, valid for at least three years and valid for as 260 | long as you offer spare parts or customer support for that product 261 | model, to give anyone who possesses the object code either (1) a 262 | copy of the Corresponding Source for all the software in the 263 | product that is covered by this License, on a durable physical 264 | medium customarily used for software interchange, for a price no 265 | more than your reasonable cost of physically performing this 266 | conveying of source, or (2) access to copy the 267 | Corresponding Source from a network server at no charge. 268 | 269 | c) Convey individual copies of the object code with a copy of the 270 | written offer to provide the Corresponding Source. This 271 | alternative is allowed only occasionally and noncommercially, and 272 | only if you received the object code with such an offer, in accord 273 | with subsection 6b. 274 | 275 | d) Convey the object code by offering access from a designated 276 | place (gratis or for a charge), and offer equivalent access to the 277 | Corresponding Source in the same way through the same place at no 278 | further charge. You need not require recipients to copy the 279 | Corresponding Source along with the object code. If the place to 280 | copy the object code is a network server, the Corresponding Source 281 | may be on a different server (operated by you or a third party) 282 | that supports equivalent copying facilities, provided you maintain 283 | clear directions next to the object code saying where to find the 284 | Corresponding Source. Regardless of what server hosts the 285 | Corresponding Source, you remain obligated to ensure that it is 286 | available for as long as needed to satisfy these requirements. 287 | 288 | e) Convey the object code using peer-to-peer transmission, provided 289 | you inform other peers where the object code and Corresponding 290 | Source of the work are being offered to the general public at no 291 | charge under subsection 6d. 292 | 293 | A separable portion of the object code, whose source code is excluded 294 | from the Corresponding Source as a System Library, need not be 295 | included in conveying the object code work. 296 | 297 | A "User Product" is either (1) a "consumer product", which means any 298 | tangible personal property which is normally used for personal, family, 299 | or household purposes, or (2) anything designed or sold for incorporation 300 | into a dwelling. In determining whether a product is a consumer product, 301 | doubtful cases shall be resolved in favor of coverage. For a particular 302 | product received by a particular user, "normally used" refers to a 303 | typical or common use of that class of product, regardless of the status 304 | of the particular user or of the way in which the particular user 305 | actually uses, or expects or is expected to use, the product. A product 306 | is a consumer product regardless of whether the product has substantial 307 | commercial, industrial or non-consumer uses, unless such uses represent 308 | the only significant mode of use of the product. 309 | 310 | "Installation Information" for a User Product means any methods, 311 | procedures, authorization keys, or other information required to install 312 | and execute modified versions of a covered work in that User Product from 313 | a modified version of its Corresponding Source. The information must 314 | suffice to ensure that the continued functioning of the modified object 315 | code is in no case prevented or interfered with solely because 316 | modification has been made. 317 | 318 | If you convey an object code work under this section in, or with, or 319 | specifically for use in, a User Product, and the conveying occurs as 320 | part of a transaction in which the right of possession and use of the 321 | User Product is transferred to the recipient in perpetuity or for a 322 | fixed term (regardless of how the transaction is characterized), the 323 | Corresponding Source conveyed under this section must be accompanied 324 | by the Installation Information. But this requirement does not apply 325 | if neither you nor any third party retains the ability to install 326 | modified object code on the User Product (for example, the work has 327 | been installed in ROM). 328 | 329 | The requirement to provide Installation Information does not include a 330 | requirement to continue to provide support service, warranty, or updates 331 | for a work that has been modified or installed by the recipient, or for 332 | the User Product in which it has been modified or installed. Access to a 333 | network may be denied when the modification itself materially and 334 | adversely affects the operation of the network or violates the rules and 335 | protocols for communication across the network. 336 | 337 | Corresponding Source conveyed, and Installation Information provided, 338 | in accord with this section must be in a format that is publicly 339 | documented (and with an implementation available to the public in 340 | source code form), and must require no special password or key for 341 | unpacking, reading or copying. 342 | 343 | 7. Additional Terms. 344 | 345 | "Additional permissions" are terms that supplement the terms of this 346 | License by making exceptions from one or more of its conditions. 347 | Additional permissions that are applicable to the entire Program shall 348 | be treated as though they were included in this License, to the extent 349 | that they are valid under applicable law. If additional permissions 350 | apply only to part of the Program, that part may be used separately 351 | under those permissions, but the entire Program remains governed by 352 | this License without regard to the additional permissions. 353 | 354 | When you convey a copy of a covered work, you may at your option 355 | remove any additional permissions from that copy, or from any part of 356 | it. (Additional permissions may be written to require their own 357 | removal in certain cases when you modify the work.) You may place 358 | additional permissions on material, added by you to a covered work, 359 | for which you have or can give appropriate copyright permission. 360 | 361 | Notwithstanding any other provision of this License, for material you 362 | add to a covered work, you may (if authorized by the copyright holders of 363 | that material) supplement the terms of this License with terms: 364 | 365 | a) Disclaiming warranty or limiting liability differently from the 366 | terms of sections 15 and 16 of this License; or 367 | 368 | b) Requiring preservation of specified reasonable legal notices or 369 | author attributions in that material or in the Appropriate Legal 370 | Notices displayed by works containing it; or 371 | 372 | c) Prohibiting misrepresentation of the origin of that material, or 373 | requiring that modified versions of such material be marked in 374 | reasonable ways as different from the original version; or 375 | 376 | d) Limiting the use for publicity purposes of names of licensors or 377 | authors of the material; or 378 | 379 | e) Declining to grant rights under trademark law for use of some 380 | trade names, trademarks, or service marks; or 381 | 382 | f) Requiring indemnification of licensors and authors of that 383 | material by anyone who conveys the material (or modified versions of 384 | it) with contractual assumptions of liability to the recipient, for 385 | any liability that these contractual assumptions directly impose on 386 | those licensors and authors. 387 | 388 | All other non-permissive additional terms are considered "further 389 | restrictions" within the meaning of section 10. If the Program as you 390 | received it, or any part of it, contains a notice stating that it is 391 | governed by this License along with a term that is a further 392 | restriction, you may remove that term. If a license document contains 393 | a further restriction but permits relicensing or conveying under this 394 | License, you may add to a covered work material governed by the terms 395 | of that license document, provided that the further restriction does 396 | not survive such relicensing or conveying. 397 | 398 | If you add terms to a covered work in accord with this section, you 399 | must place, in the relevant source files, a statement of the 400 | additional terms that apply to those files, or a notice indicating 401 | where to find the applicable terms. 402 | 403 | Additional terms, permissive or non-permissive, may be stated in the 404 | form of a separately written license, or stated as exceptions; 405 | the above requirements apply either way. 406 | 407 | 8. Termination. 408 | 409 | You may not propagate or modify a covered work except as expressly 410 | provided under this License. Any attempt otherwise to propagate or 411 | modify it is void, and will automatically terminate your rights under 412 | this License (including any patent licenses granted under the third 413 | paragraph of section 11). 414 | 415 | However, if you cease all violation of this License, then your 416 | license from a particular copyright holder is reinstated (a) 417 | provisionally, unless and until the copyright holder explicitly and 418 | finally terminates your license, and (b) permanently, if the copyright 419 | holder fails to notify you of the violation by some reasonable means 420 | prior to 60 days after the cessation. 421 | 422 | Moreover, your license from a particular copyright holder is 423 | reinstated permanently if the copyright holder notifies you of the 424 | violation by some reasonable means, this is the first time you have 425 | received notice of violation of this License (for any work) from that 426 | copyright holder, and you cure the violation prior to 30 days after 427 | your receipt of the notice. 428 | 429 | Termination of your rights under this section does not terminate the 430 | licenses of parties who have received copies or rights from you under 431 | this License. If your rights have been terminated and not permanently 432 | reinstated, you do not qualify to receive new licenses for the same 433 | material under section 10. 434 | 435 | 9. Acceptance Not Required for Having Copies. 436 | 437 | You are not required to accept this License in order to receive or 438 | run a copy of the Program. Ancillary propagation of a covered work 439 | occurring solely as a consequence of using peer-to-peer transmission 440 | to receive a copy likewise does not require acceptance. However, 441 | nothing other than this License grants you permission to propagate or 442 | modify any covered work. These actions infringe copyright if you do 443 | not accept this License. Therefore, by modifying or propagating a 444 | covered work, you indicate your acceptance of this License to do so. 445 | 446 | 10. Automatic Licensing of Downstream Recipients. 447 | 448 | Each time you convey a covered work, the recipient automatically 449 | receives a license from the original licensors, to run, modify and 450 | propagate that work, subject to this License. You are not responsible 451 | for enforcing compliance by third parties with this License. 452 | 453 | An "entity transaction" is a transaction transferring control of an 454 | organization, or substantially all assets of one, or subdividing an 455 | organization, or merging organizations. If propagation of a covered 456 | work results from an entity transaction, each party to that 457 | transaction who receives a copy of the work also receives whatever 458 | licenses to the work the party's predecessor in interest had or could 459 | give under the previous paragraph, plus a right to possession of the 460 | Corresponding Source of the work from the predecessor in interest, if 461 | the predecessor has it or can get it with reasonable efforts. 462 | 463 | You may not impose any further restrictions on the exercise of the 464 | rights granted or affirmed under this License. For example, you may 465 | not impose a license fee, royalty, or other charge for exercise of 466 | rights granted under this License, and you may not initiate litigation 467 | (including a cross-claim or counterclaim in a lawsuit) alleging that 468 | any patent claim is infringed by making, using, selling, offering for 469 | sale, or importing the Program or any portion of it. 470 | 471 | 11. Patents. 472 | 473 | A "contributor" is a copyright holder who authorizes use under this 474 | License of the Program or a work on which the Program is based. The 475 | work thus licensed is called the contributor's "contributor version". 476 | 477 | A contributor's "essential patent claims" are all patent claims 478 | owned or controlled by the contributor, whether already acquired or 479 | hereafter acquired, that would be infringed by some manner, permitted 480 | by this License, of making, using, or selling its contributor version, 481 | but do not include claims that would be infringed only as a 482 | consequence of further modification of the contributor version. For 483 | purposes of this definition, "control" includes the right to grant 484 | patent sublicenses in a manner consistent with the requirements of 485 | this License. 486 | 487 | Each contributor grants you a non-exclusive, worldwide, royalty-free 488 | patent license under the contributor's essential patent claims, to 489 | make, use, sell, offer for sale, import and otherwise run, modify and 490 | propagate the contents of its contributor version. 491 | 492 | In the following three paragraphs, a "patent license" is any express 493 | agreement or commitment, however denominated, not to enforce a patent 494 | (such as an express permission to practice a patent or covenant not to 495 | sue for patent infringement). To "grant" such a patent license to a 496 | party means to make such an agreement or commitment not to enforce a 497 | patent against the party. 498 | 499 | If you convey a covered work, knowingly relying on a patent license, 500 | and the Corresponding Source of the work is not available for anyone 501 | to copy, free of charge and under the terms of this License, through a 502 | publicly available network server or other readily accessible means, 503 | then you must either (1) cause the Corresponding Source to be so 504 | available, or (2) arrange to deprive yourself of the benefit of the 505 | patent license for this particular work, or (3) arrange, in a manner 506 | consistent with the requirements of this License, to extend the patent 507 | license to downstream recipients. "Knowingly relying" means you have 508 | actual knowledge that, but for the patent license, your conveying the 509 | covered work in a country, or your recipient's use of the covered work 510 | in a country, would infringe one or more identifiable patents in that 511 | country that you have reason to believe are valid. 512 | 513 | If, pursuant to or in connection with a single transaction or 514 | arrangement, you convey, or propagate by procuring conveyance of, a 515 | covered work, and grant a patent license to some of the parties 516 | receiving the covered work authorizing them to use, propagate, modify 517 | or convey a specific copy of the covered work, then the patent license 518 | you grant is automatically extended to all recipients of the covered 519 | work and works based on it. 520 | 521 | A patent license is "discriminatory" if it does not include within 522 | the scope of its coverage, prohibits the exercise of, or is 523 | conditioned on the non-exercise of one or more of the rights that are 524 | specifically granted under this License. You may not convey a covered 525 | work if you are a party to an arrangement with a third party that is 526 | in the business of distributing software, under which you make payment 527 | to the third party based on the extent of your activity of conveying 528 | the work, and under which the third party grants, to any of the 529 | parties who would receive the covered work from you, a discriminatory 530 | patent license (a) in connection with copies of the covered work 531 | conveyed by you (or copies made from those copies), or (b) primarily 532 | for and in connection with specific products or compilations that 533 | contain the covered work, unless you entered into that arrangement, 534 | or that patent license was granted, prior to 28 March 2007. 535 | 536 | Nothing in this License shall be construed as excluding or limiting 537 | any implied license or other defenses to infringement that may 538 | otherwise be available to you under applicable patent law. 539 | 540 | 12. No Surrender of Others' Freedom. 541 | 542 | If conditions are imposed on you (whether by court order, agreement or 543 | otherwise) that contradict the conditions of this License, they do not 544 | excuse you from the conditions of this License. If you cannot convey a 545 | covered work so as to satisfy simultaneously your obligations under this 546 | License and any other pertinent obligations, then as a consequence you may 547 | not convey it at all. For example, if you agree to terms that obligate you 548 | to collect a royalty for further conveying from those to whom you convey 549 | the Program, the only way you could satisfy both those terms and this 550 | License would be to refrain entirely from conveying the Program. 551 | 552 | 13. Use with the GNU Affero General Public License. 553 | 554 | Notwithstanding any other provision of this License, you have 555 | permission to link or combine any covered work with a work licensed 556 | under version 3 of the GNU Affero General Public License into a single 557 | combined work, and to convey the resulting work. The terms of this 558 | License will continue to apply to the part which is the covered work, 559 | but the special requirements of the GNU Affero General Public License, 560 | section 13, concerning interaction through a network will apply to the 561 | combination as such. 562 | 563 | 14. Revised Versions of this License. 564 | 565 | The Free Software Foundation may publish revised and/or new versions of 566 | the GNU General Public License from time to time. Such new versions will 567 | be similar in spirit to the present version, but may differ in detail to 568 | address new problems or concerns. 569 | 570 | Each version is given a distinguishing version number. If the 571 | Program specifies that a certain numbered version of the GNU General 572 | Public License "or any later version" applies to it, you have the 573 | option of following the terms and conditions either of that numbered 574 | version or of any later version published by the Free Software 575 | Foundation. If the Program does not specify a version number of the 576 | GNU General Public License, you may choose any version ever published 577 | by the Free Software Foundation. 578 | 579 | If the Program specifies that a proxy can decide which future 580 | versions of the GNU General Public License can be used, that proxy's 581 | public statement of acceptance of a version permanently authorizes you 582 | to choose that version for the Program. 583 | 584 | Later license versions may give you additional or different 585 | permissions. However, no additional obligations are imposed on any 586 | author or copyright holder as a result of your choosing to follow a 587 | later version. 588 | 589 | 15. Disclaimer of Warranty. 590 | 591 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY 592 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT 593 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY 594 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, 595 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 596 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM 597 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF 598 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 599 | 600 | 16. Limitation of Liability. 601 | 602 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING 603 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS 604 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY 605 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE 606 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF 607 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD 608 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), 609 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF 610 | SUCH DAMAGES. 611 | 612 | 17. Interpretation of Sections 15 and 16. 613 | 614 | If the disclaimer of warranty and limitation of liability provided 615 | above cannot be given local legal effect according to their terms, 616 | reviewing courts shall apply local law that most closely approximates 617 | an absolute waiver of all civil liability in connection with the 618 | Program, unless a warranty or assumption of liability accompanies a 619 | copy of the Program in return for a fee. 620 | 621 | END OF TERMS AND CONDITIONS 622 | 623 | How to Apply These Terms to Your New Programs 624 | 625 | If you develop a new program, and you want it to be of the greatest 626 | possible use to the public, the best way to achieve this is to make it 627 | free software which everyone can redistribute and change under these terms. 628 | 629 | To do so, attach the following notices to the program. It is safest 630 | to attach them to the start of each source file to most effectively 631 | state the exclusion of warranty; and each file should have at least 632 | the "copyright" line and a pointer to where the full notice is found. 633 | 634 | 635 | Copyright (C) 636 | 637 | This program is free software: you can redistribute it and/or modify 638 | it under the terms of the GNU General Public License as published by 639 | the Free Software Foundation, either version 3 of the License, or 640 | (at your option) any later version. 641 | 642 | This program is distributed in the hope that it will be useful, 643 | but WITHOUT ANY WARRANTY; without even the implied warranty of 644 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 645 | GNU General Public License for more details. 646 | 647 | You should have received a copy of the GNU General Public License 648 | along with this program. If not, see . 649 | 650 | Also add information on how to contact you by electronic and paper mail. 651 | 652 | If the program does terminal interaction, make it output a short 653 | notice like this when it starts in an interactive mode: 654 | 655 | Copyright (C) 656 | This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. 657 | This is free software, and you are welcome to redistribute it 658 | under certain conditions; type `show c' for details. 659 | 660 | The hypothetical commands `show w' and `show c' should show the appropriate 661 | parts of the General Public License. Of course, your program's commands 662 | might be different; for a GUI interface, you would use an "about box". 663 | 664 | You should also get your employer (if you work as a programmer) or school, 665 | if any, to sign a "copyright disclaimer" for the program, if necessary. 666 | For more information on this, and how to apply and follow the GNU GPL, see 667 | . 668 | 669 | The GNU General Public License does not permit incorporating your program 670 | into proprietary programs. If your program is a subroutine library, you 671 | may consider it more useful to permit linking proprietary applications with 672 | the library. If this is what you want to do, use the GNU Lesser General 673 | Public License instead of this License. But first, please read 674 | . 675 | 676 | -------------------------------------------------------------------------------- /ISSUE_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | ## Hello! If you found an error in this app, follow this template: 2 | 3 | * Use a descriptive title 4 | * Describe the issue 5 | * State what version you're using 6 | * Take a screenshot of the problem, then use the Markdown property for it: 7 | `![any_name_here](url_here)` 8 | * If the problem needs it, show me a video of the problem with Screenkey running in the background. 9 | 10 | 11 | Thanks! 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 2 | 3 | # Countdown 4 | 5 | Track events until they happen or since they happened 6 | 7 | ### 8 | 9 | [![Please do not theme this app](https://stopthemingmy.app/badge.svg)](https://stopthemingmy.app) 10 | [![License: GPL v3](https://img.shields.io/badge/License-GPL%20v3-blue.svg)](http://www.gnu.org/licenses/gpl-3.0) 11 | 12 | ![Light screenshot](data/shot.png#gh-light-mode-only) 13 | ![Dark screenshot](data/shot-dark.png#gh-dark-mode-only) 14 | 15 |

Download on Flathub

16 | 17 | ## 💝 Donations 18 | 19 | Would you like to support the development of this app to new heights? 20 | Then become a GitHub Sponsor or check my Patreon, buttons in the sidebar. 21 | 22 | ## 🛠️ Dependencies 23 | 24 | Please make sure you have these dependencies first before building. 25 | 26 | ```bash 27 | gtk4 28 | libadwaita-1 29 | libjson-glib 30 | libgee-0.8 31 | meson 32 | vala 33 | ``` 34 | 35 | ## 🏗️ Building 36 | 37 | Simply clone this repo, then: 38 | 39 | ```bash 40 | meson _build --prefix=/usr && cd _build 41 | sudo ninja install 42 | ``` 43 | -------------------------------------------------------------------------------- /data/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lainsce/countdown/45cf6d171a664b066bdfd509a8d5ef997ce266eb/data/icon.png -------------------------------------------------------------------------------- /data/icons/hicolor/meson.build: -------------------------------------------------------------------------------- 1 | install_subdir('scalable', install_dir: icondir) 2 | -------------------------------------------------------------------------------- /data/icons/hicolor/scalable/actions/color-picker-symbolic.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | 94 | 95 | 96 | 97 | 98 | 99 | 100 | 101 | 102 | 103 | 104 | 105 | 106 | 107 | 108 | 109 | 110 | 111 | 112 | 113 | 114 | 115 | 116 | 117 | 118 | 119 | 120 | 121 | 122 | 123 | 124 | 125 | 126 | 127 | 128 | 129 | 130 | 131 | 132 | 133 | 134 | 135 | 136 | 137 | 138 | 139 | 140 | 141 | 142 | 143 | 144 | 145 | 146 | 147 | 148 | 149 | 150 | 151 | -------------------------------------------------------------------------------- /data/icons/io.github.lainsce.Countdown-symbolic.svg: -------------------------------------------------------------------------------- 1 | 2 | 14 | 34 | 36 | 40 | 45 | 48 | 49 | 54 | 57 | 58 | 59 | -------------------------------------------------------------------------------- /data/icons/io.github.lainsce.Countdown.Devel.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | -------------------------------------------------------------------------------- /data/icons/io.github.lainsce.Countdown.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | -------------------------------------------------------------------------------- /data/icons/meson.build: -------------------------------------------------------------------------------- 1 | icondir = join_paths(get_option('datadir'), 'icons/hicolor') 2 | 3 | install_data( 4 | '@0@.svg'.format(app_id), 5 | install_dir: join_paths(icondir, 'scalable/apps'), 6 | ) 7 | 8 | install_data( 9 | meson.project_name() + '-symbolic.svg', 10 | install_dir: join_paths(icondir, 'symbolic/apps'), 11 | rename: '@0@-symbolic.svg'.format(app_id), 12 | ) 13 | 14 | subdir('hicolor') 15 | -------------------------------------------------------------------------------- /data/io.github.lainsce.Countdown.desktop.in: -------------------------------------------------------------------------------- 1 | [Desktop Entry] 2 | Name=Countdown 3 | Exec=io.github.lainsce.Countdown 4 | Icon=@app_id@ 5 | Keywords=event;countdown;days; 6 | Terminal=false 7 | Type=Application 8 | StartupNotify=true 9 | Categories=GTK;Utility; 10 | -------------------------------------------------------------------------------- /data/io.github.lainsce.Countdown.gschema.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /data/io.github.lainsce.Countdown.metainfo.xml.in: -------------------------------------------------------------------------------- 1 | 2 | 3 | @app_id@ 4 | CC0-1.0 5 | GPL-3.0+ 6 | Countdown 7 | Track events until they happen or since they happened 8 | 9 |

Tracking events until they happen, or since they happened, easier.

10 |
11 | 12 | io.github.lainsce.Countdown 13 | 14 | @app_id@.desktop 15 | 16 | ModernToolkit 17 | HiDpiIcon 18 | 19 | Lains 20 | 21 | Lains 22 | 23 | https://github.com/lainsce/Countdown/ 24 | https://github.com/lainsce/Countdown/issues 25 | https://www.ko-fi.com/lainsce/ 26 | https://github.com/lainsce/Countdown/ 27 | https://github.com/lainsce/Countdown/blob/master/po/README.md 28 | 29 | 30 | https://raw.githubusercontent.com/lainsce/Countdown/master/data/shot.png 31 | 32 | 33 | 34 | 35 | keyboard 36 | pointing 37 | touch 38 | 39 | 40 | 360 41 | 42 | 43 | [(196, 165, 44)] 44 | 45 | io.github.lainsce.Countdown 46 | 47 | 48 | 49 |

Release: First Release

50 |
51 |
52 |
53 |
54 | -------------------------------------------------------------------------------- /data/meson.build: -------------------------------------------------------------------------------- 1 | conf = configuration_data() 2 | conf.set('app_id', app_id) 3 | 4 | desktop_conf = configure_file( 5 | input: meson.project_name() + '.desktop.in', 6 | output: '@0@.desktop.in'.format(app_id), 7 | configuration: conf 8 | ) 9 | 10 | desktop_file = i18n.merge_file( 11 | input: desktop_conf, 12 | output: '@0@.desktop'.format(app_id), 13 | type: 'desktop', 14 | po_dir: '../po', 15 | install: true, 16 | install_dir: join_paths(get_option('datadir'), 'applications') 17 | ) 18 | 19 | # Validate Desktop file 20 | desktop_file_validate = find_program('desktop-file-validate', required: false) 21 | if desktop_file_validate.found() 22 | test('validate-desktop', desktop_file_validate, 23 | args: [desktop_file] 24 | ) 25 | endif 26 | 27 | appstream_conf = configure_file( 28 | input: meson.project_name() + '.metainfo.xml.in', 29 | output: '@0@.metainfo.xml.in'.format(app_id), 30 | configuration: conf 31 | ) 32 | 33 | appstream_file = i18n.merge_file( 34 | input: appstream_conf, 35 | output: '@0@.metainfo.xml'.format(app_id), 36 | po_dir: '../po', 37 | install: true, 38 | install_dir: join_paths(get_option('datadir'), 'metainfo') 39 | ) 40 | 41 | #Validate Appstream file 42 | appstream_file_validate = find_program('appstream-util', required: false) 43 | if appstream_file_validate.found() 44 | test('validate-appstream', appstream_file_validate, 45 | args: ['validate', '--nonet', appstream_file] 46 | ) 47 | endif 48 | 49 | gnome.compile_schemas(build_by_default: true) 50 | install_data( 51 | meson.project_name() + '.gschema.xml', 52 | install_dir: join_paths(get_option('datadir'), 'glib-2.0/schemas') 53 | ) 54 | 55 | subdir('icons') 56 | -------------------------------------------------------------------------------- /data/shot-dark.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lainsce/countdown/45cf6d171a664b066bdfd509a8d5ef997ce266eb/data/shot-dark.png -------------------------------------------------------------------------------- /data/shot.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lainsce/countdown/45cf6d171a664b066bdfd509a8d5ef997ce266eb/data/shot.png -------------------------------------------------------------------------------- /io.github.lainsce.Countdown.doap: -------------------------------------------------------------------------------- 1 | 2 | 7 | Countdown 8 | 9 | Track events until they happen or since they happened 10 | 11 | 12 |

Tracking events until they happen, or since they happened, easier.

13 |
14 | 15 | 16 | Vala 17 | 18 | 19 | Lains 20 | 21 | lainsce 22 | 23 | 24 |
25 | -------------------------------------------------------------------------------- /io.github.lainsce.Countdown.json: -------------------------------------------------------------------------------- 1 | { 2 | "app-id" : "io.github.lainsce.Countdown.Devel", 3 | "runtime" : "org.gnome.Sdk", 4 | "runtime-version" : "44", 5 | "sdk" : "org.gnome.Sdk", 6 | "command" : "io.github.lainsce.Countdown", 7 | "finish-args" : [ 8 | "--share=ipc", 9 | "--socket=fallback-x11", 10 | "--device=dri", 11 | "--socket=wayland" 12 | ], 13 | "desktop-file-name-suffix" : " (Development)", 14 | "cleanup" : [ 15 | "/include", 16 | "/lib/pkgconfig", 17 | "/man", 18 | "/share/doc", 19 | "/share/gtk-doc", 20 | "/share/man", 21 | "/share/pkgconfig", 22 | "/share/vala", 23 | "*.la", 24 | "*.a" 25 | ], 26 | "modules" : [ 27 | { 28 | "name" : "Countdown", 29 | "builddir" : true, 30 | "buildsystem" : "meson", 31 | "config-opts" : [ 32 | "-Ddevelopment=true" 33 | ], 34 | "sources" : [ 35 | { 36 | "type" : "dir", 37 | "path" : "." 38 | } 39 | ] 40 | } 41 | ] 42 | } 43 | -------------------------------------------------------------------------------- /meson.build: -------------------------------------------------------------------------------- 1 | project('io.github.lainsce.Countdown', 'vala', 2 | version: '0.1.0', 3 | meson_version: '>= 0.59.0' 4 | ) 5 | 6 | gnome = import('gnome') 7 | i18n = import('i18n') 8 | 9 | add_project_arguments([ 10 | '--target-glib=2.68', 11 | ], 12 | language: 'vala', 13 | ) 14 | 15 | if get_option('development') 16 | app_id = 'io.github.lainsce.Countdown.Devel' 17 | name_suffix = ' (Development)' 18 | else 19 | app_id = 'io.github.lainsce.Countdown' 20 | name_suffix = '' 21 | endif 22 | 23 | conf = configuration_data() 24 | conf.set_quoted('APP_ID', app_id) 25 | conf.set_quoted('NAME_SUFFIX', name_suffix) 26 | conf.set_quoted('VERSION', meson.project_version()) 27 | conf.set_quoted('GETTEXT_PACKAGE', app_id) 28 | conf.set_quoted('LOCALEDIR', join_paths(get_option('prefix'), get_option('localedir'))) 29 | conf.set10('DEVELOPMENT', get_option('development')) 30 | configure_file(output: 'config.h', configuration: conf) 31 | config_h_dir = include_directories('.') 32 | 33 | add_project_arguments( 34 | '-include', 'config.h', 35 | '-DGETTEXT_PACKAGE="@0@"'.format(app_id), 36 | language: 'c' 37 | ) 38 | 39 | asresources = gnome.compile_resources( 40 | 'as-resources', 'src/io.github.lainsce.Countdown.gresource.xml', 41 | source_dir: 'src', 42 | c_name: 'as' 43 | ) 44 | 45 | sources = [ 46 | 'src/app.vala', 47 | 'src/Widgets/dialog.ui.vala', 48 | 'src/Widgets/mainwindow.ui.vala', 49 | 'src/Widgets/eventrowcontent.ui.vala', 50 | 'src/Models/event.vala', 51 | 'src/Repositories/eventrepository.vala', 52 | 'src/ViewModels/eventviewmodel.vala', 53 | 'src/Views/view.vala', 54 | 'src/Views/upcominglistview.ui.vala', 55 | 'src/Repositories/pasteventrepository.vala', 56 | 'src/ViewModels/pasteventviewmodel.vala', 57 | 'src/Views/pastlistview.ui.vala', 58 | 'src/Utils/file.vala', 59 | 'src/Utils/misc.vala', 60 | 'src/Utils/obs_list.vala', 61 | 'src/Utils/sorter.vala', 62 | 'src/Utils/thread.vala', 63 | ] 64 | 65 | dependencies = [ 66 | dependency('gio-2.0'), 67 | dependency('gtk4'), 68 | dependency('glib-2.0'), 69 | dependency('gobject-2.0'), 70 | dependency('gee-0.8'), 71 | dependency('libadwaita-1'), 72 | dependency('gmodule-2.0'), 73 | dependency('json-glib-1.0'), 74 | meson.get_compiler('c').find_library('m', required: true) 75 | ] 76 | 77 | executable( 78 | meson.project_name(), 79 | sources, 80 | asresources, 81 | dependencies: dependencies, 82 | vala_args: [meson.project_source_root() + '/src/config.vapi'], 83 | install : true 84 | ) 85 | 86 | subdir('data') 87 | subdir('po') 88 | 89 | gnome.post_install( 90 | glib_compile_schemas: true, 91 | gtk_update_icon_cache: true, 92 | update_desktop_database: true 93 | ) 94 | -------------------------------------------------------------------------------- /meson_options.txt: -------------------------------------------------------------------------------- 1 | option('development', type: 'boolean', value: false, description: 'If this is a development build') 2 | -------------------------------------------------------------------------------- /po/LINGUAS: -------------------------------------------------------------------------------- 1 | es 2 | it 3 | nl 4 | ru 5 | tr 6 | -------------------------------------------------------------------------------- /po/POTFILES: -------------------------------------------------------------------------------- 1 | data/io.github.lainsce.Countdown.desktop.in 2 | data/io.github.lainsce.Countdown.metainfo.xml.in 3 | data/io.github.lainsce.Countdown.gschema.xml 4 | 5 | src/app.vala 6 | src/Widgets/keys.ui 7 | src/Widgets/menu.ui 8 | src/Widgets/mainwindow.ui 9 | src/Widgets/dialog.ui 10 | src/Widgets/eventrowcontent.ui 11 | src/Widgets/mainwindow.ui.vala 12 | src/Widgets/dialog.ui.vala 13 | src/Widgets/eventrowcontent.ui.vala 14 | -------------------------------------------------------------------------------- /po/README.md: -------------------------------------------------------------------------------- 1 | # 🌐 How to Translate Countdown 2 | 3 | ## ✏️ First Things First 4 | 5 | * Fork the repository here on github with the Fork button at the top-right 6 | * Clone this repository by opening the terminal in a folder of your choice and typing `git clone https://github.com//Countdown` 7 | * (Optional) Check [Regenerate translations files](https://github.com/lainsce/Countdown/tree/main/po#-regenerate-translations-files) section if files haven't been recently updated. 8 | 9 | ## 📃 Basics 10 | 11 | * You'll need to know your language's code (ex. en = English). 12 | * Add that code to the LINGUAS file, in a new line, after the last line. 13 | * Translate the .pot file using the PO editor of your choice (I recommend POEdit). 14 | * Save it as .po in this folder. 15 | 16 | ## 📝 Not so Basics 17 | 18 | * Next, in the folder you've cloned this repo in, open a terminal and type: ```git checkout -b "Translation ``` 19 | * Then, type ```git add *``` 20 | * Finally, ```git commit -m "Translated your app for " && git push```, follow the instructions in the terminal if need be, then type your github username and password. 21 | 22 | And that's it! You've successfully translated Countdown for your language! 23 | 24 | ## 🔁 Regenerate translations files 25 | * Initialize the project build by typing `meson _build` (make sure you have [dependencies](https://github.com/lainsce/Countdown#%EF%B8%8F-dependencies) installed!). 26 | * Compile .pot files, type `meson compile -C _build io.github.lainsce.Countdown-pot` 27 | * (Optional) Compile .po files instead replacing `-pot` with `-update-po` in the previous command. 28 | 29 | Note: install `appstream` package in order to generate release strings. 30 | -------------------------------------------------------------------------------- /po/es.po: -------------------------------------------------------------------------------- 1 | # Spanish translations for io.github.lainsce.Countdown package. 2 | # Copyright (C) 2022 THE io.github.lainsce.Countdown'S COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # Óscar Fernández Díaz , 2022. 5 | # 6 | msgid "" 7 | msgstr "" 8 | "Project-Id-Version: io.github.lainsce.Countdown\n" 9 | "Report-Msgid-Bugs-To: \n" 10 | "POT-Creation-Date: 2022-05-08 13:40+0200\n" 11 | "PO-Revision-Date: 2022-05-08 13:50+0200\n" 12 | "Last-Translator: Óscar Fernández Díaz \n" 13 | "Language-Team: none\n" 14 | "Language: es\n" 15 | "MIME-Version: 1.0\n" 16 | "Content-Type: text/plain; charset=UTF-8\n" 17 | "Content-Transfer-Encoding: 8bit\n" 18 | "Plural-Forms: nplurals=2; plural=(n != 1);\n" 19 | "X-Generator: Poedit 3.0.1\n" 20 | 21 | #: data/io.github.lainsce.Countdown.desktop.in:3 22 | msgid "Countdown" 23 | msgstr "Countdown" 24 | 25 | #: data/io.github.lainsce.Countdown.desktop.in:6 26 | msgid "event;countdown;days;" 27 | msgstr "event;countdown;days;evento;cuenta;atrás;días;" 28 | 29 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 30 | msgid "Track events until they happen or since they happened" 31 | msgstr "" 32 | "Hacer un seguimiento de los eventos hasta que se produzcan o desde que se " 33 | "producieron" 34 | 35 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 36 | msgid "Tracking events until they happen, or since they happened, easier." 37 | msgstr "" 38 | "Hacer un seguimiento de los eventos hasta que se produzcan, o desde que se " 39 | "producieron, más fácil." 40 | 41 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 42 | msgid "Release: First Release" 43 | msgstr "Versión: Primera versión" 44 | 45 | #: src/Widgets/keys.ui:12 46 | msgctxt "shortcut window" 47 | msgid "General" 48 | msgstr "General" 49 | 50 | #: src/Widgets/keys.ui:16 51 | msgctxt "shortcut window" 52 | msgid "Quit" 53 | msgstr "Salir" 54 | 55 | #: src/Widgets/keys.ui:22 56 | msgctxt "shortcut window" 57 | msgid "New Event" 58 | msgstr "Evento nuevo" 59 | 60 | #: src/Widgets/menu.ui:7 61 | msgid "Keyboard Shortcuts" 62 | msgstr "Atajos del teclado" 63 | 64 | #: src/Widgets/menu.ui:11 65 | msgid "About Countdown" 66 | msgstr "Acerca de Countdown" 67 | 68 | #: src/Widgets/mainwindow.ui:37 69 | msgid "Upcoming" 70 | msgstr "Próximos" 71 | 72 | #: src/Widgets/mainwindow.ui:48 73 | msgid "Past" 74 | msgstr "Pasados" 75 | 76 | #: src/Widgets/mainwindow.ui:64 77 | msgid "• events" 78 | msgstr "• eventos" 79 | 80 | #: src/Widgets/mainwindow.ui:168 81 | msgid "Add Event…" 82 | msgstr "Añadir evento…" 83 | 84 | #: src/Widgets/dialog.ui:24 85 | msgid "Add Event" 86 | msgstr "Añadir evento" 87 | 88 | #: src/Widgets/dialog.ui:34 89 | msgid "New Event" 90 | msgstr "Evento nuevo" 91 | 92 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 93 | #: src/Widgets/eventrowcontent.ui.vala:79 94 | msgid "Cancel" 95 | msgstr "Cancelar" 96 | 97 | #: src/Widgets/dialog.ui:57 98 | msgid "Event Name" 99 | msgstr "Nombre del evento" 100 | 101 | #: src/Widgets/dialog.ui:68 102 | msgid "Event Date" 103 | msgstr "Fecha del evento" 104 | 105 | #: src/Widgets/eventrowcontent.ui:65 106 | msgid "days" 107 | msgstr "días" 108 | 109 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 110 | #: src/Widgets/mainwindow.ui.vala:120 111 | msgid "events" 112 | msgstr "eventos" 113 | 114 | #: src/Widgets/mainwindow.ui.vala:154 115 | msgid "Track events until they happen or since they happened." 116 | msgstr "" 117 | "Hacer un seguimiento de los eventos hasta que se produzcan o desde que se " 118 | "producieron." 119 | 120 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 121 | #: src/Widgets/mainwindow.ui.vala:161 122 | msgid "translator-credits" 123 | msgstr "Óscar Fernández Díaz " 124 | 125 | #: src/Widgets/eventrowcontent.ui.vala:41 126 | msgid "Delete Past Event?" 127 | msgstr "¿Eliminar evento pasado?" 128 | 129 | #: src/Widgets/eventrowcontent.ui.vala:42 130 | msgid "" 131 | "Deleting this past event will not save it in the past log and will stop " 132 | "tracking the days since the event happened." 133 | msgstr "" 134 | "La eliminación de este evento pasado no lo guardará en el registro del " 135 | "pasado y dejará de registrar los días desde que se produjo el evento." 136 | 137 | #: src/Widgets/eventrowcontent.ui.vala:45 138 | #: src/Widgets/eventrowcontent.ui.vala:80 139 | msgid "Delete" 140 | msgstr "Eliminar" 141 | 142 | #: src/Widgets/eventrowcontent.ui.vala:76 143 | msgid "Delete Upcoming Event?" 144 | msgstr "¿Eliminar evento próximo?" 145 | 146 | #: src/Widgets/eventrowcontent.ui.vala:77 147 | msgid "" 148 | "Deleting this upcoming event will not save it in the upcoming log and will " 149 | "stop tracking the days until the event happens." 150 | msgstr "" 151 | "La eliminación de este evento próximo no lo guardará en el registro " 152 | "próximo y dejará de registrar los días hasta que se produzca el evento." 153 | -------------------------------------------------------------------------------- /po/io.github.lainsce.Countdown.pot: -------------------------------------------------------------------------------- 1 | # SOME DESCRIPTIVE TITLE. 2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # FIRST AUTHOR , YEAR. 5 | # 6 | #, fuzzy 7 | msgid "" 8 | msgstr "" 9 | "Project-Id-Version: io.github.lainsce.Countdown\n" 10 | "Report-Msgid-Bugs-To: \n" 11 | "POT-Creation-Date: 2022-06-15 22:50+0200\n" 12 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" 13 | "Last-Translator: FULL NAME \n" 14 | "Language-Team: LANGUAGE \n" 15 | "Language: \n" 16 | "MIME-Version: 1.0\n" 17 | "Content-Type: text/plain; charset=UTF-8\n" 18 | "Content-Transfer-Encoding: 8bit\n" 19 | 20 | #: data/io.github.lainsce.Countdown.desktop.in:3 21 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:6 22 | msgid "Countdown" 23 | msgstr "" 24 | 25 | #: data/io.github.lainsce.Countdown.desktop.in:6 26 | msgid "event;countdown;days;" 27 | msgstr "" 28 | 29 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 30 | msgid "Track events until they happen or since they happened" 31 | msgstr "" 32 | 33 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 34 | msgid "Tracking events until they happen, or since they happened, easier." 35 | msgstr "" 36 | 37 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:19 38 | msgid "Lains" 39 | msgstr "" 40 | 41 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 42 | msgid "Release: First Release" 43 | msgstr "" 44 | 45 | #: src/Widgets/keys.ui:12 46 | msgctxt "shortcut window" 47 | msgid "General" 48 | msgstr "" 49 | 50 | #: src/Widgets/keys.ui:16 51 | msgctxt "shortcut window" 52 | msgid "Quit" 53 | msgstr "" 54 | 55 | #: src/Widgets/keys.ui:22 56 | msgctxt "shortcut window" 57 | msgid "New Event" 58 | msgstr "" 59 | 60 | #: src/Widgets/menu.ui:7 61 | msgid "Keyboard Shortcuts" 62 | msgstr "" 63 | 64 | #: src/Widgets/menu.ui:11 65 | msgid "About Countdown" 66 | msgstr "" 67 | 68 | #: src/Widgets/mainwindow.ui:37 69 | msgid "Upcoming" 70 | msgstr "" 71 | 72 | #: src/Widgets/mainwindow.ui:48 73 | msgid "Past" 74 | msgstr "" 75 | 76 | #: src/Widgets/mainwindow.ui:64 77 | msgid "• events" 78 | msgstr "" 79 | 80 | #: src/Widgets/mainwindow.ui:168 81 | msgid "Add Event…" 82 | msgstr "" 83 | 84 | #: src/Widgets/dialog.ui:24 85 | msgid "Add Event" 86 | msgstr "" 87 | 88 | #: src/Widgets/dialog.ui:34 89 | msgid "New Event" 90 | msgstr "" 91 | 92 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 93 | #: src/Widgets/eventrowcontent.ui.vala:79 94 | msgid "Cancel" 95 | msgstr "" 96 | 97 | #: src/Widgets/dialog.ui:57 98 | msgid "Event Name" 99 | msgstr "" 100 | 101 | #: src/Widgets/dialog.ui:68 102 | msgid "Event Date" 103 | msgstr "" 104 | 105 | #: src/Widgets/eventrowcontent.ui:65 106 | msgid "days" 107 | msgstr "" 108 | 109 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 110 | #: src/Widgets/mainwindow.ui.vala:120 111 | msgid "events" 112 | msgstr "" 113 | 114 | #: src/Widgets/mainwindow.ui.vala:154 115 | msgid "Track events until they happen or since they happened." 116 | msgstr "" 117 | 118 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 119 | #: src/Widgets/mainwindow.ui.vala:161 120 | msgid "translator-credits" 121 | msgstr "" 122 | 123 | #: src/Widgets/eventrowcontent.ui.vala:41 124 | msgid "Delete Past Event?" 125 | msgstr "" 126 | 127 | #: src/Widgets/eventrowcontent.ui.vala:42 128 | msgid "" 129 | "Deleting this past event will not save it in the past log and will stop " 130 | "tracking the days since the event happened." 131 | msgstr "" 132 | 133 | #: src/Widgets/eventrowcontent.ui.vala:45 134 | #: src/Widgets/eventrowcontent.ui.vala:80 135 | msgid "Delete" 136 | msgstr "" 137 | 138 | #: src/Widgets/eventrowcontent.ui.vala:76 139 | msgid "Delete Upcoming Event?" 140 | msgstr "" 141 | 142 | #: src/Widgets/eventrowcontent.ui.vala:77 143 | msgid "" 144 | "Deleting this upcoming event will not save it in the upcoming log and will " 145 | "stop tracking the days until the event happens." 146 | msgstr "" 147 | -------------------------------------------------------------------------------- /po/it.po: -------------------------------------------------------------------------------- 1 | # ITALIAN TRANSLATION FOR COUNTDOWN. 2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # ALBANO BATTISTELLA , 2022. 5 | # 6 | msgid "" 7 | msgstr "" 8 | "Project-Id-Version: io.github.lainsce.Countdown\n" 9 | "Report-Msgid-Bugs-To: \n" 10 | "POT-Creation-Date: 2022-06-15 22:50+0200\n" 11 | "PO-Revision-Date: 2022-06-25 16:32+0100\n" 12 | "Last-Translator: Albano Battistella \n" 13 | "Language-Team: Italian \n" 14 | "Language: it\n" 15 | "MIME-Version: 1.0\n" 16 | "Content-Type: text/plain; charset=UTF-8\n" 17 | "Content-Transfer-Encoding: 8bit\n" 18 | 19 | #: data/io.github.lainsce.Countdown.desktop.in:3 20 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:6 21 | msgid "Countdown" 22 | msgstr "Countdown" 23 | 24 | #: data/io.github.lainsce.Countdown.desktop.in:6 25 | msgid "event;countdown;days;" 26 | msgstr "evento;conto alla rovescia;giorni;" 27 | 28 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 29 | msgid "Track events until they happen or since they happened" 30 | msgstr "Tieni traccia degli eventi finché non si verificano o da quando sono accaduti" 31 | 32 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 33 | msgid "Tracking events until they happen, or since they happened, easier." 34 | msgstr "Tracciare gli eventi fino a quando non si verificano, o da quando sono accaduti,facilmente." 35 | 36 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:19 37 | msgid "Lains" 38 | msgstr "Lains" 39 | 40 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 41 | msgid "Release: First Release" 42 | msgstr "Versione: Primo rilascio" 43 | 44 | #: src/Widgets/keys.ui:12 45 | msgctxt "shortcut window" 46 | msgid "General" 47 | msgstr "Generale" 48 | 49 | #: src/Widgets/keys.ui:16 50 | msgctxt "shortcut window" 51 | msgid "Quit" 52 | msgstr "Esci" 53 | 54 | #: src/Widgets/keys.ui:22 55 | msgctxt "shortcut window" 56 | msgid "New Event" 57 | msgstr "Nuovo evento" 58 | 59 | #: src/Widgets/menu.ui:7 60 | msgid "Keyboard Shortcuts" 61 | msgstr "Scorciatoie da tastiera" 62 | 63 | #: src/Widgets/menu.ui:11 64 | msgid "About Countdown" 65 | msgstr "Informazioni su Countdown" 66 | 67 | #: src/Widgets/mainwindow.ui:37 68 | msgid "Upcoming" 69 | msgstr "Prossimamente" 70 | 71 | #: src/Widgets/mainwindow.ui:48 72 | msgid "Past" 73 | msgstr "Incolla" 74 | 75 | #: src/Widgets/mainwindow.ui:64 76 | msgid "• events" 77 | msgstr "• eventi" 78 | 79 | #: src/Widgets/mainwindow.ui:168 80 | msgid "Add Event…" 81 | msgstr "Aggiungi evento…" 82 | 83 | #: src/Widgets/dialog.ui:24 84 | msgid "Add Event" 85 | msgstr "Aggiungi evento" 86 | 87 | #: src/Widgets/dialog.ui:34 88 | msgid "New Event" 89 | msgstr "Nuovo evento" 90 | 91 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 92 | #: src/Widgets/eventrowcontent.ui.vala:79 93 | msgid "Cancel" 94 | msgstr "Cancella" 95 | 96 | #: src/Widgets/dialog.ui:57 97 | msgid "Event Name" 98 | msgstr "Nome evento" 99 | 100 | #: src/Widgets/dialog.ui:68 101 | msgid "Event Date" 102 | msgstr "Data evento" 103 | 104 | #: src/Widgets/eventrowcontent.ui:65 105 | msgid "days" 106 | msgstr "giorni" 107 | 108 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 109 | #: src/Widgets/mainwindow.ui.vala:120 110 | msgid "events" 111 | msgstr "eventi" 112 | 113 | #: src/Widgets/mainwindow.ui.vala:154 114 | msgid "Track events until they happen or since they happened." 115 | msgstr "Tieni traccia degli eventi finché non si verificano o da quando sono accaduti." 116 | 117 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 118 | #: src/Widgets/mainwindow.ui.vala:161 119 | msgid "translator-credits" 120 | msgstr "Albano Battistella" 121 | 122 | #: src/Widgets/eventrowcontent.ui.vala:41 123 | msgid "Delete Past Event?" 124 | msgstr "Eliminare l'evento passato?" 125 | 126 | #: src/Widgets/eventrowcontent.ui.vala:42 127 | msgid "" 128 | "Deleting this past event will not save it in the past log and will stop " 129 | "tracking the days since the event happened." 130 | msgstr "" 131 | "L'eliminazione di questo evento passato non lo salverà nel file log e verrà interrotto " 132 | "il tracking dei giorni trascorsi dall'evento." 133 | 134 | #: src/Widgets/eventrowcontent.ui.vala:45 135 | #: src/Widgets/eventrowcontent.ui.vala:80 136 | msgid "Delete" 137 | msgstr "Cancella" 138 | 139 | #: src/Widgets/eventrowcontent.ui.vala:76 140 | msgid "Delete Upcoming Event?" 141 | msgstr "Eliminare il prossimo evento?" 142 | 143 | #: src/Widgets/eventrowcontent.ui.vala:77 144 | msgid "" 145 | "Deleting this upcoming event will not save it in the upcoming log and will " 146 | "stop tracking the days until the event happens." 147 | msgstr "" 148 | "L'eliminazione di questo evento imminente non lo salverà nel file log e verrà interrotto " 149 | "il tracking dei giorni prima che si verifichi l'evento." 150 | -------------------------------------------------------------------------------- /po/meson.build: -------------------------------------------------------------------------------- 1 | i18n.gettext(app_id, preset: 'glib') 2 | -------------------------------------------------------------------------------- /po/nl.po: -------------------------------------------------------------------------------- 1 | # SOME DESCRIPTIVE TITLE. 2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # FIRST AUTHOR , YEAR. 5 | # 6 | msgid "" 7 | msgstr "" 8 | "Project-Id-Version: io.github.lainsce.Countdown\n" 9 | "Report-Msgid-Bugs-To: \n" 10 | "POT-Creation-Date: 2022-06-15 22:50+0200\n" 11 | "PO-Revision-Date: 2022-06-15 23:09+0200\n" 12 | "Last-Translator: Philip Goto \n" 13 | "Language-Team: \n" 14 | "Language: nl\n" 15 | "MIME-Version: 1.0\n" 16 | "Content-Type: text/plain; charset=UTF-8\n" 17 | "Content-Transfer-Encoding: 8bit\n" 18 | "Plural-Forms: nplurals=2; plural=(n != 1);\n" 19 | "X-Generator: Poedit 3.1\n" 20 | 21 | #: data/io.github.lainsce.Countdown.desktop.in:3 22 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:6 23 | msgid "Countdown" 24 | msgstr "Countdown" 25 | 26 | #: data/io.github.lainsce.Countdown.desktop.in:6 27 | msgid "event;countdown;days;" 28 | msgstr "event;gebeurtenis;countdown;aftel;days;dagen;" 29 | 30 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 31 | msgid "Track events until they happen or since they happened" 32 | msgstr "Volg gebeurtenissen tot ze gebeuren of sinds ze gebeurd zijn" 33 | 34 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 35 | msgid "Tracking events until they happen, or since they happened, easier." 36 | msgstr "" 37 | "Gebeurtenissen makkelijker volgen tot ze gebeuren, of sinds ze gebeurd " 38 | "zijn." 39 | 40 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:19 41 | msgid "Lains" 42 | msgstr "Lains" 43 | 44 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 45 | msgid "Release: First Release" 46 | msgstr "Uitgave: Eerste uitgave" 47 | 48 | #: src/Widgets/keys.ui:12 49 | msgctxt "shortcut window" 50 | msgid "General" 51 | msgstr "Algemeen" 52 | 53 | #: src/Widgets/keys.ui:16 54 | msgctxt "shortcut window" 55 | msgid "Quit" 56 | msgstr "Sluiten" 57 | 58 | #: src/Widgets/keys.ui:22 59 | msgctxt "shortcut window" 60 | msgid "New Event" 61 | msgstr "Nieuwe gebeurtenis" 62 | 63 | #: src/Widgets/menu.ui:7 64 | msgid "Keyboard Shortcuts" 65 | msgstr "Sneltoetsen" 66 | 67 | #: src/Widgets/menu.ui:11 68 | msgid "About Countdown" 69 | msgstr "Over Countdown" 70 | 71 | #: src/Widgets/mainwindow.ui:37 72 | msgid "Upcoming" 73 | msgstr "Aankomend" 74 | 75 | #: src/Widgets/mainwindow.ui:48 76 | msgid "Past" 77 | msgstr "Afgelopen" 78 | 79 | #: src/Widgets/mainwindow.ui:64 80 | msgid "• events" 81 | msgstr "• gebeurtenissen" 82 | 83 | #: src/Widgets/mainwindow.ui:168 84 | msgid "Add Event…" 85 | msgstr "Gebeurtenis toevoegen…" 86 | 87 | #: src/Widgets/dialog.ui:24 88 | msgid "Add Event" 89 | msgstr "Gebeurtenis toevoegen" 90 | 91 | #: src/Widgets/dialog.ui:34 92 | msgid "New Event" 93 | msgstr "Nieuwe gebeurtenis" 94 | 95 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 96 | #: src/Widgets/eventrowcontent.ui.vala:79 97 | msgid "Cancel" 98 | msgstr "Annuleren" 99 | 100 | #: src/Widgets/dialog.ui:57 101 | msgid "Event Name" 102 | msgstr "Naam van gebeurtenis" 103 | 104 | #: src/Widgets/dialog.ui:68 105 | msgid "Event Date" 106 | msgstr "Datum van gebeurtenis" 107 | 108 | #: src/Widgets/eventrowcontent.ui:65 109 | msgid "days" 110 | msgstr "dagen" 111 | 112 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 113 | #: src/Widgets/mainwindow.ui.vala:120 114 | msgid "events" 115 | msgstr "gebeurtenissen" 116 | 117 | #: src/Widgets/mainwindow.ui.vala:154 118 | msgid "Track events until they happen or since they happened." 119 | msgstr "Volg gebeurtenissen tot ze gebeuren, of sinds ze gebeurd zijn." 120 | 121 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 122 | #: src/Widgets/mainwindow.ui.vala:161 123 | msgid "translator-credits" 124 | msgstr "Philip Goto" 125 | 126 | #: src/Widgets/eventrowcontent.ui.vala:41 127 | msgid "Delete Past Event?" 128 | msgstr "Afgelopen gebeurtenis verwijderen?" 129 | 130 | #: src/Widgets/eventrowcontent.ui.vala:42 131 | msgid "" 132 | "Deleting this past event will not save it in the past log and will stop " 133 | "tracking the days since the event happened." 134 | msgstr "" 135 | "Door deze afgelopen gebeurtenis te verwijderen zal deze niet in de " 136 | "logboeken worden opgeslagen, en zal het aantal dagen sinds deze " 137 | "gebeurtenis niet meer worden bijgehouden." 138 | 139 | #: src/Widgets/eventrowcontent.ui.vala:45 140 | #: src/Widgets/eventrowcontent.ui.vala:80 141 | msgid "Delete" 142 | msgstr "Verwijderen" 143 | 144 | #: src/Widgets/eventrowcontent.ui.vala:76 145 | msgid "Delete Upcoming Event?" 146 | msgstr "Aankomende gebeurtenis verwijderen?" 147 | 148 | #: src/Widgets/eventrowcontent.ui.vala:77 149 | msgid "" 150 | "Deleting this upcoming event will not save it in the upcoming log and will " 151 | "stop tracking the days until the event happens." 152 | msgstr "" 153 | "Door deze aankomende gebeurtenis te verwijderen zal deze niet in de " 154 | "logboeken worden opgeslagen, en zal het aantal dagen tot deze gebeurtenis " 155 | "niet meer worden bijgehouden." 156 | -------------------------------------------------------------------------------- /po/ru.po: -------------------------------------------------------------------------------- 1 | # SOME DESCRIPTIVE TITLE. 2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # David Lapshin , 2022. 5 | # 6 | #, fuzzy 7 | msgid "" 8 | msgstr "" 9 | "Project-Id-Version: io.github.lainsce.Countdown\n" 10 | "Report-Msgid-Bugs-To: \n" 11 | "POT-Creation-Date: 2022-06-15 22:50+0200\n" 12 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" 13 | "Last-Translator: David Lapshin \n" 14 | "Language-Team: LANGUAGE \n" 15 | "Language: \n" 16 | "MIME-Version: 1.0\n" 17 | "Content-Type: text/plain; charset=UTF-8\n" 18 | "Content-Transfer-Encoding: 8bit\n" 19 | 20 | #: data/io.github.lainsce.Countdown.desktop.in:3 21 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:6 22 | msgid "Countdown" 23 | msgstr "Countdown" 24 | 25 | #: data/io.github.lainsce.Countdown.desktop.in:6 26 | msgid "event;countdown;days;" 27 | msgstr "событие;обратный отсчет;дни;" 28 | 29 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 30 | msgid "Track events until they happen or since they happened" 31 | msgstr "Отслеживайте событий до того, как они произошли, или после того, как они произошли" 32 | 33 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 34 | msgid "Tracking events until they happen, or since they happened, easier." 35 | msgstr "Отслеживание событий до того, как они произошли, или после того, как они произошли, проще." 36 | 37 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:19 38 | msgid "Lains" 39 | msgstr "Lains" 40 | 41 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 42 | msgid "Release: First Release" 43 | msgstr "Выпуск: Первый выпуск" 44 | 45 | #: src/Widgets/keys.ui:12 46 | msgctxt "shortcut window" 47 | msgid "General" 48 | msgstr "Основные" 49 | 50 | #: src/Widgets/keys.ui:16 51 | msgctxt "shortcut window" 52 | msgid "Quit" 53 | msgstr "Выйти" 54 | 55 | #: src/Widgets/keys.ui:22 56 | msgctxt "shortcut window" 57 | msgid "New Event" 58 | msgstr "Новое событие" 59 | 60 | #: src/Widgets/menu.ui:7 61 | msgid "Keyboard Shortcuts" 62 | msgstr "Комбинации клавиш" 63 | 64 | #: src/Widgets/menu.ui:11 65 | msgid "About Countdown" 66 | msgstr "О Countdown" 67 | 68 | #: src/Widgets/mainwindow.ui:37 69 | msgid "Upcoming" 70 | msgstr "Предстоящие" 71 | 72 | #: src/Widgets/mainwindow.ui:48 73 | msgid "Past" 74 | msgstr "Прошедшие" 75 | 76 | #: src/Widgets/mainwindow.ui:64 77 | msgid "• events" 78 | msgstr "• событий" 79 | 80 | #: src/Widgets/mainwindow.ui:168 81 | msgid "Add Event…" 82 | msgstr "Добавить событие…" 83 | 84 | #: src/Widgets/dialog.ui:24 85 | msgid "Add Event" 86 | msgstr "Добавить событие" 87 | 88 | #: src/Widgets/dialog.ui:34 89 | msgid "New Event" 90 | msgstr "Новое событие" 91 | 92 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 93 | #: src/Widgets/eventrowcontent.ui.vala:79 94 | msgid "Cancel" 95 | msgstr "Отмена" 96 | 97 | #: src/Widgets/dialog.ui:57 98 | msgid "Event Name" 99 | msgstr "Название события" 100 | 101 | #: src/Widgets/dialog.ui:68 102 | msgid "Event Date" 103 | msgstr "Дата события" 104 | 105 | #: src/Widgets/eventrowcontent.ui:65 106 | msgid "days" 107 | msgstr "дней" 108 | 109 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 110 | #: src/Widgets/mainwindow.ui.vala:120 111 | msgid "events" 112 | msgstr "событий" 113 | 114 | #: src/Widgets/mainwindow.ui.vala:154 115 | msgid "Track events until they happen or since they happened." 116 | msgstr "Отслеживайте событий до того, как они произошли, или после того, как они произошли" 117 | 118 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 119 | #: src/Widgets/mainwindow.ui.vala:161 120 | msgid "translator-credits" 121 | msgstr "Давид Лапшин " 122 | 123 | #: src/Widgets/eventrowcontent.ui.vala:41 124 | msgid "Delete Past Event?" 125 | msgstr "Удалить прошедшее событие?" 126 | 127 | #: src/Widgets/eventrowcontent.ui.vala:42 128 | msgid "" 129 | "Deleting this past event will not save it in the past log and will stop " 130 | "tracking the days since the event happened." 131 | msgstr "" 132 | "Удаление этого прошедшего события не сохранит его в журнале прошлых событий и остановит " 133 | "отслеживание дней, прошедших с момента события." 134 | 135 | #: src/Widgets/eventrowcontent.ui.vala:45 136 | #: src/Widgets/eventrowcontent.ui.vala:80 137 | msgid "Delete" 138 | msgstr "Удалить" 139 | 140 | #: src/Widgets/eventrowcontent.ui.vala:76 141 | msgid "Delete Upcoming Event?" 142 | msgstr "Удалить предстоящие событие?" 143 | 144 | #: src/Widgets/eventrowcontent.ui.vala:77 145 | msgid "" 146 | "Deleting this upcoming event will not save it in the upcoming log and will " 147 | "stop tracking the days until the event happens." 148 | msgstr "" 149 | "Удаление этого предстоящего события не сохранит его в журнале предстоящих событий и остановит " 150 | "отслеживание дней, пока событие не произойдет." 151 | -------------------------------------------------------------------------------- /po/tr.po: -------------------------------------------------------------------------------- 1 | # Turkish translation of io.github.lainsce.Countdown. 2 | # Copyright (C) 2023 io.github.lainsce.Countdown's COPYRIGHT HOLDER 3 | # This file is distributed under the same license as the io.github.lainsce.Countdown package. 4 | # 5 | # Sabri Ünal , 2023. 6 | # 7 | msgid "" 8 | msgstr "" 9 | "Project-Id-Version: io.github.lainsce.Countdown\n" 10 | "Report-Msgid-Bugs-To: \n" 11 | "POT-Creation-Date: 2022-06-15 22:50+0200\n" 12 | "PO-Revision-Date: 2023-06-08 09:33+0300\n" 13 | "Last-Translator: Sabri Ünal \n" 14 | "Language-Team: Turkish \n" 15 | "Language: tr\n" 16 | "MIME-Version: 1.0\n" 17 | "Content-Type: text/plain; charset=UTF-8\n" 18 | "Content-Transfer-Encoding: 8bit\n" 19 | "Plural-Forms: nplurals=1; plural=0;\n" 20 | "X-Generator: Poedit 3.2.2\n" 21 | 22 | #: data/io.github.lainsce.Countdown.desktop.in:3 23 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:6 24 | msgid "Countdown" 25 | msgstr "Geri Sayım" 26 | 27 | #: data/io.github.lainsce.Countdown.desktop.in:6 28 | msgid "event;countdown;days;" 29 | msgstr "event;countdown;days;olay;geri sayım;gerisayım;gün;" 30 | 31 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:7 32 | msgid "Track events until they happen or since they happened" 33 | msgstr "Olayları gerçekleşene kadar veya sonrasında takip edin" 34 | 35 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:9 36 | msgid "Tracking events until they happen, or since they happened, easier." 37 | msgstr "Olayları gerçekleşene kadar veya sonrasında kolayca takip edin." 38 | 39 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:19 40 | msgid "Lains" 41 | msgstr "Lains" 42 | 43 | #: data/io.github.lainsce.Countdown.metainfo.xml.in:45 44 | msgid "Release: First Release" 45 | msgstr "Sürüm: İlk sürüm" 46 | 47 | #: src/Widgets/keys.ui:12 48 | msgctxt "shortcut window" 49 | msgid "General" 50 | msgstr "Genel" 51 | 52 | #: src/Widgets/keys.ui:16 53 | msgctxt "shortcut window" 54 | msgid "Quit" 55 | msgstr "Çık" 56 | 57 | #: src/Widgets/keys.ui:22 58 | msgctxt "shortcut window" 59 | msgid "New Event" 60 | msgstr "Yeni Olay" 61 | 62 | #: src/Widgets/menu.ui:7 63 | msgid "Keyboard Shortcuts" 64 | msgstr "Klavye Kısayolları" 65 | 66 | #: src/Widgets/menu.ui:11 67 | msgid "About Countdown" 68 | msgstr "Geri Sayım Hakkında" 69 | 70 | #: src/Widgets/mainwindow.ui:37 71 | msgid "Upcoming" 72 | msgstr "Yaklaşan" 73 | 74 | #: src/Widgets/mainwindow.ui:48 75 | msgid "Past" 76 | msgstr "Geçmiş" 77 | 78 | #: src/Widgets/mainwindow.ui:64 79 | msgid "• events" 80 | msgstr "• olaylar" 81 | 82 | #: src/Widgets/mainwindow.ui:168 83 | msgid "Add Event…" 84 | msgstr "Olay Ekle…" 85 | 86 | #: src/Widgets/dialog.ui:24 87 | msgid "Add Event" 88 | msgstr "Olay Ekle" 89 | 90 | #: src/Widgets/dialog.ui:34 91 | msgid "New Event" 92 | msgstr "Yeni Olay" 93 | 94 | #: src/Widgets/dialog.ui:43 src/Widgets/eventrowcontent.ui.vala:44 95 | #: src/Widgets/eventrowcontent.ui.vala:79 96 | msgid "Cancel" 97 | msgstr "İptal" 98 | 99 | #: src/Widgets/dialog.ui:57 100 | msgid "Event Name" 101 | msgstr "Olay Adı" 102 | 103 | #: src/Widgets/dialog.ui:68 104 | msgid "Event Date" 105 | msgstr "Olay Tarihi" 106 | 107 | #: src/Widgets/eventrowcontent.ui:65 108 | msgid "days" 109 | msgstr "gün" 110 | 111 | #: src/Widgets/mainwindow.ui.vala:80 src/Widgets/mainwindow.ui.vala:113 112 | #: src/Widgets/mainwindow.ui.vala:120 113 | msgid "events" 114 | msgstr "olay" 115 | 116 | #: src/Widgets/mainwindow.ui.vala:154 117 | msgid "Track events until they happen or since they happened." 118 | msgstr "Olayları gerçekleşene kadar veya sonrasında takip edin." 119 | 120 | #. TRANSLATORS: 'Name ' or 'Name https://website.example' 121 | #: src/Widgets/mainwindow.ui.vala:161 122 | msgid "translator-credits" 123 | msgstr "Sabri Ünal " 124 | 125 | #: src/Widgets/eventrowcontent.ui.vala:41 126 | msgid "Delete Past Event?" 127 | msgstr "Geçmiş Olaylar Silinsin Mi?" 128 | 129 | #: src/Widgets/eventrowcontent.ui.vala:42 130 | msgid "" 131 | "Deleting this past event will not save it in the past log and will stop " 132 | "tracking the days since the event happened." 133 | msgstr "" 134 | "Bu geçmiş olayın silinmesi, onu geçmiş günlüğe kaydetmez ve olayın " 135 | "gerçekleşmesinden bu yana geçen günleri izlemeyi durdurur." 136 | 137 | #: src/Widgets/eventrowcontent.ui.vala:45 138 | #: src/Widgets/eventrowcontent.ui.vala:80 139 | msgid "Delete" 140 | msgstr "Sil" 141 | 142 | #: src/Widgets/eventrowcontent.ui.vala:76 143 | msgid "Delete Upcoming Event?" 144 | msgstr "Yaklayan Olaylar Silinsin Mi?" 145 | 146 | #: src/Widgets/eventrowcontent.ui.vala:77 147 | msgid "" 148 | "Deleting this upcoming event will not save it in the upcoming log and will " 149 | "stop tracking the days until the event happens." 150 | msgstr "" 151 | "Bu yaklaşan olayın silinmesi, olayı yaklaşan günlüğüne kaydetmeyecek ve " 152 | "olay gerçekleşene kadar geçen günleri izlemeyi durduracaktır." 153 | -------------------------------------------------------------------------------- /src/Models/event.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | namespace Countdown { 20 | public class Event : Object, Json.Serializable { 21 | public string id { get; set; default = Uuid.string_random (); } 22 | public string title { get; set; } 23 | public DateTime date { get; set; } 24 | public bool passed { get; set; } 25 | 26 | public static Event from_json (Json.Node node) requires (node.get_node_type () == OBJECT) { 27 | return (Event) Json.gobject_deserialize (typeof (Event), node); 28 | } 29 | 30 | public static List list_from_json (Json.Node node) requires (node.get_node_type () == ARRAY) { 31 | var result = new List (); 32 | 33 | var json_array = node.get_array (); 34 | json_array.foreach_element ((_, __, element_node) => { 35 | result.append (Event.from_json (element_node)); 36 | }); 37 | 38 | return (owned) result; 39 | } 40 | 41 | public Json.Node to_json () { 42 | return Json.gobject_serialize (this); 43 | } 44 | 45 | bool deserialize_property (string property_name, out Value @value, ParamSpec pspec, Json.Node property_node) { 46 | if (property_name == "date") { 47 | if (property_node.get_string ().length == 0) return false; 48 | 49 | string[] d = property_node.get_string ().split ("-", 2); 50 | var date = new DateTime.from_iso8601 ("%04d-%s".printf (int.parse (d[0]), d[1]), null); 51 | if (date == null) return false; 52 | 53 | var vout = Value (typeof (DateTime)); 54 | vout.set_boxed (date); 55 | @value = vout; 56 | 57 | return true; 58 | } else { 59 | return default_deserialize_property (property_name, out @value, pspec, property_node); 60 | } 61 | } 62 | 63 | Json.Node serialize_property (string property_name, Value @value, ParamSpec pspec) { 64 | if (property_name == "date") { 65 | var dt = @value as DateTime; 66 | var node = new Json.Node (VALUE); 67 | var date = ""; 68 | if (dt != null) date = dt.to_string(); 69 | node.set_string (date); 70 | 71 | return node; 72 | } else { 73 | return default_serialize_property (property_name, @value, pspec); 74 | } 75 | } 76 | } 77 | } 78 | -------------------------------------------------------------------------------- /src/Repositories/eventrepository.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public class Countdown.EventRepository : Object { 20 | const string FILENAME = "saved_events.json"; 21 | 22 | Queue insert_queue = new Queue (); 23 | public Queue update_queue = new Queue (); 24 | Queue delete_queue = new Queue (); 25 | 26 | public async List get_events () { 27 | try { 28 | var contents = yield FileUtils.read_text_file (FILENAME); 29 | 30 | if (contents == null) 31 | return new List (); 32 | 33 | var json = Json.from_string (contents); 34 | 35 | if (json.get_node_type () != ARRAY) 36 | return new List (); 37 | 38 | return Event.list_from_json (json); 39 | } catch (Error err) { 40 | critical ("Error: %s", err.message); 41 | return new List (); 42 | } 43 | } 44 | 45 | public void insert_event (Event event) { 46 | insert_queue.push_tail (event); 47 | } 48 | 49 | public void update_event (Event event) { 50 | update_queue.push_tail (event); 51 | } 52 | 53 | public void delete_event (string id) { 54 | delete_queue.push_tail (id); 55 | } 56 | 57 | public async bool save () { 58 | var events = yield get_events (); 59 | 60 | Event? event = null; 61 | while ((event = update_queue.pop_head ()) != null) { 62 | var current_event = search_event_by_id (events, event.id); 63 | 64 | if (current_event == null) { 65 | insert_queue.push_tail (event); 66 | continue; 67 | } 68 | current_event.title = event.title; 69 | current_event.date = event.date; 70 | } 71 | 72 | string? event_id = null; 73 | while ((event_id = delete_queue.pop_head ()) != null) { 74 | event = search_event_by_id (events, event_id); 75 | 76 | if (event == null) 77 | continue; 78 | 79 | events.remove (event); 80 | } 81 | 82 | event = null; 83 | while ((event = insert_queue.pop_head ()) != null) 84 | events.append (event); 85 | 86 | var json_array = new Json.Array (); 87 | foreach (var item in events) 88 | json_array.add_element (item.to_json ()); 89 | 90 | var node = new Json.Node (ARRAY); 91 | node.set_array (json_array); 92 | 93 | var str = Json.to_string (node, false); 94 | 95 | try { 96 | return yield FileUtils.create_text_file (FILENAME, str); 97 | } catch (Error err) { 98 | critical ("Error: %s", err.message); 99 | return false; 100 | } 101 | } 102 | 103 | public inline Event? search_event_by_id (List events, string id) { 104 | unowned var link = events.search (id, (event, id) => strcmp (event.id, id)); 105 | return link?.data; 106 | } 107 | } 108 | -------------------------------------------------------------------------------- /src/Repositories/pasteventrepository.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public class Countdown.PastEventRepository : Object { 20 | const string FILENAME = "saved_events_past.json"; 21 | 22 | Queue insert_queue = new Queue (); 23 | public Queue update_queue = new Queue (); 24 | Queue delete_queue = new Queue (); 25 | 26 | public async List get_events () { 27 | try { 28 | var contents = yield FileUtils.read_text_file (FILENAME); 29 | 30 | if (contents == null) 31 | return new List (); 32 | 33 | var json = Json.from_string (contents); 34 | 35 | if (json.get_node_type () != ARRAY) 36 | return new List (); 37 | 38 | return Event.list_from_json (json); 39 | } catch (Error err) { 40 | critical ("Error: %s", err.message); 41 | return new List (); 42 | } 43 | } 44 | 45 | public void insert_event (Event event) { 46 | insert_queue.push_tail (event); 47 | } 48 | 49 | public void update_event (Event event) { 50 | update_queue.push_tail (event); 51 | } 52 | 53 | public void delete_event (string id) { 54 | delete_queue.push_tail (id); 55 | } 56 | 57 | public async bool save () { 58 | var events = yield get_events (); 59 | 60 | Event? event = null; 61 | while ((event = update_queue.pop_head ()) != null) { 62 | var current_event = search_event_by_id (events, event.id); 63 | 64 | if (current_event == null) { 65 | insert_queue.push_tail (event); 66 | continue; 67 | } 68 | current_event.title = event.title; 69 | current_event.date = event.date; 70 | } 71 | 72 | string? event_id = null; 73 | while ((event_id = delete_queue.pop_head ()) != null) { 74 | event = search_event_by_id (events, event_id); 75 | 76 | if (event == null) 77 | continue; 78 | 79 | events.remove (event); 80 | } 81 | 82 | event = null; 83 | while ((event = insert_queue.pop_head ()) != null) 84 | events.append (event); 85 | 86 | var json_array = new Json.Array (); 87 | foreach (var item in events) 88 | json_array.add_element (item.to_json ()); 89 | 90 | var node = new Json.Node (ARRAY); 91 | node.set_array (json_array); 92 | 93 | var str = Json.to_string (node, false); 94 | 95 | try { 96 | return yield FileUtils.create_text_file (FILENAME, str); 97 | } catch (Error err) { 98 | critical ("Error: %s", err.message); 99 | return false; 100 | } 101 | } 102 | 103 | public inline Event? search_event_by_id (List events, string id) { 104 | unowned var link = events.search (id, (event, id) => strcmp (event.id, id)); 105 | return link?.data; 106 | } 107 | } 108 | -------------------------------------------------------------------------------- /src/Utils/file.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | namespace Countdown.FileUtils { 20 | async bool create_text_file (string filename, string contents, Cancellable? cancellable = null) throws Error { 21 | return yield ThreadUtils.run_in_thread (() => { 22 | var dir_path = Path.build_filename (Environment.get_user_data_dir (), "/io.github.lainsce.Countdown/"); 23 | 24 | if (DirUtils.create_with_parents (dir_path, 0755) != 0) { 25 | throw new Error (FileError.quark (), GLib.FileUtils.error_from_errno (errno), "%s", strerror (errno)); 26 | } 27 | 28 | var file_path = Path.build_filename (dir_path, filename); 29 | GLib.FileUtils.set_contents (file_path, contents); 30 | 31 | return true; 32 | }); 33 | } 34 | 35 | async string? read_text_file (string filename, Cancellable? cancellable = null) throws Error { 36 | return yield ThreadUtils.run_in_thread (() => { 37 | var file_path = Path.build_filename (Environment.get_user_data_dir (), "/io.github.lainsce.Countdown/", filename); 38 | 39 | string contents = ""; 40 | 41 | try { 42 | GLib.FileUtils.get_contents (file_path, out contents); 43 | } catch (Error err) { 44 | if (err is FileError.NOENT) 45 | return null; 46 | 47 | throw err; 48 | } 49 | 50 | return contents; 51 | }); 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /src/Utils/misc.vala: -------------------------------------------------------------------------------- 1 | namespace Countdown.MiscUtils { 2 | public T find_ancestor_of_type (Gtk.Widget? ancestor) { 3 | while ((ancestor = ancestor.get_parent ()) != null) { 4 | if (ancestor.get_type ().is_a (typeof (T))) 5 | return (T) ancestor; 6 | } 7 | 8 | return null; 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /src/Utils/obs_list.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public class Countdown.ObservableList : Object, ListModel { 20 | List data = new List (); 21 | 22 | public void add (T item) { 23 | var position = data.length (); 24 | 25 | data.append (item); 26 | 27 | items_changed (position, 0, 1); 28 | } 29 | 30 | public void add_all (List items) { 31 | var position = data.length (); 32 | 33 | foreach (var item in items) 34 | data.append (item); 35 | 36 | items_changed (position, 0, items.length ()); 37 | } 38 | 39 | public void remove_all () { 40 | var current_size = data.length (); 41 | data = new List (); 42 | items_changed (0, current_size, 0); 43 | } 44 | 45 | public new T @get (uint index) { 46 | return data.nth_data (index); 47 | } 48 | 49 | public bool remove (T item) { 50 | var position = data.index (item); 51 | 52 | if (position == -1) 53 | return false; 54 | 55 | data.remove (item); 56 | items_changed (position, 1, 0); 57 | 58 | return true; 59 | } 60 | 61 | Object? get_item (uint position) { 62 | return this[position] as Object; 63 | } 64 | 65 | Type get_item_type () { 66 | return typeof (T); 67 | } 68 | 69 | uint get_n_items () { 70 | return data.length (); 71 | } 72 | } 73 | -------------------------------------------------------------------------------- /src/Utils/sorter.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2017-2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | class Countdown.EventSorter : Gtk.Sorter { 20 | protected override Gtk.Ordering compare (Object? item1, Object? item2) { 21 | return Gtk.Ordering.from_cmpfunc ((item1 as Event)?.date.compare ((item2 as Event)?.date)); 22 | } 23 | 24 | protected override Gtk.SorterOrder get_order () { 25 | return TOTAL; 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /src/Utils/thread.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2017-2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | delegate void Countdown.WorkerFunc (); 20 | delegate G Countdown.ThreadFunc () throws Error; 21 | 22 | [Compact (opaque = true)] 23 | class Countdown.Worker { 24 | WorkerFunc func; 25 | 26 | public Worker (owned WorkerFunc func) { 27 | this.func = (owned) func; 28 | } 29 | 30 | public void run () { 31 | func (); 32 | } 33 | } 34 | 35 | namespace Countdown.ThreadUtils { 36 | Once> _once; 37 | 38 | unowned ThreadPool _get_thread_pool () { 39 | return _once.once (() => { 40 | var tp = new ThreadPool.with_owned_data (worker => worker.run (), 1, false); 41 | return tp; 42 | }); 43 | } 44 | 45 | async G run_in_thread (owned ThreadFunc func) throws Error { 46 | unowned var thread_pool = _get_thread_pool (); 47 | 48 | G result = null; 49 | Error? error = null; 50 | 51 | thread_pool.add (new Worker (() => { 52 | try { 53 | result = func (); 54 | } catch (Error err) { 55 | error = err; 56 | } 57 | 58 | Idle.add (run_in_thread.callback); 59 | })); 60 | 61 | yield; 62 | 63 | if (error != null) 64 | throw error; 65 | 66 | return result; 67 | } 68 | } 69 | -------------------------------------------------------------------------------- /src/ViewModels/eventviewmodel.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2017-2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public class Countdown.EventViewModel : Object { 20 | uint timeout_id = 0; 21 | 22 | public ObservableList events { get; default = new ObservableList (); } 23 | public EventRepository? repository { private get; construct; } 24 | 25 | public EventViewModel (EventRepository repository) { 26 | Object (repository: repository); 27 | } 28 | 29 | construct { 30 | populate_events.begin (); 31 | } 32 | 33 | public void create_new_event (Event? event) { 34 | events.add (event); 35 | repository.insert_event (event); 36 | save_events (); 37 | } 38 | 39 | public void update_event (Event event) { 40 | repository.update_event (event); 41 | 42 | save_events (); 43 | } 44 | 45 | public void delete_event (Event event) { 46 | events.remove (event); 47 | 48 | repository.delete_event (event.id); 49 | save_events (); 50 | } 51 | 52 | async void populate_events () { 53 | var events = yield repository.get_events (); 54 | this.events.add_all (events); 55 | } 56 | 57 | void save_events () { 58 | if (timeout_id != 0) 59 | Source.remove (timeout_id); 60 | 61 | timeout_id = Timeout.add (500, () => { 62 | timeout_id = 0; 63 | 64 | repository.save.begin (); 65 | 66 | return Source.REMOVE; 67 | }); 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /src/ViewModels/pasteventviewmodel.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2017-2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public class Countdown.PastEventViewModel : Object { 20 | uint timeout_id = 0; 21 | 22 | public ObservableList events { get; default = new ObservableList (); } 23 | public PastEventRepository? repository { private get; construct; } 24 | 25 | public PastEventViewModel (PastEventRepository repository) { 26 | Object (repository: repository); 27 | } 28 | 29 | construct { 30 | populate_events.begin (); 31 | } 32 | 33 | public void create_new_event (Event? event) { 34 | events.add (event); 35 | repository.insert_event (event); 36 | save_events (); 37 | } 38 | 39 | public void update_event (Event event) { 40 | repository.update_event (event); 41 | 42 | save_events (); 43 | } 44 | 45 | public void delete_event (Event event) { 46 | events.remove (event); 47 | 48 | repository.delete_event (event.id); 49 | save_events (); 50 | } 51 | 52 | async void populate_events () { 53 | var events = yield repository.get_events (); 54 | this.events.add_all (events); 55 | } 56 | 57 | void save_events () { 58 | if (timeout_id != 0) 59 | Source.remove (timeout_id); 60 | 61 | timeout_id = Timeout.add (500, () => { 62 | timeout_id = 0; 63 | 64 | repository.save.begin (); 65 | 66 | return Source.REMOVE; 67 | }); 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /src/Views/pastlistview.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 48 | 49 | 50 | -------------------------------------------------------------------------------- /src/Views/pastlistview.ui.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | [GtkTemplate (ui = "/io/github/lainsce/Countdown/pastlistview.ui")] 20 | public class Countdown.PastListView : View { 21 | public ObservableList? events { get; set; } 22 | public PastEventViewModel? past_view_model { get; set; } 23 | 24 | public signal void past_event_requested (); 25 | public signal void past_event_update_requested (Event event); 26 | } 27 | -------------------------------------------------------------------------------- /src/Views/upcominglistview.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 48 | 49 | 50 | -------------------------------------------------------------------------------- /src/Views/upcominglistview.ui.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | [GtkTemplate (ui = "/io/github/lainsce/Countdown/upcominglistview.ui")] 20 | public class Countdown.UpcomingListView : View { 21 | public ObservableList? events { get; set; } 22 | public EventViewModel? view_model { get; set; } 23 | 24 | public signal void new_event_requested (); 25 | public signal void event_update_requested (Event event); 26 | } 27 | -------------------------------------------------------------------------------- /src/Views/view.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | public abstract class Countdown.View : Gtk.Widget, Gtk.Buildable { 20 | Adw.Bin? child_bin = new Adw.Bin () { vexpand = true }; 21 | 22 | public Gtk.Widget? child { 23 | get { return child_bin.child; } 24 | set { child_bin.child = value; } 25 | } 26 | 27 | construct { 28 | layout_manager = new Gtk.BoxLayout (VERTICAL); 29 | 30 | child_bin?.set_parent (this); 31 | } 32 | 33 | protected override void dispose () { 34 | child_bin?.unparent (); 35 | child_bin = null; 36 | 37 | base.dispose (); 38 | } 39 | 40 | void add_child (Gtk.Builder builder, Object child, string? type) { 41 | if (child is Gtk.Widget) { 42 | this.child = (Gtk.Widget) child; 43 | return; 44 | } 45 | 46 | base.add_child (builder, child, type); 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /src/Widgets/dialog.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 115 | 116 | -------------------------------------------------------------------------------- /src/Widgets/dialog.ui.vala: -------------------------------------------------------------------------------- 1 | namespace Countdown { 2 | [GtkTemplate (ui = "/io/github/lainsce/Countdown/dialog.ui")] 3 | public class Widgets.Dialog : Adw.Window { 4 | [GtkChild] 5 | public unowned Gtk.Button new_button; 6 | [GtkChild] 7 | public unowned Gtk.Entry event_name_entry; 8 | [GtkChild] 9 | public unowned Gtk.Entry event_date_day_entry; 10 | [GtkChild] 11 | public unowned Gtk.Entry event_date_month_entry; 12 | [GtkChild] 13 | public unowned Gtk.Entry event_date_year_entry; 14 | 15 | public EventViewModel vm { get; construct; } 16 | public PastEventViewModel pvm { get; construct; } 17 | 18 | public Dialog (EventViewModel vm, PastEventViewModel pvm) { 19 | Object ( 20 | vm: vm, 21 | pvm: pvm 22 | ); 23 | } 24 | 25 | construct { 26 | event_date_day_entry.max_length = 2; 27 | event_date_month_entry.max_length = 2; 28 | event_date_year_entry.max_length = 4; 29 | 30 | event_name_entry.notify["text"].connect (update_new_button); 31 | event_date_day_entry.notify["text"].connect (update_new_button); 32 | event_date_month_entry.notify["text"].connect (update_new_button); 33 | event_date_year_entry.notify["text"].connect (update_new_button); 34 | } 35 | 36 | private DateTime get_datetime () { 37 | return new DateTime.local (int.parse (event_date_year_entry.get_text ()), 38 | int.parse (event_date_month_entry.get_text ()), 39 | int.parse (event_date_day_entry.get_text ()), 40 | 0, 41 | 0, 42 | 0.0); 43 | } 44 | 45 | private void update_new_button () { 46 | bool date_is_valid = get_datetime () != null; 47 | bool name_is_valid = event_name_entry.get_text () != ""; 48 | new_button.sensitive = (date_is_valid && name_is_valid); 49 | } 50 | 51 | [GtkCallback] 52 | public void on_new_event_requested () { 53 | var d = get_datetime (); 54 | if (d == null) return; 55 | 56 | var event = new Event (); 57 | event.title = event_name_entry.get_text (); 58 | event.date = d; 59 | 60 | GLib.TimeSpan res = 0; 61 | var e = new GLib.DateTime.now_local (); 62 | 63 | res = d.difference(e) / 86400000000; 64 | 65 | if (res <= 0) { 66 | event.passed = true; 67 | pvm.create_new_event (event); 68 | } else { 69 | event.passed = false; 70 | vm.create_new_event (event); 71 | } 72 | this.dispose (); 73 | } 74 | 75 | [GtkCallback] 76 | public void on_cancel_requested () { 77 | this.dispose (); 78 | } 79 | } 80 | } 81 | -------------------------------------------------------------------------------- /src/Widgets/eventrow.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 12 | 13 | -------------------------------------------------------------------------------- /src/Widgets/eventrowcontent.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 76 | 77 | -------------------------------------------------------------------------------- /src/Widgets/eventrowcontent.ui.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2017-2022 Lains 3 | * 4 | * This program is free software; you can redistribute it &&/or 5 | * modify it under the terms of the GNU General Public 6 | * License as published by the Free Software Foundation; either 7 | * version 3 of the License, or (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 | * General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public 15 | * License along with this program; if not, write to the 16 | * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 17 | * Boston, MA 02110-1301 USA 18 | */ 19 | [GtkTemplate (ui = "/io/github/lainsce/Countdown/eventrowcontent.ui")] 20 | public class Countdown.EventRowContent : Adw.Bin { 21 | [GtkChild] 22 | unowned Gtk.Button delete_button; 23 | 24 | public EventViewModel vm { get; set; } 25 | public PastEventViewModel pvm { get; set; } 26 | 27 | Event? _event; 28 | public Event? event { 29 | get { return _event; } 30 | set { 31 | if (value == _event) 32 | return; 33 | 34 | _event = value; 35 | 36 | delete_button.clicked.connect (() => { 37 | if (_event.passed == true) { 38 | var dialog = new Gtk.MessageDialog (((MainWindow)MiscUtils.find_ancestor_of_type(this)), 0, 0, 0, null); 39 | dialog.modal = true; 40 | 41 | dialog.set_title (_("Delete Past Event?")); 42 | dialog.text = (_("Deleting this past event will not save it in the past log and will stop tracking the days since the event happened.")); 43 | 44 | dialog.add_button (_("Cancel"), Gtk.ResponseType.CANCEL); 45 | var no_button = dialog.add_button (_("Delete"), Gtk.ResponseType.OK); 46 | no_button.get_style_context ().add_class ("destructive-action"); 47 | 48 | dialog.response.connect ((response_id) => { 49 | switch (response_id) { 50 | case Gtk.ResponseType.OK: 51 | ((MainWindow)MiscUtils.find_ancestor_of_type(this)).past_view_model.delete_event (_event);; 52 | dialog.close (); 53 | break; 54 | case Gtk.ResponseType.NO: 55 | dialog.close (); 56 | break; 57 | case Gtk.ResponseType.CANCEL: 58 | case Gtk.ResponseType.CLOSE: 59 | case Gtk.ResponseType.DELETE_EVENT: 60 | default: 61 | dialog.close (); 62 | return; 63 | } 64 | }); 65 | 66 | if (dialog != null) { 67 | dialog.present (); 68 | return; 69 | } else { 70 | dialog.show (); 71 | } 72 | } else { 73 | var dialog = new Gtk.MessageDialog (((MainWindow)MiscUtils.find_ancestor_of_type(this)), 0, 0, 0, null); 74 | dialog.modal = true; 75 | 76 | dialog.set_title (_("Delete Upcoming Event?")); 77 | dialog.text = (_("Deleting this upcoming event will not save it in the upcoming log and will stop tracking the days until the event happens.")); 78 | 79 | dialog.add_button (_("Cancel"), Gtk.ResponseType.CANCEL); 80 | var no_button = dialog.add_button (_("Delete"), Gtk.ResponseType.OK); 81 | no_button.get_style_context ().add_class ("destructive-action"); 82 | 83 | dialog.response.connect ((response_id) => { 84 | switch (response_id) { 85 | case Gtk.ResponseType.OK: 86 | ((MainWindow)MiscUtils.find_ancestor_of_type(this)).view_model.delete_event (_event); 87 | dialog.close (); 88 | break; 89 | case Gtk.ResponseType.NO: 90 | dialog.close (); 91 | break; 92 | case Gtk.ResponseType.CANCEL: 93 | case Gtk.ResponseType.CLOSE: 94 | case Gtk.ResponseType.DELETE_EVENT: 95 | default: 96 | dialog.close (); 97 | return; 98 | } 99 | }); 100 | 101 | if (dialog != null) { 102 | dialog.present (); 103 | return; 104 | } else { 105 | dialog.show (); 106 | } 107 | } 108 | }); 109 | } 110 | } 111 | 112 | public EventRowContent (Event event) { 113 | Object( 114 | event: event 115 | ); 116 | } 117 | 118 | construct { 119 | Timeout.add_seconds (60, () => { 120 | update_date_line (event.date); 121 | }); 122 | } 123 | 124 | [GtkCallback] 125 | string get_date_line () { 126 | return update_date_line (event.date); 127 | } 128 | 129 | public string update_date_line (DateTime date) { 130 | print ("Updated event!\n"); 131 | var res = ""; 132 | 133 | var e = new GLib.DateTime.now_local (); 134 | res = "%s".printf(((date.difference(e) / 86400000000).to_string())); 135 | if (int.parse(res) == 0 && event.passed != true) { 136 | event.passed = true; 137 | res = "-1"; 138 | ((MainWindow)MiscUtils.find_ancestor_of_type(this)).past_view_model.create_new_event (event); 139 | ((MainWindow)MiscUtils.find_ancestor_of_type(this)).view_model.delete_event (event); 140 | } 141 | 142 | if (int.parse(res) < 0) { 143 | return (int.parse(res) * -1).to_string(); 144 | } else { 145 | return res; 146 | } 147 | } 148 | } 149 | -------------------------------------------------------------------------------- /src/Widgets/keys.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 1 6 | 7 | 8 | shortcuts 9 | 12 10 | 11 | 12 | General 13 | 14 | 15 | <ctrl>q 16 | Quit 17 | 18 | 19 | 20 | 21 | <ctrl>n 22 | New Event 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | -------------------------------------------------------------------------------- /src/Widgets/mainwindow.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 187 | 188 | -------------------------------------------------------------------------------- /src/Widgets/mainwindow.ui.vala: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright 2022 Lains 3 | * 4 | * This program is free software: you can redistribute it and/or modify 5 | * it under the terms of the GNU General Public License as published by 6 | * the Free Software Foundation, either version 3 of the License, or 7 | * (at your option) any later version. 8 | * 9 | * This program is distributed in the hope that it will be useful, 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 | * GNU General Public License for more details. 13 | * 14 | * You should have received a copy of the GNU General Public License 15 | * along with this program. If not, see . 16 | */ 17 | 18 | namespace Countdown { 19 | [GtkTemplate (ui = "/io/github/lainsce/Countdown/mainwindow.ui")] 20 | public class MainWindow : Adw.ApplicationWindow { 21 | [GtkChild] 22 | unowned Gtk.MenuButton menu_button; 23 | [GtkChild] 24 | unowned Gtk.Stack event_stack; 25 | [GtkChild] 26 | unowned Gtk.SearchEntry event_searchbar; 27 | 28 | public EventViewModel view_model { get; construct; } 29 | public PastEventViewModel past_view_model { get; construct; } 30 | 31 | public SimpleActionGroup actions { get; construct; } 32 | public const string ACTION_PREFIX = "win."; 33 | public const string ACTION_ABOUT = "action_about"; 34 | public const string ACTION_KEYS = "action_keys"; 35 | public const string ACTION_NEW_EVENT = "action_new_event"; 36 | public static Gee.MultiMap action_accelerators = new Gee.HashMultiMap (); 37 | private const GLib.ActionEntry[] ACTION_ENTRIES = { 38 | {ACTION_ABOUT, action_about}, 39 | {ACTION_KEYS, action_keys}, 40 | {ACTION_NEW_EVENT, action_new_event}, 41 | }; 42 | 43 | public Adw.Application app { get; construct; } 44 | public MainWindow (Adw.Application app, EventViewModel view_model, PastEventViewModel past_view_model) { 45 | Object ( 46 | application: app, 47 | view_model: view_model, 48 | past_view_model: past_view_model, 49 | app: app 50 | ); 51 | } 52 | 53 | static construct { 54 | } 55 | 56 | construct { 57 | actions = new SimpleActionGroup (); 58 | actions.add_action_entries (ACTION_ENTRIES, this); 59 | insert_action_group ("win", actions); 60 | 61 | foreach (var action in action_accelerators.get_keys ()) { 62 | var accels_array = action_accelerators[action].to_array (); 63 | accels_array += null; 64 | 65 | app.set_accels_for_action (ACTION_PREFIX + action, accels_array); 66 | } 67 | app.set_accels_for_action("app.quit", {"q"}); 68 | app.set_accels_for_action("win.action_keys", {"question"}); 69 | app.set_accels_for_action("win.action_new_event", {"n"}); 70 | 71 | weak Gtk.IconTheme default_theme = Gtk.IconTheme.get_for_display (Gdk.Display.get_default ()); 72 | default_theme.add_resource_path ("/io/github/lainsce/Countdown"); 73 | 74 | var builder = new Gtk.Builder.from_resource ("/io/github/lainsce/Countdown/menu.ui"); 75 | menu_button.menu_model = (MenuModel)builder.get_object ("menu"); 76 | 77 | // The stack will be on Upcoming by default, so update number of events of there on opening. 78 | Timeout.add_seconds(1, () => { 79 | uint num = view_model.events.get_n_items (); 80 | event_searchbar.placeholder_text = num.to_string() + " " + (_("events")); 81 | return false; 82 | }); 83 | 84 | this.set_size_request (360, 360); 85 | this.show (); 86 | } 87 | 88 | // IO? 89 | [GtkCallback] 90 | void on_new_event_requested () { 91 | view_model.create_new_event (null); 92 | } 93 | 94 | [GtkCallback] 95 | public void on_event_update_requested (Event event) { 96 | view_model.update_event (event); 97 | } 98 | 99 | [GtkCallback] 100 | void on_past_event_requested () { 101 | past_view_model.create_new_event (null); 102 | } 103 | 104 | [GtkCallback] 105 | public void on_past_event_update_requested (Event event) { 106 | past_view_model.update_event (event); 107 | } 108 | 109 | [GtkCallback] 110 | void on_upcoming_stack_requested () { 111 | event_stack.set_visible_child_name ("upcoming"); 112 | uint num = view_model.events.get_n_items (); 113 | event_searchbar.placeholder_text = num.to_string() + " " + (_("events")); 114 | } 115 | 116 | [GtkCallback] 117 | void on_past_stack_requested () { 118 | event_stack.set_visible_child_name ("past"); 119 | uint num = past_view_model.events.get_n_items (); 120 | event_searchbar.placeholder_text = num.to_string() + " " + (_("events")); 121 | } 122 | 123 | public void action_new_event () { 124 | var new_event_dialog = new Widgets.Dialog (view_model, past_view_model); 125 | new_event_dialog.set_transient_for (this); 126 | new_event_dialog.show (); 127 | } 128 | 129 | public void action_keys () { 130 | try { 131 | var build = new Gtk.Builder (); 132 | build.add_from_resource ("/io/github/lainsce/Countdown/keys.ui"); 133 | var window = (Gtk.ShortcutsWindow) build.get_object ("shortcuts-countdown"); 134 | window.set_transient_for (this); 135 | window.show (); 136 | } catch (Error e) { 137 | warning ("Failed to open shortcuts window: %s\n", e.message); 138 | } 139 | } 140 | 141 | public void action_about () { 142 | const string COPYRIGHT = "Copyright \xc2\xa9 2022 Paulo \"Lains\" Galardi\n"; 143 | 144 | const string? AUTHORS[] = { 145 | "Paulo \"Lains\" Galardi", 146 | null 147 | }; 148 | 149 | var program_name = _("Countdown") + Config.NAME_SUFFIX; 150 | var about_window = new Adw.AboutWindow (); 151 | 152 | about_window.set_application_name(program_name); 153 | about_window.set_application_icon(Config.APP_ID); 154 | about_window.set_version(Config.VERSION); 155 | about_window.set_comments(_("Track events until they happen or since they happened.")); 156 | about_window.set_copyright(COPYRIGHT); 157 | about_window.set_developers(AUTHORS); 158 | about_window.set_license_type(Gtk.License.GPL_3_0); 159 | about_window.set_translator_credits(_("translator-credits")); 160 | 161 | about_window.set_transient_for(this); 162 | about_window.present(); 163 | } 164 | } 165 | } 166 | -------------------------------------------------------------------------------- /src/Widgets/menu.ui: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 |
6 | 7 | Keyboard Shortcuts 8 | win.action_keys 9 | 10 | 11 | About Countdown 12 | win.action_about 13 | 14 |
15 |
16 |
17 | -------------------------------------------------------------------------------- /src/app.vala: -------------------------------------------------------------------------------- 1 | public class Countdown.Application : Adw.Application { 2 | private const GLib.ActionEntry app_entries[] = { 3 | { "quit", quit }, 4 | }; 5 | 6 | public Application () { 7 | Object (application_id: Config.APP_ID); 8 | } 9 | public static int main (string[] args) { 10 | Intl.bindtextdomain (Config.GETTEXT_PACKAGE, Config.LOCALEDIR); 11 | Intl.textdomain (Config.GETTEXT_PACKAGE); 12 | 13 | var app = new Countdown.Application (); 14 | return app.run (args); 15 | } 16 | protected override void startup () { 17 | resource_base_path = "/io/github/lainsce/Countdown"; 18 | 19 | base.startup (); 20 | 21 | add_action_entries (app_entries, this); 22 | 23 | typeof (UpcomingListView).ensure (); 24 | 25 | var repo = new EventRepository (); 26 | var view_model = new EventViewModel (repo); 27 | 28 | var prepo = new PastEventRepository (); 29 | var pview_model = new PastEventViewModel (prepo); 30 | 31 | new MainWindow (this, view_model, pview_model); 32 | } 33 | protected override void activate () { 34 | active_window?.present (); 35 | } 36 | } 37 | -------------------------------------------------------------------------------- /src/config.vapi: -------------------------------------------------------------------------------- 1 | [CCode (cprefix = "", lower_case_cprefix = "", cheader_filename = "config.h")] 2 | namespace Config { 3 | public const string APP_ID; 4 | public const string NAME_SUFFIX; 5 | public const string VERSION; 6 | public const string GETTEXT_PACKAGE; 7 | public const string LOCALEDIR; 8 | public const bool DEVELOPMENT; 9 | } 10 | -------------------------------------------------------------------------------- /src/io.github.lainsce.Countdown.gresource.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Widgets/mainwindow.ui 5 | Widgets/dialog.ui 6 | Widgets/menu.ui 7 | Widgets/keys.ui 8 | Widgets/eventrow.ui 9 | Widgets/eventrowcontent.ui 10 | Views/upcominglistview.ui 11 | Views/pastlistview.ui 12 | 13 | style.css 14 | style-dark.css 15 | style-hc.css 16 | 17 | 18 | -------------------------------------------------------------------------------- /src/style-dark.css: -------------------------------------------------------------------------------- 1 | @define-color accent_color #ffba4b; 2 | @define-color accent_bg_color #ba5e1a; 3 | @define-color accent_fg_color #ffdbc6; 4 | @define-color window_bg_color #111; 5 | @define-color headerbar_bg_color #111; 6 | 7 | .event-row button { 8 | background: none; 9 | margin: 0; 10 | } 11 | .event-row button:hover { 12 | color: #ffb4a9; 13 | } 14 | -------------------------------------------------------------------------------- /src/style-hc.css: -------------------------------------------------------------------------------- 1 | .main-box { 2 | border: 1px solid @borders; 3 | } 4 | 5 | .list-action-button { 6 | border: 1px solid @borders; 7 | } 8 | 9 | .event-list row:not(.event-row) { 10 | border: 1px solid @borders; 11 | } 12 | -------------------------------------------------------------------------------- /src/style.css: -------------------------------------------------------------------------------- 1 | @define-color accent_color #9a4600; 2 | @define-color accent_bg_color #fa904a; 3 | @define-color accent_fg_color #331200; 4 | @define-color window_bg_color #fff; 5 | @define-color headerbar_bg_color #fff; 6 | 7 | .list-action-button { 8 | padding: 10px; 9 | border-radius: 999px; 10 | } 11 | .list-action-button.accent-button { 12 | background: @accent_bg_color; 13 | color: @accent_fg_color; 14 | box-shadow: 0 4px 8px 3px alpha(@accent_bg_color, 0.30); 15 | } 16 | .list-action-button.accent-button:hover { 17 | background: shade(@accent_bg_color, 1.024); 18 | box-shadow: 0 6px 10px 4px alpha(@accent_bg_color, 0.30); 19 | } 20 | .list-action-button.accent-button:active { 21 | background: shade(@accent_bg_color, 0.95); 22 | box-shadow: 0 4px 8px 3px alpha(@accent_bg_color, 0.30); 23 | } 24 | 25 | .main-box { 26 | background: linear-gradient(135deg, @window_bg_color, mix(@window_bg_color, @accent_bg_color, 0.2)); 27 | } 28 | 29 | .event-row button { 30 | background: none; 31 | margin: 0; 32 | } 33 | .event-row button:hover { 34 | color: #ba1b1b; 35 | } 36 | 37 | .event-list { 38 | margin: 0; 39 | padding: 0; 40 | } 41 | .event-list .event-row { 42 | background: none; 43 | padding: 8px 0; 44 | border-radius: 12px; 45 | margin-bottom: 0px; 46 | } 47 | .event-list .event-row:hover { 48 | background: none; 49 | } 50 | .event-list row { 51 | background-color: alpha(#888, 0.11); 52 | margin-bottom: 4px; 53 | border-radius: 12px; 54 | padding: 0 8px 0 0; 55 | } 56 | .event-list row:hover { 57 | background-color: alpha(#888, 0.22); 58 | } 59 | 60 | .event-list row { 61 | border-radius: 8px; 62 | } 63 | 64 | .dialog-event list { 65 | background: @view_bg_color; 66 | box-shadow: none; 67 | } 68 | 69 | .event-title { 70 | font-size: 1.5rem; 71 | font-weight: 400; 72 | } 73 | 74 | .stack-button { 75 | font-size: 1.5rem; 76 | font-weight: 300; 77 | padding: 6px 0; 78 | margin: 0; 79 | } 80 | 81 | .stack-button:checked { 82 | font-weight: 600; 83 | background: none; 84 | } 85 | 86 | .search-bar { 87 | border-radius: 999px; 88 | } 89 | --------------------------------------------------------------------------------