Compare commits
No commits in common. "trunk" and "old-wip-ver" have entirely different histories.
trunk
...
old-wip-ve
179 changed files with 1449 additions and 54942 deletions
8
.gitignore
vendored
8
.gitignore
vendored
|
@ -1,9 +1,7 @@
|
|||
*~
|
||||
*.beam
|
||||
*.ez
|
||||
/build
|
||||
build
|
||||
erl_crash.dump
|
||||
/output
|
||||
.DS_Store
|
||||
/data
|
||||
node_modules
|
||||
/static
|
||||
/output
|
||||
|
|
|
@ -1,2 +1,2 @@
|
|||
gleam 1.4.1
|
||||
nodejs 20.10.0
|
||||
nodejs 18.16.0
|
||||
gleam 0.30.2
|
||||
|
|
25
CHANGELOG.md
25
CHANGELOG.md
|
@ -1,25 +0,0 @@
|
|||
# Changelog
|
||||
|
||||
## 2.0.1
|
||||
|
||||
- Bumped dependencies to build on the latest stdlib, which is now also required.
|
||||
|
||||
## 2.0.0
|
||||
|
||||
- Added Gettext support so that the created blog can be translated.
|
||||
- Added header `lang` that can set the language of an individual page or post.
|
||||
- Added language information to the Atom feed.
|
||||
- Split `tags` and `archives` components out of the base layout for easier reuse.
|
||||
- Made base layout `pre_render` function public so base layout is easier to customise.
|
||||
|
||||
## 1.0.2
|
||||
|
||||
Adjusted the title sizing.
|
||||
|
||||
## 1.0.1
|
||||
|
||||
Fix dependency to `lustre_ssg`.
|
||||
|
||||
## 1.0.0
|
||||
|
||||
Initial release.
|
681
LICENSE.txt
681
LICENSE.txt
|
@ -1,681 +0,0 @@
|
|||
Scriptorium, Gleam blog generator
|
||||
Copyright © 2024 Mikko Ahlroth
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU Affero General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU Affero General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Affero General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
============================
|
||||
|
||||
AGPL 3 license text follows:
|
||||
|
||||
GNU AFFERO GENERAL PUBLIC LICENSE
|
||||
Version 3, 19 November 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU Affero General Public License is a free, copyleft license for
|
||||
software and other kinds of works, specifically designed to ensure
|
||||
cooperation with the community in the case of network server software.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
our General Public Licenses are intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
Developers that use our General Public Licenses protect your rights
|
||||
with two steps: (1) assert copyright on the software, and (2) offer
|
||||
you this License which gives you legal permission to copy, distribute
|
||||
and/or modify the software.
|
||||
|
||||
A secondary benefit of defending all users' freedom is that
|
||||
improvements made in alternate versions of the program, if they
|
||||
receive widespread use, become available for other developers to
|
||||
incorporate. Many developers of free software are heartened and
|
||||
encouraged by the resulting cooperation. However, in the case of
|
||||
software used on network servers, this result may fail to come about.
|
||||
The GNU General Public License permits making a modified version and
|
||||
letting the public access it on a server without ever releasing its
|
||||
source code to the public.
|
||||
|
||||
The GNU Affero General Public License is designed specifically to
|
||||
ensure that, in such cases, the modified source code becomes available
|
||||
to the community. It requires the operator of a network server to
|
||||
provide the source code of the modified version running there to the
|
||||
users of that server. Therefore, public use of a modified version, on
|
||||
a publicly accessible server, gives the public access to the source
|
||||
code of the modified version.
|
||||
|
||||
An older license, called the Affero General Public License and
|
||||
published by Affero, was designed to accomplish similar goals. This is
|
||||
a different license, not a version of the Affero GPL, but Affero has
|
||||
released a new version of the Affero GPL which permits relicensing under
|
||||
this license.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU Affero General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Remote Network Interaction; Use with the GNU General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, if you modify the
|
||||
Program, your modified version must prominently offer all users
|
||||
interacting with it remotely through a computer network (if your version
|
||||
supports such interaction) an opportunity to receive the Corresponding
|
||||
Source of your version by providing access to the Corresponding Source
|
||||
from a network server at no charge, through some standard or customary
|
||||
means of facilitating copying of software. This Corresponding Source
|
||||
shall include the Corresponding Source for any work covered by version 3
|
||||
of the GNU General Public License that is incorporated pursuant to the
|
||||
following paragraph.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the work with which it is combined will remain governed by version
|
||||
3 of the GNU General Public License.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU Affero General Public License from time to time. Such new versions
|
||||
will be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU Affero General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU Affero General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU Affero General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU Affero General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU Affero General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Affero General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If your software can interact with users remotely through a computer
|
||||
network, you should also make sure that it provides a way for users to
|
||||
get its source. For example, if your program is a web application, its
|
||||
interface could display a "Source" link that leads users to an archive
|
||||
of the code. There are many ways you could offer source, and different
|
||||
solutions will be better for different programs; see section 13 for the
|
||||
specific requirements.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU AGPL, see
|
||||
<https://www.gnu.org/licenses/>.
|
88
README.md
88
README.md
|
@ -1,86 +1,24 @@
|
|||
# scriptorium
|
||||
# pilkahdus
|
||||
|
||||
Scriptorium is a simple blog generator for Gleam, using [Lustre](https://hexdocs.pm/lustre) and
|
||||
[lustre_ssg](https://hexdocs.pm/lustre_ssg).
|
||||
[![Package Version](https://img.shields.io/hexpm/v/pilkahdus)](https://hex.pm/packages/pilkahdus)
|
||||
[![Hex Docs](https://img.shields.io/badge/hex-docs-ffaff3)](https://hexdocs.pm/pilkahdus/)
|
||||
|
||||
Scriptorium runs on the JavaScript target and is tested with Node.js.
|
||||
A Gleam project
|
||||
|
||||
[![Package Version](https://img.shields.io/hexpm/v/scriptorium)](https://hex.pm/packages/scriptorium)
|
||||
[![Hex Docs](https://img.shields.io/badge/hex-docs-ffaff3)](https://hexdocs.pm/scriptorium/)
|
||||
|
||||
## Quickstart
|
||||
|
||||
To use Scriptorium, you will need to create a new Gleam project and add Scriptorium as a
|
||||
dependency.
|
||||
## Quick start
|
||||
|
||||
```sh
|
||||
gleam add scriptorium
|
||||
gleam run # Run the project
|
||||
gleam test # Run the tests
|
||||
gleam shell # Run an Erlang shell
|
||||
```
|
||||
|
||||
To generate a default blog, use the following in your main file:
|
||||
## Installation
|
||||
|
||||
```gleam
|
||||
import gleam/result
|
||||
import gleam/option
|
||||
import gleam/io
|
||||
import scriptorium/builder
|
||||
import scriptorium/config.{type Configuration, Configuration}
|
||||
import scriptorium/defaults
|
||||
If available on Hex this package can be added to your Gleam project:
|
||||
|
||||
pub fn main() {
|
||||
let config =
|
||||
defaults.default_config(
|
||||
"My Blog",
|
||||
"https://my.blog.example/",
|
||||
"en",
|
||||
config.Author(
|
||||
"Person McPerson",
|
||||
option.Some("person@example.com"),
|
||||
option.Some("https://fedi.instance.example/@person"),
|
||||
),
|
||||
"© Person McPerson",
|
||||
)
|
||||
|
||||
io.debug(build(config))
|
||||
}
|
||||
|
||||
pub fn build(config: Configuration) {
|
||||
// Parse the files
|
||||
use db <- result.try(builder.parse(config))
|
||||
// Compile Markdown into HTML
|
||||
let compiled = builder.compile(db, config)
|
||||
// Render content into Lustre elements
|
||||
let rendered = builder.render(db, compiled, config)
|
||||
// Write rendered content into the filesystem
|
||||
builder.write(rendered, config)
|
||||
}
|
||||
```sh
|
||||
gleam add pilkahdus
|
||||
```
|
||||
|
||||
Now you will need to add source files. Create the following directory tree in the root of the project
|
||||
folder:
|
||||
|
||||
```
|
||||
your_project
|
||||
└── data
|
||||
├── pages
|
||||
└── posts
|
||||
```
|
||||
|
||||
In `./data/posts`, create a file `2024-04-21-hello-world.md` with the following contents:
|
||||
|
||||
```markdown
|
||||
Hello, World!
|
||||
meta
|
||||
|
||||
This is the first blog entry.
|
||||
```
|
||||
|
||||
Then run `gleam run --target javascript` to build the blog and see what was generated in `./output`.
|
||||
|
||||
For further usage help, see the documentation as described below.
|
||||
|
||||
## Documentation
|
||||
|
||||
- [User's Guide](https://nicd.gitlab.io/scriptorium_blog/guide.html)
|
||||
- [Scriptorium API reference](https://hexdocs.pm/scriptorium)
|
||||
- [Scriptorium Demo Blog](https://nicd.gitlab.io/scriptorium_blog)
|
||||
and its documentation can be found at <https://hexdocs.pm/pilkahdus>.
|
||||
|
|
38
gleam.toml
38
gleam.toml
|
@ -1,36 +1,12 @@
|
|||
name = "scriptorium"
|
||||
version = "2.0.1"
|
||||
name = "gloss"
|
||||
version = "1.0.0"
|
||||
description = "A static blog generator"
|
||||
target = "javascript"
|
||||
gleam = "~> 1.1"
|
||||
|
||||
# Fill out these fields if you intend to generate HTML documentation or publish
|
||||
# your project to the Hex package manager.
|
||||
#
|
||||
description = "A simple Gleam blog generator"
|
||||
licences = ["AGPL-3.0-or-later"]
|
||||
repository = { type = "forgejo", host = "git.ahlcode.fi", user = "nicd", repo = "scriptorium" }
|
||||
links = [
|
||||
{ title = "User's Guide", href = "https://nicd.gitlab.io/scriptorium_blog/guide.html" },
|
||||
{ title = "Demo Blog", href = "https://nicd.gitlab.io/scriptorium_blog" },
|
||||
]
|
||||
#
|
||||
# For a full reference of all the available options, you can have a look at
|
||||
# https://gleam.run/writing-gleam/gleam-toml/.
|
||||
|
||||
[documentation]
|
||||
pages = [
|
||||
{ title = "Changelog", path = "changelog.html", source = "./CHANGELOG.md" },
|
||||
]
|
||||
gleam = ">= 0.30.0"
|
||||
|
||||
[dependencies]
|
||||
gleam_stdlib = ">= 0.40.0 and < 2.0.0"
|
||||
lustre = ">= 4.4.3 and < 5.0.0"
|
||||
lustre_ssg = "0.8.3"
|
||||
gleam_javascript = ">= 0.12.0 and < 1.0.0"
|
||||
ranged_int = ">= 2.0.0 and < 3.0.0"
|
||||
bigi = ">= 3.0.0 and < 4.0.0"
|
||||
simplifile = ">= 2.1.0 and < 3.0.0"
|
||||
kielet = ">= 2.0.0 and < 3.0.0"
|
||||
gleam_stdlib = "~> 0.30"
|
||||
glemplate = "~> 5.0"
|
||||
gleam_javascript = "~> 0.4"
|
||||
|
||||
[dev-dependencies]
|
||||
gleeunit = ">= 1.2.0 and < 2.0.0"
|
||||
|
|
|
@ -2,36 +2,14 @@
|
|||
# You typically do not need to edit this file
|
||||
|
||||
packages = [
|
||||
{ name = "bigi", version = "3.1.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "bigi", source = "hex", outer_checksum = "59F1DBCDE4C21B8C47D83A0A8012468E1A89071D326C31A353F12D7D23471523" },
|
||||
{ name = "envoy", version = "1.0.1", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "envoy", source = "hex", outer_checksum = "CFAACCCFC47654F7E8B75E614746ED924C65BD08B1DE21101548AC314A8B6A41" },
|
||||
{ name = "exception", version = "2.0.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "exception", source = "hex", outer_checksum = "F5580D584F16A20B7FCDCABF9E9BE9A2C1F6AC4F9176FA6DD0B63E3B20D450AA" },
|
||||
{ name = "filepath", version = "1.0.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "filepath", source = "hex", outer_checksum = "EFB6FF65C98B2A16378ABC3EE2B14124168C0CE5201553DE652E2644DCFDB594" },
|
||||
{ name = "gleam_crypto", version = "1.3.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleam_crypto", source = "hex", outer_checksum = "ADD058DEDE8F0341F1ADE3AAC492A224F15700829D9A3A3F9ADF370F875C51B7" },
|
||||
{ name = "gleam_erlang", version = "0.26.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleam_erlang", source = "hex", outer_checksum = "3DF72F95F4716883FA51396FB0C550ED3D55195B541568CAF09745984FD37AD1" },
|
||||
{ name = "gleam_javascript", version = "0.12.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleam_javascript", source = "hex", outer_checksum = "6EB652538B31E852FE0A8307A8B6314DEB34930944B6DDC41CCC31CA344DA35D" },
|
||||
{ name = "gleam_json", version = "1.0.1", build_tools = ["gleam"], requirements = ["gleam_stdlib", "thoas"], otp_app = "gleam_json", source = "hex", outer_checksum = "9063D14D25406326C0255BDA0021541E797D8A7A12573D849462CAFED459F6EB" },
|
||||
{ name = "gleam_otp", version = "0.12.0", build_tools = ["gleam"], requirements = ["gleam_erlang", "gleam_stdlib"], otp_app = "gleam_otp", source = "hex", outer_checksum = "CD5FC777E99673BDB390092DF85E34EAA6B8EE1882147496290AB3F45A4960B1" },
|
||||
{ name = "gleam_stdlib", version = "0.40.0", build_tools = ["gleam"], requirements = [], otp_app = "gleam_stdlib", source = "hex", outer_checksum = "86606B75A600BBD05E539EB59FABC6E307EEEA7B1E5865AFB6D980A93BCB2181" },
|
||||
{ name = "gleeunit", version = "1.2.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleeunit", source = "hex", outer_checksum = "F7A7228925D3EE7D0813C922E062BFD6D7E9310F0BEE585D3A42F3307E3CFD13" },
|
||||
{ name = "jot", version = "1.0.1", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "jot", source = "hex", outer_checksum = "A0A52BD8D079AB0ABF80BE11DC63B85CF5791125991A029FAD7D17820D9419D8" },
|
||||
{ name = "kielet", version = "2.0.0", build_tools = ["gleam"], requirements = ["gleam_stdlib", "nibble"], otp_app = "kielet", source = "hex", outer_checksum = "C6F91EECDC54EAE7C55DE1679C8E7CC4823259782DB13E656E4EC2A2590005B8" },
|
||||
{ name = "lustre", version = "4.4.3", build_tools = ["gleam"], requirements = ["gleam_erlang", "gleam_json", "gleam_otp", "gleam_stdlib"], otp_app = "lustre", source = "hex", outer_checksum = "AA54C98497C0CF8750B0247F351D7E38EB43E46B9E8AA755900E4C64103BD9F7" },
|
||||
{ name = "lustre_ssg", version = "0.8.3", build_tools = ["gleam"], requirements = ["filepath", "gleam_stdlib", "jot", "lustre", "simplifile", "temporary", "tom"], otp_app = "lustre_ssg", source = "hex", outer_checksum = "5F1DF752AE6AAD0BD332CCB52E1CF2086997CF55EB59894080F4E3ADF691B95C" },
|
||||
{ name = "nibble", version = "1.1.1", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "nibble", source = "hex", outer_checksum = "67C6BEBC1AB6D771AB893B4A7B3E66C92668C6E7774C335FEFCD545B06435FE5" },
|
||||
{ name = "ranged_int", version = "2.0.0", build_tools = ["gleam"], requirements = ["bigi", "gleam_stdlib"], otp_app = "ranged_int", source = "hex", outer_checksum = "9FCDA804C1884015FC25F3F8BE429FC450D402F861B5C561464479F5B1162A41" },
|
||||
{ name = "simplifile", version = "2.1.0", build_tools = ["gleam"], requirements = ["filepath", "gleam_stdlib"], otp_app = "simplifile", source = "hex", outer_checksum = "BDD04F5D31D6D34E2EDFAEF0B68A6297AEC939888C3BFCE61133DE13857F6DA2" },
|
||||
{ name = "temporary", version = "1.0.0", build_tools = ["gleam"], requirements = ["envoy", "exception", "filepath", "gleam_crypto", "gleam_stdlib", "simplifile"], otp_app = "temporary", source = "hex", outer_checksum = "51C0FEF4D72CE7CA507BD188B21C1F00695B3D5B09D7DFE38240BFD3A8E1E9B3" },
|
||||
{ name = "thoas", version = "1.2.1", build_tools = ["rebar3"], requirements = [], otp_app = "thoas", source = "hex", outer_checksum = "E38697EDFFD6E91BD12CEA41B155115282630075C2A727E7A6B2947F5408B86A" },
|
||||
{ name = "tom", version = "1.0.1", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "tom", source = "hex", outer_checksum = "9EECB60150E834A07238BD5C7DF1FF07F7D4C5862BB8A773923D1981C7875FB0" },
|
||||
{ name = "gleam_javascript", version = "0.4.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleam_javascript", source = "hex", outer_checksum = "E0E8D33461776BFCC124838183F85E430D3A71D7318F210C9DE0CFB52E5AC8DE" },
|
||||
{ name = "gleam_stdlib", version = "0.30.0", build_tools = ["gleam"], requirements = [], otp_app = "gleam_stdlib", source = "hex", outer_checksum = "03710B3DA047A3683117591707FCA19D32B980229DD8CE8B0603EB5B5144F6C3" },
|
||||
{ name = "glemplate", version = "5.0.0", build_tools = ["gleam"], requirements = ["gleam_stdlib", "glentities", "nibble"], otp_app = "glemplate", source = "hex", outer_checksum = "48EB0FDD937B19460D8A095370515C1157632A8218539A77E73DDA208F193B46" },
|
||||
{ name = "glentities", version = "5.0.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "glentities", source = "hex", outer_checksum = "4D3AE91ECB088C853302643C4F256BEF7B8F8EC7650A7DA6B2ACAF4550113BE2" },
|
||||
{ name = "nibble", version = "0.2.3", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "nibble", source = "hex", outer_checksum = "C9EB428EAC11007414C2FBA5B39F0E8DB27FDAB26534F2AD607D5AA6FCF32559" },
|
||||
]
|
||||
|
||||
[requirements]
|
||||
bigi = { version = ">= 3.0.0 and < 4.0.0" }
|
||||
gleam_javascript = { version = ">= 0.12.0 and < 1.0.0" }
|
||||
gleam_stdlib = { version = ">= 0.40.0 and < 2.0.0" }
|
||||
gleeunit = { version = ">= 1.2.0 and < 2.0.0" }
|
||||
kielet = { version = ">= 2.0.0 and < 3.0.0" }
|
||||
lustre = { version = ">= 4.4.3 and < 5.0.0" }
|
||||
lustre_ssg = { version = "0.8.3" }
|
||||
ranged_int = { version = ">= 2.0.0 and < 3.0.0" }
|
||||
simplifile = { version = ">= 2.1.0 and < 3.0.0" }
|
||||
gleam_javascript = { version = "~> 0.4" }
|
||||
gleam_stdlib = { version = "~> 0.30" }
|
||||
glemplate = { version = "~> 5.0" }
|
||||
|
|
16
package.json
Normal file
16
package.json
Normal file
|
@ -0,0 +1,16 @@
|
|||
{
|
||||
"name": "gloss",
|
||||
"version": "1.0.0",
|
||||
"description": "Glossy blog generator",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://gitlab.com/Nicd/gloss.git"
|
||||
},
|
||||
"author": "Mikko Ahlroth <mikko@ahlroth.fi>",
|
||||
"license": "AGPL-3.0-or-later",
|
||||
"bugs": {
|
||||
"url": "https://gitlab.com/Nicd/gloss/issues"
|
||||
},
|
||||
"homepage": "https://gitlab.com/Nicd/gloss#readme",
|
||||
"private": true
|
||||
}
|
|
@ -1,195 +0,0 @@
|
|||
@charset "UTF-8";
|
||||
|
||||
:root {
|
||||
--bg: #fbfaf5;
|
||||
--fg: #2c2c2c;
|
||||
|
||||
--sidebar-size: minmax(min-content, 1fr);
|
||||
--sidebar-max-size: 35rem;
|
||||
--sidebar-min-size: 30rem;
|
||||
--main-size: auto;
|
||||
}
|
||||
|
||||
body {
|
||||
display: grid;
|
||||
|
||||
grid-template:
|
||||
"title main ." min-content
|
||||
"sidebar main ." 1fr
|
||||
"sidebar footer ." auto
|
||||
/ var(--sidebar-size) var(--main-size) var(--sidebar-size);
|
||||
}
|
||||
|
||||
#title {
|
||||
grid-area: title;
|
||||
justify-self: end;
|
||||
width: 100%;
|
||||
max-width: var(--sidebar-max-size);
|
||||
min-width: var(--sidebar-min-size);
|
||||
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
#title h1 {
|
||||
margin: 0;
|
||||
padding: 3rem 1rem 1rem 1rem;
|
||||
font-size: 2.25rem;
|
||||
}
|
||||
|
||||
body > main {
|
||||
grid-area: main;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
#sidebar {
|
||||
grid-area: sidebar;
|
||||
justify-self: end;
|
||||
width: 100%;
|
||||
max-width: var(--sidebar-max-size);
|
||||
min-width: var(--sidebar-min-size);
|
||||
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
#title > nav {
|
||||
padding: 0.7rem;
|
||||
}
|
||||
|
||||
#title > nav > ul {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
justify-content: space-around;
|
||||
align-items: baseline;
|
||||
flex-wrap: wrap;
|
||||
gap: 1rem;
|
||||
}
|
||||
|
||||
#title > nav > ul > li {
|
||||
display: inline-block;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
body > footer {
|
||||
grid-area: footer;
|
||||
}
|
||||
|
||||
.page-nav ul {
|
||||
list-style-type: none;
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
flex-wrap: wrap;
|
||||
justify-content: center;
|
||||
align-items: baseline;
|
||||
gap: 0.2rem;
|
||||
}
|
||||
|
||||
.page-nav ul li a {
|
||||
display: inline-block;
|
||||
padding: 0.8rem;
|
||||
}
|
||||
|
||||
#tags {
|
||||
list-style-type: none;
|
||||
font-size: 2.75rem;
|
||||
}
|
||||
|
||||
#tags li {
|
||||
display: inline-block;
|
||||
margin: 0.2rem;
|
||||
}
|
||||
|
||||
#tags a {
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
#tags a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
#archives li {
|
||||
margin-left: 2rem;
|
||||
}
|
||||
|
||||
.post-list {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 4rem;
|
||||
}
|
||||
|
||||
.post > header h2,
|
||||
.page > header h2 {
|
||||
font-size: 3rem;
|
||||
line-height: 3rem;
|
||||
}
|
||||
|
||||
.post > header,
|
||||
.page > header {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.post > header nav ul {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
flex-wrap: wrap;
|
||||
justify-content: flex-start;
|
||||
align-items: baseline;
|
||||
gap: 0.5rem;
|
||||
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.post > header nav ul li {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.post > footer {
|
||||
text-align: left;
|
||||
|
||||
margin: 0;
|
||||
padding-left: 0;
|
||||
padding-right: 0;
|
||||
}
|
||||
|
||||
/* Fix code blocks making the page wide */
|
||||
pre {
|
||||
white-space: pre-wrap;
|
||||
word-break: normal;
|
||||
overflow-wrap: anywhere;
|
||||
tab-size: 4;
|
||||
}
|
||||
|
||||
@media (max-width: 900px) {
|
||||
body {
|
||||
grid-template: "title" "main" "sidebar" "footer";
|
||||
}
|
||||
|
||||
#title {
|
||||
justify-self: stretch;
|
||||
max-width: unset;
|
||||
min-width: unset;
|
||||
}
|
||||
|
||||
#sidebar {
|
||||
justify-self: stretch;
|
||||
max-width: unset;
|
||||
min-width: unset;
|
||||
padding-left: 2rem;
|
||||
padding-right: 2rem;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 600px) {
|
||||
main {
|
||||
padding: 0.5rem;
|
||||
}
|
||||
|
||||
#sidebar {
|
||||
padding-left: 0.7rem;
|
||||
padding-right: 0.7rem;
|
||||
}
|
||||
}
|
|
@ -1,476 +0,0 @@
|
|||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Libre";
|
||||
font-style: italic;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Libre/AveriaLibre-Italic.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Libre";
|
||||
font-style: italic;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Libre/AveriaLibre-BoldItalic.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Libre";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Libre/AveriaLibre-Regular.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Libre";
|
||||
font-style: normal;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Libre/AveriaLibre-Bold.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Serif Libre";
|
||||
font-style: italic;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Serif_Libre/AveriaSerifLibre-LightItalic.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Serif Libre";
|
||||
font-style: italic;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Serif_Libre/AveriaSerifLibre-BoldItalic.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Serif Libre";
|
||||
font-style: normal;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Serif_Libre/AveriaSerifLibre-Light.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Averia Serif Libre";
|
||||
font-style: normal;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Averia_Serif_Libre/AveriaSerifLibre-Bold.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* cyrillic-ext */
|
||||
@font-face {
|
||||
font-family: "Caveat";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Caveat/Caveat-VariableFont_wght.ttf") format("truetype");
|
||||
unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F,
|
||||
U+FE2E-FE2F;
|
||||
}
|
||||
/* cyrillic */
|
||||
@font-face {
|
||||
font-family: "Caveat";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Caveat/Caveat-VariableFont_wght.ttf") format("truetype");
|
||||
unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Caveat";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Caveat/Caveat-VariableFont_wght.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Caveat";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Caveat/Caveat-VariableFont_wght.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: italic;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Italic.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: italic;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Italic.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: italic;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-BoldItalic.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: italic;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-BoldItalic.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Regular.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Regular.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: normal;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Bold.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Courier Prime";
|
||||
font-style: normal;
|
||||
font-weight: 700;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Courier_Prime/CourierPrime-Bold.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* braille */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+2800-28FF;
|
||||
}
|
||||
/* math */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0302-0303, U+0305, U+0307-0308, U+0330, U+0391-03A1,
|
||||
U+03A3-03A9, U+03B1-03C9, U+03D1, U+03D5-03D6, U+03F0-03F1, U+03F4-03F5,
|
||||
U+2034-2037, U+2057, U+20D0-20DC, U+20E1, U+20E5-20EF, U+2102, U+210A-210E,
|
||||
U+2110-2112, U+2115, U+2119-211D, U+2124, U+2128, U+212C-212D, U+212F-2131,
|
||||
U+2133-2138, U+213C-2140, U+2145-2149, U+2190, U+2192, U+2194-21AE,
|
||||
U+21B0-21E5, U+21F1-21F2, U+21F4-2211, U+2213-2214, U+2216-22FF, U+2308-230B,
|
||||
U+2310, U+2319, U+231C-2321, U+2336-237A, U+237C, U+2395, U+239B-23B6,
|
||||
U+23D0, U+23DC-23E1, U+2474-2475, U+25AF, U+25B3, U+25B7, U+25BD, U+25C1,
|
||||
U+25CA, U+25CC, U+25FB, U+266D-266F, U+27C0-27FF, U+2900-2AFF, U+2B0E-2B11,
|
||||
U+2B30-2B4C, U+2BFE, U+FF5B, U+FF5D, U+1D400-1D7FF, U+1EE00-1EEFF;
|
||||
}
|
||||
/* mayan-numerals */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+1D2E0-1D2F3;
|
||||
}
|
||||
/* symbols */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0001-000C, U+000E-001F, U+007F-009F, U+20DD-20E0, U+20E2-20E4,
|
||||
U+2150-218F, U+2190, U+2192, U+2194-2199, U+21AF, U+21E6-21F0, U+21F3,
|
||||
U+2218-2219, U+2299, U+22C4-22C6, U+2300-243F, U+2440-244A, U+2460-24FF,
|
||||
U+25A0-27BF, U+2800-28FF, U+2921-2922, U+2981, U+29BF, U+29EB, U+2B00-2BFF,
|
||||
U+4DC0-4DFF, U+FFF9-FFFB, U+10140-1018E, U+10190-1019C, U+101A0,
|
||||
U+101D0-101FD, U+102E0-102FB, U+10E60-10E7E, U+1D2C0-1D2D3, U+1D2E0-1D37F,
|
||||
U+1F000-1F0FF, U+1F100-1F1AD, U+1F1E6-1F1FF, U+1F30D-1F30F, U+1F315, U+1F31C,
|
||||
U+1F31E, U+1F320-1F32C, U+1F336, U+1F378, U+1F37D, U+1F382, U+1F393-1F39F,
|
||||
U+1F3A7-1F3A8, U+1F3AC-1F3AF, U+1F3C2, U+1F3C4-1F3C6, U+1F3CA-1F3CE,
|
||||
U+1F3D4-1F3E0, U+1F3ED, U+1F3F1-1F3F3, U+1F3F5-1F3F7, U+1F408, U+1F415,
|
||||
U+1F41F, U+1F426, U+1F43F, U+1F441-1F442, U+1F444, U+1F446-1F449,
|
||||
U+1F44C-1F44E, U+1F453, U+1F46A, U+1F47D, U+1F4A3, U+1F4B0, U+1F4B3, U+1F4B9,
|
||||
U+1F4BB, U+1F4BF, U+1F4C8-1F4CB, U+1F4D6, U+1F4DA, U+1F4DF, U+1F4E3-1F4E6,
|
||||
U+1F4EA-1F4ED, U+1F4F7, U+1F4F9-1F4FB, U+1F4FD-1F4FE, U+1F503, U+1F507-1F50B,
|
||||
U+1F50D, U+1F512-1F513, U+1F53E-1F54A, U+1F54F-1F5FA, U+1F610, U+1F650-1F67F,
|
||||
U+1F687, U+1F68D, U+1F691, U+1F694, U+1F698, U+1F6AD, U+1F6B2, U+1F6B9-1F6BA,
|
||||
U+1F6BC, U+1F6C6-1F6CF, U+1F6D3-1F6D7, U+1F6E0-1F6EA, U+1F6F0-1F6F3,
|
||||
U+1F6F7-1F6FC, U+1F700-1F7FF, U+1F800-1F80B, U+1F810-1F847, U+1F850-1F859,
|
||||
U+1F860-1F887, U+1F890-1F8AD, U+1F8B0-1F8B1, U+1F900-1F90B, U+1F93B, U+1F946,
|
||||
U+1F984, U+1F996, U+1F9E9, U+1FA00-1FA6F, U+1FA70-1FA7C, U+1FA80-1FA88,
|
||||
U+1FA90-1FABD, U+1FABF-1FAC5, U+1FACE-1FADB, U+1FAE0-1FAE8, U+1FAF0-1FAF8,
|
||||
U+1FB00-1FBFF;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Noto Sans Symbols 2";
|
||||
font-style: normal;
|
||||
font-weight: 400;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Noto_Sans_Symbols_2/NotoSansSymbols2-Regular.ttf")
|
||||
format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* cyrillic */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-LightItalic.ttf") format("truetype");
|
||||
unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||
}
|
||||
/* vietnamese */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-LightItalic.ttf") format("truetype");
|
||||
unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1,
|
||||
U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329,
|
||||
U+1EA0-1EF9, U+20AB;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-LightItalic.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-LightItalic.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* cyrillic */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBoldItalic.ttf") format("truetype");
|
||||
unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||
}
|
||||
/* vietnamese */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBoldItalic.ttf") format("truetype");
|
||||
unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1,
|
||||
U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329,
|
||||
U+1EA0-1EF9, U+20AB;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBoldItalic.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: italic;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBoldItalic.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* cyrillic */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-Light.ttf") format("truetype");
|
||||
unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||
}
|
||||
/* vietnamese */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-Light.ttf") format("truetype");
|
||||
unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1,
|
||||
U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329,
|
||||
U+1EA0-1EF9, U+20AB;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-Light.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 300;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-Light.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
||||
/* cyrillic */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBold.ttf") format("truetype");
|
||||
unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
|
||||
}
|
||||
/* vietnamese */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBold.ttf") format("truetype");
|
||||
unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1,
|
||||
U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329,
|
||||
U+1EA0-1EF9, U+20AB;
|
||||
}
|
||||
/* latin-ext */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBold.ttf") format("truetype");
|
||||
unicode-range: U+0100-02AF, U+0304, U+0308, U+0329, U+1E00-1E9F, U+1EF2-1EFF,
|
||||
U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;
|
||||
}
|
||||
/* latin */
|
||||
@font-face {
|
||||
font-family: "Spectral";
|
||||
font-style: normal;
|
||||
font-weight: 600;
|
||||
font-display: swap;
|
||||
src: url("../fonts/Spectral/Spectral-SemiBold.ttf") format("truetype");
|
||||
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA,
|
||||
U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191,
|
||||
U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||
}
|
|
@ -1,751 +0,0 @@
|
|||
/*
|
||||
* Magick CSS
|
||||
* by: winterveil (https://github.com/wintermute-cell/)
|
||||
* license: MIT
|
||||
* version: 1.0.5
|
||||
*/
|
||||
|
||||
@charset "UTF-8";
|
||||
|
||||
/* Importing the fonts. */
|
||||
@import url("./fonts.css");
|
||||
|
||||
/* Simple CSS Reset */
|
||||
* {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
:root {
|
||||
/* Theme colors */
|
||||
--fg: #0e0e0e;
|
||||
--bg: #fefefe;
|
||||
--form-bg: #fbfbfb;
|
||||
--form-fg: #00004d;
|
||||
--form-fg-placeholder: #00004d9a;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
:root {
|
||||
--fg: #fefefe;
|
||||
--bg: #0e0e0e;
|
||||
--form-bg: #1a1a1a;
|
||||
--form-fg: #fefefe;
|
||||
--form-fg-placeholder: #fefefe9a;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
1) Modify the base font-size to 62.5% so that 1.6rem = 16px.
|
||||
2) Set box-sizing globally to handle padding and border widths.
|
||||
*/
|
||||
html {
|
||||
font-size: 62.5%; /* 1 */
|
||||
box-sizing: border-box; /* 2 */
|
||||
}
|
||||
|
||||
/* Use smaller sizes on mobile devices. */
|
||||
@media (max-width: 600px) {
|
||||
html {
|
||||
font-size: 56%;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set the base font-size to 18px with a normal weight.
|
||||
2) Set the text and background colors to match the theme.
|
||||
3) Use the 'Averia Serif Libre' font for the body text.
|
||||
4) Reset the counter for sidenotes.
|
||||
*/
|
||||
body {
|
||||
font-size: 1.8rem; /* 1 */
|
||||
background-color: var(--bg); /* 2 */
|
||||
color: var(--fg); /* 2 */
|
||||
font-family: "Averia Serif Libre", serif; /* 3 */
|
||||
font-style: normal; /* 3 */
|
||||
line-height: 2.2rem; /* 3 */
|
||||
font-weight: 300; /* 3 */
|
||||
|
||||
counter-reset: sidenote-counter; /* 4 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) Center the main content.
|
||||
2) Set the width of the element to 760px, with lower padding on mobile devices.
|
||||
3) Relative position as the default allows for absolute positioning of child elements.
|
||||
*/
|
||||
article,
|
||||
main {
|
||||
margin: auto; /* 1 */
|
||||
max-width: 76rem; /* 2 */
|
||||
padding: 0 1rem; /* 2 */
|
||||
width: 100%; /* 2 */
|
||||
position: relative; /* 3 */
|
||||
}
|
||||
|
||||
@media (max-width: 600px) {
|
||||
article,
|
||||
main {
|
||||
padding: 0 0.2rem; /* 2 */
|
||||
}
|
||||
}
|
||||
|
||||
/* ========================================================================================================================= */
|
||||
/* Structure & Layout ===================================================================================================== */
|
||||
/* ========================================================================================================================= */
|
||||
|
||||
/* Display the header, main, and footer sections as distinctly separate blocks. */
|
||||
header,
|
||||
section,
|
||||
footer {
|
||||
margin: 0.7rem;
|
||||
padding: 0.7rem;
|
||||
}
|
||||
|
||||
/* On mobile devices, a smaller margin looks more fitting due to the smaller view. */
|
||||
@media (max-width: 600px) {
|
||||
header,
|
||||
section,
|
||||
footer {
|
||||
margin-top: 0.2rem;
|
||||
margin-bottom: 0.2rem;
|
||||
}
|
||||
}
|
||||
|
||||
/* Avoid double margin on the last child of each section */
|
||||
header > *:last-child,
|
||||
section > *:last-child,
|
||||
footer > *:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
/* Add large margins to the header to visually separate it from the main content. */
|
||||
header {
|
||||
margin-top: 12rem;
|
||||
margin-bottom: 8rem;
|
||||
}
|
||||
|
||||
/* On mobile devices, reduce the margin around the header to save space. */
|
||||
@media (max-width: 600px) {
|
||||
header {
|
||||
margin: 0;
|
||||
padding: 1.2rem;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
1) Center align the text in the footer.
|
||||
2) Add a margin to the top of the footer to visually separate it from the main content.
|
||||
3) Add a margin to the bottom of the footer to not have it stuck to the bottom of the page.
|
||||
*/
|
||||
footer {
|
||||
text-align: center; /* 1 */
|
||||
margin-top: 5rem; /* 2 */
|
||||
margin-bottom: 2rem; /* 3 */
|
||||
}
|
||||
|
||||
/* ========================================================================================================================= */
|
||||
/* Typography & Links ===================================================================================================== */
|
||||
/* ========================================================================================================================= */
|
||||
|
||||
/* Add some space between paragraphs. */
|
||||
p {
|
||||
margin: 1.6rem 0; /* 1 */
|
||||
}
|
||||
|
||||
/* Remove link color. */
|
||||
a {
|
||||
color: var(--fg);
|
||||
}
|
||||
|
||||
/* Make string a little more bold, to adjust for the font. */
|
||||
b,
|
||||
strong {
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
/* Make emphasized text a little larger, to adjust for the font. */
|
||||
i,
|
||||
em {
|
||||
font-size: calc(1em + 0.1rem);
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set the font-family, color, and font-style for the headings.
|
||||
2) Add a margin to the top and bottom of the headings.
|
||||
*/
|
||||
h1,
|
||||
h2,
|
||||
h3,
|
||||
h4 {
|
||||
font-family: "Averia Libre", cursive; /* 1 */
|
||||
color: var(--fg); /* 1 */
|
||||
font-style: normal; /* 1 */
|
||||
font-weight: 600; /* 1 */
|
||||
margin: 1.6rem 0 1.6rem 0; /* 2 */
|
||||
}
|
||||
|
||||
/* h1 headings are uppercase and 2x the size of the base font. */
|
||||
h1 {
|
||||
text-transform: uppercase;
|
||||
font-size: 3.6rem;
|
||||
line-height: 3.3rem;
|
||||
}
|
||||
|
||||
/* h2 headings are uppercase and 1.250x (major third) the size of the base font. */
|
||||
h2 {
|
||||
font-size: 2.25rem;
|
||||
text-transform: uppercase;
|
||||
margin: 1.2rem 0 1.2rem 0;
|
||||
}
|
||||
|
||||
/* h3 headings are 1.125x (major second) the size of the base font. */
|
||||
h3 {
|
||||
font-size: 2.025rem;
|
||||
}
|
||||
|
||||
/* Add a decorative element before h3 headings. */
|
||||
h3:before {
|
||||
font-family: "Noto Sans Symbols 2", sans-serif;
|
||||
content: "🙛 ";
|
||||
}
|
||||
|
||||
/* h4 headings are the same size as h3 headings, but italic and without the decorative element. */
|
||||
h4 {
|
||||
font-style: italic;
|
||||
font-size: 2.025rem;
|
||||
}
|
||||
|
||||
/* A uniquely styled h1 for the header */
|
||||
header h1 {
|
||||
font-size: 4rem;
|
||||
color: var(--fg);
|
||||
text-align: center;
|
||||
padding: 4rem 0 1.2rem 0;
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
/* Additional decorations for the header h1 */
|
||||
header h1:before,
|
||||
header h1:after {
|
||||
content: "✦";
|
||||
color: var(--fg);
|
||||
font-size: 1.5rem;
|
||||
vertical-align: middle;
|
||||
padding: 0 0.5rem;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Remove any list symbols.
|
||||
2) Center align the nav links.
|
||||
3) Add a margin to the top of the nav links.
|
||||
4) Remove the default padding from the list.
|
||||
*/
|
||||
header nav ul {
|
||||
list-style-type: none; /* 1 */
|
||||
text-align: center; /* 2 */
|
||||
margin-top: 1rem; /* 3 */
|
||||
padding-inline-start: 0; /* 4 */
|
||||
}
|
||||
|
||||
/* Display the navigation links as a centered, horizontal list. */
|
||||
header nav ul li {
|
||||
display: inline;
|
||||
margin: 0 1.2rem;
|
||||
}
|
||||
|
||||
/* Remove default link styles. */
|
||||
header nav ul li a {
|
||||
text-decoration: none;
|
||||
color: var(--fg);
|
||||
}
|
||||
|
||||
/* Add a hover effect to the navigation links. */
|
||||
header nav ul li a::before {
|
||||
content: "❯ "; /* 1 */
|
||||
opacity: 0; /* 1 */
|
||||
}
|
||||
header nav ul li a:hover::before {
|
||||
opacity: 1; /* 1 */
|
||||
}
|
||||
|
||||
/* ===================================================================================================================== */
|
||||
/* Lists ===================================================================================================== */
|
||||
/* ===================================================================================================================== */
|
||||
|
||||
/*
|
||||
1) Add some indentation to the list items.
|
||||
2) Add a margin to the top and bottom of the list.
|
||||
*/
|
||||
:where(main ol, main ul) {
|
||||
margin-inline-start: 0; /* 1 */
|
||||
padding-inline-start: 3rem; /* 1 */
|
||||
margin: 0.8rem 0; /* 2 */
|
||||
}
|
||||
|
||||
/* Add some vertical space around a definition list. */
|
||||
dl {
|
||||
margin: 0.8rem 0;
|
||||
}
|
||||
|
||||
/* Add an indent to the definition term. */
|
||||
dd {
|
||||
margin: 0 1.6rem;
|
||||
}
|
||||
|
||||
/* ===================================================================================================================== */
|
||||
/* Media ===================================================================================================== */
|
||||
/* ===================================================================================================================== */
|
||||
|
||||
/*
|
||||
1) Set the maximum width of the image to 100% so they don't overflow the main column.
|
||||
2) Set the height in respect to the width to prevent the image from stretching.
|
||||
3) Add some margin to standalone images.
|
||||
*/
|
||||
img {
|
||||
max-width: 100%; /* 1 */
|
||||
height: auto; /* 2 */
|
||||
margin: 0.8rem 0; /* 3 */
|
||||
}
|
||||
|
||||
/* Images in figures should not have their own margins */
|
||||
figure img {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set margins and padding for figures.
|
||||
2) Center align any text inside figures.
|
||||
*/
|
||||
figure {
|
||||
margin: 2rem 0; /* 1 */
|
||||
padding: 0; /* 1 */
|
||||
text-align: center; /* 2 */
|
||||
}
|
||||
|
||||
/* Center align any element that is part of a figure */
|
||||
figure * {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
/* We don't want to center prealigned text or code in figures */
|
||||
figure code,
|
||||
figure pre {
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
/* Set the typography for the figure captions */
|
||||
figcaption {
|
||||
margin: 0.8rem 0;
|
||||
font-size: 1.8rem;
|
||||
}
|
||||
|
||||
/* ===================================================================================================================== */
|
||||
/* Forms & Inputs ===================================================================================================== */
|
||||
/* ===================================================================================================================== */
|
||||
|
||||
/*
|
||||
1) Display the form elements in a grid layout, two columns wide.
|
||||
2) Add some space between the form elements.
|
||||
3) Set padding and margin for the form.
|
||||
4) Give the form a pop out paper note look.
|
||||
*/
|
||||
form {
|
||||
display: grid; /* 1 */
|
||||
grid-template-columns: 1fr 1fr; /* 1 */
|
||||
gap: 10px; /* 2 */
|
||||
padding: 1rem; /* 3 */
|
||||
margin: 0.8rem 0; /* 3 */
|
||||
box-shadow: 0 0 10px 0 rgba(0, 0, 0, 0.1); /* 4 */
|
||||
background-color: var(--form-bg); /* 4 */
|
||||
}
|
||||
|
||||
/* Make form inputs and labels span two columns. (They get their own line) */
|
||||
form input[type="email"],
|
||||
form input[type="number"],
|
||||
form input[type="password"],
|
||||
form input[type="search"],
|
||||
form input[type="tel"],
|
||||
form input[type="text"],
|
||||
form input[type="url"],
|
||||
form label,
|
||||
form fieldset,
|
||||
form textarea {
|
||||
grid-column: span 2;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Adjust the elements to take up full width of their grid cell.
|
||||
2) Prevent textarea from being resized horizontally and overflowing the main column.
|
||||
*/
|
||||
form input,
|
||||
form button,
|
||||
form textarea {
|
||||
width: 100%; /* 1 */
|
||||
resize: vertical; /* 2 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) Prevent radio and checkbox from taking up full width, so they can be put next to each other.
|
||||
2) Add space between radio and checkbox options.
|
||||
*/
|
||||
input[type="radio"],
|
||||
input[type="checkbox"] {
|
||||
width: auto; /* 1 */
|
||||
margin-right: 0.5rem; /* 2 */
|
||||
}
|
||||
input[type="radio"] + label,
|
||||
input[type="checkbox"] + label {
|
||||
margin-right: 2rem; /* 2 */
|
||||
}
|
||||
|
||||
/* Add a disabled variant for radio and checkbox inputs */
|
||||
input[type="radio"]:disabled + label,
|
||||
input[type="checkbox"]:disabled + label {
|
||||
text-decoration: line-through;
|
||||
opacity: 0.5;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Remove the default appearance of the input.
|
||||
2) Set padding and margin for the input.
|
||||
3) Apply some custom styles in place of the default ones.
|
||||
4) Give the input a handwritten look.
|
||||
*/
|
||||
input[type="email"],
|
||||
input[type="number"],
|
||||
input[type="password"],
|
||||
input[type="search"],
|
||||
input[type="tel"],
|
||||
input[type="text"],
|
||||
input[type="url"],
|
||||
textarea,
|
||||
select {
|
||||
-webkit-appearance: none; /* 1 */
|
||||
-moz-appearance: none; /* 1 */
|
||||
appearance: none; /* 1 */
|
||||
box-shadow: none; /* 1 */
|
||||
box-sizing: inherit; /* 1 */
|
||||
border: none; /* 1 */
|
||||
|
||||
padding: 0.4rem 1rem; /* 2 */
|
||||
margin-bottom: 1.6rem; /* 2 */
|
||||
|
||||
font-size: 2rem; /* 3 */
|
||||
color: var(--fg); /* 3 */
|
||||
background-color: transparent; /* 3 */
|
||||
border-bottom: 1px solid var(--fg); /* 3 */
|
||||
border-radius: 0; /* 3 */
|
||||
font-size: 1.8rem; /* 3 */
|
||||
|
||||
font-family: "Caveat", cursive; /* 4 */
|
||||
font-size: 2.6rem; /* 4 */
|
||||
color: var(--form-fg); /* 4 */
|
||||
caret-color: var(--form-fg); /* 4 */
|
||||
}
|
||||
|
||||
/* Add disabled variant for input fields */
|
||||
input[type="email"]:disabled,
|
||||
input[type="number"]:disabled,
|
||||
input[type="password"]:disabled,
|
||||
input[type="search"]:disabled,
|
||||
input[type="tel"]:disabled,
|
||||
input[type="text"]:disabled,
|
||||
input[type="url"]:disabled,
|
||||
textarea:disabled,
|
||||
select:disabled {
|
||||
border-bottom: 1px dashed var(--fg); /* 3 */
|
||||
text-decoration: line-through;
|
||||
}
|
||||
|
||||
/* A slightly more transparent color for the placeholder text. */
|
||||
input::placeholder,
|
||||
textarea::placeholder {
|
||||
color: var(--form-fg-placeholder);
|
||||
}
|
||||
|
||||
/*
|
||||
1) Remove the default focus outline.
|
||||
2) Add a thicker bottom border to the input when focused, reducing margin to prevent layout shifting.
|
||||
*/
|
||||
input[type="email"]:focus,
|
||||
input[type="number"]:focus,
|
||||
input[type="password"]:focus,
|
||||
input[type="search"]:focus,
|
||||
input[type="tel"]:focus,
|
||||
input[type="text"]:focus,
|
||||
input[type="url"]:focus,
|
||||
textarea:focus,
|
||||
select:focus {
|
||||
outline: none; /* 1 */
|
||||
border-bottom: 2px solid var(--fg); /* 2 */
|
||||
margin-bottom: calc(1.6rem - 1px); /* 2 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) Match the theme colors.
|
||||
2) Add padding.
|
||||
3) Add a top margin to visually separate the buttons for the rest of the form.
|
||||
4) Add a thin border.
|
||||
5) Add a pointer cursor on hover.
|
||||
*/
|
||||
button,
|
||||
input[type="button"],
|
||||
input[type="reset"],
|
||||
input[type="submit"] {
|
||||
background-color: transparent; /* 1 */
|
||||
color: var(--fg); /* 1 */
|
||||
padding: 10px; /* 2 */
|
||||
margin-top: 1.6rem; /* 3 */
|
||||
border: 1px solid var(--fg); /* 4 */
|
||||
cursor: pointer; /* 5 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) A thin border around the fieldset.
|
||||
2) Set the width of the fieldset to fit around the content.
|
||||
*/
|
||||
fieldset {
|
||||
border: 1px solid var(--fg); /* 1 */
|
||||
width: fit-content; /* 2 */
|
||||
}
|
||||
|
||||
/* In a form, the fieldset takes up 100% of the width. */
|
||||
form fieldset {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set the border of a disabled button to be dashed.
|
||||
2) Add the not-allowed cursor when hovering a disabled button.
|
||||
*/
|
||||
button:disabled,
|
||||
input[type="button"]:disabled,
|
||||
input[type="reset"]:disabled,
|
||||
input[type="submit"]:disabled {
|
||||
opacity: 0.5;
|
||||
border-style: dashed; /* 1 */
|
||||
cursor: not-allowed; /* 2 */
|
||||
text-decoration: line-through;
|
||||
}
|
||||
|
||||
/* ===================================================================================================================== */
|
||||
/* Tables ========================================================================================================== */
|
||||
/* ===================================================================================================================== */
|
||||
|
||||
/* Remove the distance between adjacent cells, since we don't have vertical border lines. */
|
||||
table {
|
||||
border-spacing: 0;
|
||||
}
|
||||
|
||||
/* Add some padding around table cells. */
|
||||
td,
|
||||
th {
|
||||
padding: 0.4rem 1rem;
|
||||
}
|
||||
|
||||
/* Remove left padding for first cell in a row. */
|
||||
td:first-child,
|
||||
th:first-child {
|
||||
padding-left: 0;
|
||||
}
|
||||
|
||||
/* Remove right padding for last cell in a row. */
|
||||
td:last-child,
|
||||
th:last-child {
|
||||
padding-right: 0;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Add a border under the table header.
|
||||
2) Align the text to the left in the table header.
|
||||
*/
|
||||
th {
|
||||
border-bottom: 2px solid var(--fg); /* 1 */
|
||||
text-align: left; /* 2 */
|
||||
}
|
||||
|
||||
/* ============================================================================================================================ */
|
||||
/* Preformatting, Quotes & Code ============================================================================================ */
|
||||
/* ============================================================================================================================ */
|
||||
|
||||
/*
|
||||
1) Set custom padding and margins.
|
||||
2) Hide the vertical scroll bar.
|
||||
3) Set the width to fit just around the content, but limit it to 80% of the main column.
|
||||
4) Center the blockquote horizontally and add some vertical margins.
|
||||
*/
|
||||
blockquote {
|
||||
padding: 1rem 1.6rem; /* 1 */
|
||||
overflow-y: hidden; /* 2 */
|
||||
width: fit-content; /* 3 */
|
||||
max-width: 80%; /* 3 */
|
||||
margin: 2rem auto; /* 4 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set a custom font for blockquote text paragraphs.
|
||||
2) Add some space between the paragraphs.
|
||||
*/
|
||||
blockquote p {
|
||||
font-family: "Spectral", serif; /* 1 */
|
||||
font-style: italic; /* 1 */
|
||||
font-size: 2.1rem; /* 1 */
|
||||
font-weight: 300; /* 1 */
|
||||
line-height: 2.4rem; /* 1 */
|
||||
margin: 2.1rem 0; /* 2 */
|
||||
}
|
||||
|
||||
/* Add a footer to the blockquote for citations. */
|
||||
/*
|
||||
1) Reset any margins and padding from the main footer.
|
||||
2) Set the footer to float and align to the right.
|
||||
3) Limit the width of the footer to 55% of the main column.
|
||||
4) Set a custom font for the footer.
|
||||
*/
|
||||
blockquote footer {
|
||||
margin: 0; /* 1 */
|
||||
padding: 0; /* 1 */
|
||||
float: right; /* 2 */
|
||||
text-align: right; /* 2 */
|
||||
width: 55%; /* 3 */
|
||||
font-family: "Spectral", serif; /* 4 */
|
||||
font-style: normal; /* 4 */
|
||||
font-size: 1.4rem; /* 4 */
|
||||
}
|
||||
|
||||
/* Make the actual citation italic */
|
||||
blockquote footer cite {
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set custom padding.
|
||||
2) Hide the vertical scroll bar.
|
||||
3) Set a custom monospace font.
|
||||
4) Add a top and bottom border line.
|
||||
*/
|
||||
pre:has(code) {
|
||||
padding: 1rem 1.6rem; /* 1 */
|
||||
margin: 1.6rem 0; /* 1 */
|
||||
overflow-y: hidden; /* 2 */
|
||||
font-family: "Courier Prime", monospace; /* 3 */
|
||||
font-size: 1.6rem; /* 3 */
|
||||
border-top: 2px solid var(--fg); /* 4 */
|
||||
border-bottom: 2px solid var(--fg); /* 4 */
|
||||
}
|
||||
|
||||
/* Set a custom monospace font */
|
||||
code {
|
||||
font-family: "Courier Prime", monospace;
|
||||
font-size: 1.6rem;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Float the line numbers to the left, next to the code.
|
||||
2) Make some distance between the line numbers and the code, and pull it all to the left with a negative margin.
|
||||
3) Add a vertical line to separate the line numbers from the code.
|
||||
4) Align the line numbers against the separator.
|
||||
*/
|
||||
pre .line-number {
|
||||
float: left; /* 1 */
|
||||
margin: 0 1.5rem 0 -1.5rem; /* 2 */
|
||||
border-right: 1px solid; /* 3 */
|
||||
text-align: right; /* 4 */
|
||||
}
|
||||
|
||||
/*
|
||||
1) Display as block, so we get a new line for each line number.
|
||||
2) Add some padding to the line numbers.
|
||||
*/
|
||||
pre .line-number span {
|
||||
display: block; /* 1 */
|
||||
padding: 0 0.8rem 0 1.6rem; /* 2 */
|
||||
}
|
||||
|
||||
/* ============================================================================================================================ */
|
||||
/* Sidenotes & Asides ====================================================================================================== */
|
||||
/* ============================================================================================================================ */
|
||||
|
||||
/*
|
||||
1) On mobile devices, sidenotes behave the same as asides.
|
||||
2) Float the sidenotes to the right.
|
||||
3) Make sure the sidenotes don't clash.
|
||||
4) Set the width of the sidenotes to 40% of the main column.
|
||||
5) Add padding, margins and a border for better visual separation.
|
||||
6) Adjust typography to be more compact.
|
||||
*/
|
||||
.sidenote, /* 1 */
|
||||
aside {
|
||||
float: right; /* 2 */
|
||||
clear: right; /* 3 */
|
||||
width: 40%; /* 4 */
|
||||
margin: 1rem 1rem 1rem 3rem; /* 5 */
|
||||
padding: 0.5rem 0.5rem 0.5rem 2rem; /* 5 */
|
||||
border-left: 3px solid var(--fg); /* 5 */
|
||||
font-size: 1.4rem; /* 6 */
|
||||
line-height: 1.3; /* 6 */
|
||||
}
|
||||
|
||||
/* Prevent double top margins */
|
||||
aside h1,
|
||||
aside h2,
|
||||
aside h3,
|
||||
aside h4 {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
/*
|
||||
1) Set the distance from the main column.
|
||||
2) Set the width of the element to a little less than the remaining space on one side, limited to 40% of the main column.
|
||||
3) Remove any values set for the mobile version of the sidenotes.
|
||||
4) Set the width to the variable defined above.
|
||||
5) Set a negative right margin to (self-width + distance-from-main) to pull the sidenote to the right.
|
||||
*/
|
||||
/* Sadly, CSS does not support var() and rem in media queries, so we have to hardcode pixels. */
|
||||
/* This will break if the main column width is changed without adjusting this media query. */
|
||||
@media (min-width: calc(760px + 400px)) {
|
||||
.sidenote {
|
||||
--distance-from-main: 3rem; /* 1 */
|
||||
--self-width: min(
|
||||
calc((100vw - 760px) / 2 - (var(--distance-from-main))),
|
||||
40%
|
||||
); /* 2 */
|
||||
margin: 0; /* 3 */
|
||||
padding: 0; /* 3 */
|
||||
border: none; /* 3 */
|
||||
width: var(--self-width);
|
||||
margin-right: calc(
|
||||
calc(var(--self-width) + var(--distance-from-main)) * -1
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/* Each time a sidenote anchor is encountered, increment the counter */
|
||||
.sidenote-anchor {
|
||||
counter-increment: sidenote-counter;
|
||||
}
|
||||
|
||||
/* Use a custom font for the sidenote numbers */
|
||||
.sidenote-anchor:after,
|
||||
.sidenote:before {
|
||||
font-size: 1.3rem;
|
||||
position: relative;
|
||||
font-family: "Spectral", serif;
|
||||
}
|
||||
|
||||
/* Fine-adjust the number position for the sidenote anchor */
|
||||
.sidenote-anchor:after {
|
||||
content: counter(sidenote-counter);
|
||||
top: -0.5rem;
|
||||
left: 0.1rem;
|
||||
}
|
||||
|
||||
/* Fine-adjust the number position for the sidenote */
|
||||
.sidenote:before {
|
||||
content: counter(sidenote-counter) " ";
|
||||
top: -0.5rem;
|
||||
}
|
351
priv/assets/css/normalize.css
vendored
351
priv/assets/css/normalize.css
vendored
|
@ -1,351 +0,0 @@
|
|||
/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */
|
||||
|
||||
/* Document
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Correct the line height in all browsers.
|
||||
* 2. Prevent adjustments of font size after orientation changes in iOS.
|
||||
*/
|
||||
|
||||
html {
|
||||
line-height: 1.15; /* 1 */
|
||||
-webkit-text-size-adjust: 100%; /* 2 */
|
||||
}
|
||||
|
||||
/* Sections
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the margin in all browsers.
|
||||
*/
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render the `main` element consistently in IE.
|
||||
*/
|
||||
|
||||
main {
|
||||
display: block;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the font size and margin on `h1` elements within `section` and
|
||||
* `article` contexts in Chrome, Firefox, and Safari.
|
||||
*/
|
||||
|
||||
h1 {
|
||||
font-size: 2em;
|
||||
margin: 0.67em 0;
|
||||
}
|
||||
|
||||
/* Grouping content
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Add the correct box sizing in Firefox.
|
||||
* 2. Show the overflow in Edge and IE.
|
||||
*/
|
||||
|
||||
hr {
|
||||
box-sizing: content-box; /* 1 */
|
||||
height: 0; /* 1 */
|
||||
overflow: visible; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inheritance and scaling of font size in all browsers.
|
||||
* 2. Correct the odd `em` font sizing in all browsers.
|
||||
*/
|
||||
|
||||
pre {
|
||||
font-family: monospace, monospace; /* 1 */
|
||||
font-size: 1em; /* 2 */
|
||||
}
|
||||
|
||||
/* Text-level semantics
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the gray background on active links in IE 10.
|
||||
*/
|
||||
|
||||
a {
|
||||
background-color: transparent;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Remove the bottom border in Chrome 57-
|
||||
* 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.
|
||||
*/
|
||||
|
||||
abbr[title] {
|
||||
border-bottom: none; /* 1 */
|
||||
text-decoration: underline; /* 2 */
|
||||
text-decoration: underline dotted; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct font weight in Chrome, Edge, and Safari.
|
||||
*/
|
||||
|
||||
b,
|
||||
strong {
|
||||
font-weight: bolder;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inheritance and scaling of font size in all browsers.
|
||||
* 2. Correct the odd `em` font sizing in all browsers.
|
||||
*/
|
||||
|
||||
code,
|
||||
kbd,
|
||||
samp {
|
||||
font-family: monospace, monospace; /* 1 */
|
||||
font-size: 1em; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct font size in all browsers.
|
||||
*/
|
||||
|
||||
small {
|
||||
font-size: 80%;
|
||||
}
|
||||
|
||||
/**
|
||||
* Prevent `sub` and `sup` elements from affecting the line height in
|
||||
* all browsers.
|
||||
*/
|
||||
|
||||
sub,
|
||||
sup {
|
||||
font-size: 75%;
|
||||
line-height: 0;
|
||||
position: relative;
|
||||
vertical-align: baseline;
|
||||
}
|
||||
|
||||
sub {
|
||||
bottom: -0.25em;
|
||||
}
|
||||
|
||||
sup {
|
||||
top: -0.5em;
|
||||
}
|
||||
|
||||
/* Embedded content
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the border on images inside links in IE 10.
|
||||
*/
|
||||
|
||||
img {
|
||||
border-style: none;
|
||||
}
|
||||
|
||||
/* Forms
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Change the font styles in all browsers.
|
||||
* 2. Remove the margin in Firefox and Safari.
|
||||
*/
|
||||
|
||||
button,
|
||||
input,
|
||||
optgroup,
|
||||
select,
|
||||
textarea {
|
||||
font-family: inherit; /* 1 */
|
||||
font-size: 100%; /* 1 */
|
||||
line-height: 1.15; /* 1 */
|
||||
margin: 0; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the overflow in IE.
|
||||
* 1. Show the overflow in Edge.
|
||||
*/
|
||||
|
||||
button,
|
||||
input {
|
||||
/* 1 */
|
||||
overflow: visible;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inheritance of text transform in Edge, Firefox, and IE.
|
||||
* 1. Remove the inheritance of text transform in Firefox.
|
||||
*/
|
||||
|
||||
button,
|
||||
select {
|
||||
/* 1 */
|
||||
text-transform: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the inability to style clickable types in iOS and Safari.
|
||||
*/
|
||||
|
||||
button,
|
||||
[type="button"],
|
||||
[type="reset"],
|
||||
[type="submit"] {
|
||||
-webkit-appearance: button;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inner border and padding in Firefox.
|
||||
*/
|
||||
|
||||
button::-moz-focus-inner,
|
||||
[type="button"]::-moz-focus-inner,
|
||||
[type="reset"]::-moz-focus-inner,
|
||||
[type="submit"]::-moz-focus-inner {
|
||||
border-style: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Restore the focus styles unset by the previous rule.
|
||||
*/
|
||||
|
||||
button:-moz-focusring,
|
||||
[type="button"]:-moz-focusring,
|
||||
[type="reset"]:-moz-focusring,
|
||||
[type="submit"]:-moz-focusring {
|
||||
outline: 1px dotted ButtonText;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the padding in Firefox.
|
||||
*/
|
||||
|
||||
fieldset {
|
||||
padding: 0.35em 0.75em 0.625em;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the text wrapping in Edge and IE.
|
||||
* 2. Correct the color inheritance from `fieldset` elements in IE.
|
||||
* 3. Remove the padding so developers are not caught out when they zero out
|
||||
* `fieldset` elements in all browsers.
|
||||
*/
|
||||
|
||||
legend {
|
||||
box-sizing: border-box; /* 1 */
|
||||
color: inherit; /* 2 */
|
||||
display: table; /* 1 */
|
||||
max-width: 100%; /* 1 */
|
||||
padding: 0; /* 3 */
|
||||
white-space: normal; /* 1 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct vertical alignment in Chrome, Firefox, and Opera.
|
||||
*/
|
||||
|
||||
progress {
|
||||
vertical-align: baseline;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the default vertical scrollbar in IE 10+.
|
||||
*/
|
||||
|
||||
textarea {
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Add the correct box sizing in IE 10.
|
||||
* 2. Remove the padding in IE 10.
|
||||
*/
|
||||
|
||||
[type="checkbox"],
|
||||
[type="radio"] {
|
||||
box-sizing: border-box; /* 1 */
|
||||
padding: 0; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the cursor style of increment and decrement buttons in Chrome.
|
||||
*/
|
||||
|
||||
[type="number"]::-webkit-inner-spin-button,
|
||||
[type="number"]::-webkit-outer-spin-button {
|
||||
height: auto;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the odd appearance in Chrome and Safari.
|
||||
* 2. Correct the outline style in Safari.
|
||||
*/
|
||||
|
||||
[type="search"] {
|
||||
-webkit-appearance: textfield; /* 1 */
|
||||
outline-offset: -2px; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inner padding in Chrome and Safari on macOS.
|
||||
*/
|
||||
|
||||
[type="search"]::-webkit-search-decoration {
|
||||
-webkit-appearance: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inability to style clickable types in iOS and Safari.
|
||||
* 2. Change font properties to `inherit` in Safari.
|
||||
*/
|
||||
|
||||
::-webkit-file-upload-button {
|
||||
-webkit-appearance: button; /* 1 */
|
||||
font: inherit; /* 2 */
|
||||
}
|
||||
|
||||
/* Interactive
|
||||
========================================================================== */
|
||||
|
||||
/*
|
||||
* Add the correct display in Edge, IE 10+, and Firefox.
|
||||
*/
|
||||
|
||||
details {
|
||||
display: block;
|
||||
}
|
||||
|
||||
/*
|
||||
* Add the correct display in all browsers.
|
||||
*/
|
||||
|
||||
summary {
|
||||
display: list-item;
|
||||
}
|
||||
|
||||
/* Misc
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Add the correct display in IE 10+.
|
||||
*/
|
||||
|
||||
template {
|
||||
display: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct display in IE 10.
|
||||
*/
|
||||
|
||||
[hidden] {
|
||||
display: none;
|
||||
}
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -1,94 +0,0 @@
|
|||
Copyright (c) 2011, Dan Sayers (i@iotic.com),
|
||||
with Reserved Font Name 'Averia' and 'Averia Libre'.
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -1,94 +0,0 @@
|
|||
Copyright (c) 2011, Dan Sayers (i@iotic.com),
|
||||
with Reserved Font Name 'Averia' and 'Averia Libre'.
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
Binary file not shown.
|
@ -1,93 +0,0 @@
|
|||
Copyright 2014 The Caveat Project Authors (https://github.com/googlefonts/caveat)
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
|
@ -1,66 +0,0 @@
|
|||
Caveat Variable Font
|
||||
====================
|
||||
|
||||
This download contains Caveat as both a variable font and static fonts.
|
||||
|
||||
Caveat is a variable font with this axis:
|
||||
wght
|
||||
|
||||
This means all the styles are contained in a single file:
|
||||
Caveat-VariableFont_wght.ttf
|
||||
|
||||
If your app fully supports variable fonts, you can now pick intermediate styles
|
||||
that aren’t available as static fonts. Not all apps support variable fonts, and
|
||||
in those cases you can use the static font files for Caveat:
|
||||
static/Caveat-Regular.ttf
|
||||
static/Caveat-Medium.ttf
|
||||
static/Caveat-SemiBold.ttf
|
||||
static/Caveat-Bold.ttf
|
||||
|
||||
Get started
|
||||
-----------
|
||||
|
||||
1. Install the font files you want to use
|
||||
|
||||
2. Use your app's font picker to view the font family and all the
|
||||
available styles
|
||||
|
||||
Learn more about variable fonts
|
||||
-------------------------------
|
||||
|
||||
https://developers.google.com/web/fundamentals/design-and-ux/typography/variable-fonts
|
||||
https://variablefonts.typenetwork.com
|
||||
https://medium.com/variable-fonts
|
||||
|
||||
In desktop apps
|
||||
|
||||
https://theblog.adobe.com/can-variable-fonts-illustrator-cc
|
||||
https://helpx.adobe.com/nz/photoshop/using/fonts.html#variable_fonts
|
||||
|
||||
Online
|
||||
|
||||
https://developers.google.com/fonts/docs/getting_started
|
||||
https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Fonts/Variable_Fonts_Guide
|
||||
https://developer.microsoft.com/en-us/microsoft-edge/testdrive/demos/variable-fonts
|
||||
|
||||
Installing fonts
|
||||
|
||||
MacOS: https://support.apple.com/en-us/HT201749
|
||||
Linux: https://www.google.com/search?q=how+to+install+a+font+on+gnu%2Blinux
|
||||
Windows: https://support.microsoft.com/en-us/help/314960/how-to-install-or-remove-a-font-in-windows
|
||||
|
||||
Android Apps
|
||||
|
||||
https://developers.google.com/fonts/docs/android
|
||||
https://developer.android.com/guide/topics/ui/look-and-feel/downloadable-fonts
|
||||
|
||||
License
|
||||
-------
|
||||
Please read the full license text (OFL.txt) to understand the permissions,
|
||||
restrictions and requirements for usage, redistribution, and modification.
|
||||
|
||||
You can use them in your products & projects – print or digital,
|
||||
commercial or otherwise.
|
||||
|
||||
This isn't legal advice, please consider consulting a lawyer and see the full
|
||||
license for all details.
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -1,93 +0,0 @@
|
|||
Copyright 2015 The Courier Prime Project Authors (https://github.com/quoteunquoteapps/CourierPrime).
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
Binary file not shown.
|
@ -1,93 +0,0 @@
|
|||
Copyright 2022 The Noto Project Authors (https://github.com/notofonts/symbols)
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
|
@ -1,93 +0,0 @@
|
|||
Copyright 2017 The Spectral Project Authors (http://github.com/productiontype/spectral)
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
https://openfontlicense.org
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -1,263 +0,0 @@
|
|||
# SOME DESCRIPTIVE TITLE.
|
||||
# Copyright (C) YEAR Mikko Ahlroth
|
||||
# This file is distributed under the same license as the Scriptorium package.
|
||||
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
|
||||
#
|
||||
#, fuzzy
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Scriptorium 2.0.0\n"
|
||||
"Report-Msgid-Bugs-To: mikko@ahlroth.fi\n"
|
||||
"POT-Creation-Date: 2024-05-30 21:47+0300\n"
|
||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||
"Language: \n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
|
||||
#: src/scriptorium/parser.gleam:119
|
||||
msgid "Menu parsing failed: {err}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/renderer.gleam:177
|
||||
msgid "Archives for {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/renderer.gleam:206
|
||||
msgid "Archives for {month} {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:95
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:47
|
||||
msgid "Tags"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:103
|
||||
msgid "Archives"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/list_page.gleam:22
|
||||
msgid "Pages"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:20
|
||||
msgid "Previous"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:38
|
||||
msgid "current page"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:64
|
||||
msgid "Next"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:70
|
||||
msgid "Read more…"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:88
|
||||
msgid "{date}, {time}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:102
|
||||
msgid "Posted on {datetime}."
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:126
|
||||
msgid "January"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:127
|
||||
msgid "February"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:128
|
||||
msgid "March"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:129
|
||||
msgid "April"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:130 src/scriptorium/utils/date.gleam:148
|
||||
msgid "May"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:131
|
||||
msgid "June"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:132
|
||||
msgid "July"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:133
|
||||
msgid "August"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:134
|
||||
msgid "September"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:135
|
||||
msgid "October"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:136
|
||||
msgid "November"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:137
|
||||
msgid "December"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:144
|
||||
msgid "Jan"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:145
|
||||
msgid "Feb"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:146
|
||||
msgid "Mar"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:147
|
||||
msgid "Apr"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:149
|
||||
msgid "Jun"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:150
|
||||
msgid "Jul"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:151
|
||||
msgid "Aug"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:152
|
||||
msgid "Sep"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:153
|
||||
msgid "Oct"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:154
|
||||
msgid "Nov"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:155
|
||||
msgid "Dec"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:161
|
||||
msgid "{day} {month_short_str} {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:44
|
||||
msgid "{h24_0}:{min_0}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:77
|
||||
msgid "<00> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:78
|
||||
msgid "<01> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:79
|
||||
msgid "<02> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:80
|
||||
msgid "<03> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:81
|
||||
msgid "<04> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:82
|
||||
msgid "<05> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:83
|
||||
msgid "<06> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:84
|
||||
msgid "<07> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:85
|
||||
msgid "<08> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:86
|
||||
msgid "<09> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:87
|
||||
msgid "<10> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:88
|
||||
msgid "<11> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:89
|
||||
msgid "<12> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:90
|
||||
msgid "<13> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:91
|
||||
msgid "<14> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:92
|
||||
msgid "<15> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:93
|
||||
msgid "<16> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:94
|
||||
msgid "<17> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:95
|
||||
msgid "<18> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:96
|
||||
msgid "<19> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:97
|
||||
msgid "<20> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:98
|
||||
msgid "<21> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:99
|
||||
msgid "<22> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:100
|
||||
msgid "<23> pm"
|
||||
msgstr ""
|
Binary file not shown.
|
@ -1,264 +0,0 @@
|
|||
# SOME DESCRIPTIVE TITLE.
|
||||
# Copyright (C) YEAR Mikko Ahlroth
|
||||
# This file is distributed under the same license as the Scriptorium package.
|
||||
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
|
||||
#
|
||||
#, fuzzy
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Scriptorium 2.0.0\n"
|
||||
"Report-Msgid-Bugs-To: mikko@ahlroth.fi\n"
|
||||
"POT-Creation-Date: 2024-05-30 21:47+0300\n"
|
||||
"PO-Revision-Date: 2024-05-30 21:48+0300\n"
|
||||
"Last-Translator: \n"
|
||||
"Language-Team: \n"
|
||||
"Language: fi\n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"X-Generator: Poedit 3.4.4\n"
|
||||
|
||||
#: src/scriptorium/parser.gleam:119
|
||||
msgid "Menu parsing failed: {err}"
|
||||
msgstr "Valikon jäsentäminen epäonnistui: {err}"
|
||||
|
||||
#: src/scriptorium/renderer.gleam:177
|
||||
msgid "Archives for {year}"
|
||||
msgstr "Arkistot vuodelle {year}"
|
||||
|
||||
#: src/scriptorium/renderer.gleam:206
|
||||
msgid "Archives for {month} {year}"
|
||||
msgstr "Arkistot {month}lle {year}"
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:95
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:47
|
||||
msgid "Tags"
|
||||
msgstr "Tagit"
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:103
|
||||
msgid "Archives"
|
||||
msgstr "Arkistot"
|
||||
|
||||
#: src/scriptorium/rendering/views/list_page.gleam:22
|
||||
msgid "Pages"
|
||||
msgstr "Sivut"
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:20
|
||||
msgid "Previous"
|
||||
msgstr "Edellinen"
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:38
|
||||
msgid "current page"
|
||||
msgstr "tämä sivu"
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:64
|
||||
msgid "Next"
|
||||
msgstr "Seuraava"
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:70
|
||||
msgid "Read more…"
|
||||
msgstr "Lue lisää…"
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:88
|
||||
msgid "{date}, {time}"
|
||||
msgstr "{date} klo {time}"
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:102
|
||||
msgid "Posted on {datetime}."
|
||||
msgstr "Julkaistu {datetime}."
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:126
|
||||
msgid "January"
|
||||
msgstr "tammikuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:127
|
||||
msgid "February"
|
||||
msgstr "helmikuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:128
|
||||
msgid "March"
|
||||
msgstr "maaliskuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:129
|
||||
msgid "April"
|
||||
msgstr "huhtikuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:130 src/scriptorium/utils/date.gleam:148
|
||||
msgid "May"
|
||||
msgstr "toukokuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:131
|
||||
msgid "June"
|
||||
msgstr "kesäkuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:132
|
||||
msgid "July"
|
||||
msgstr "heinäkuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:133
|
||||
msgid "August"
|
||||
msgstr "elokuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:134
|
||||
msgid "September"
|
||||
msgstr "syyskuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:135
|
||||
msgid "October"
|
||||
msgstr "lokakuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:136
|
||||
msgid "November"
|
||||
msgstr "marraskuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:137
|
||||
msgid "December"
|
||||
msgstr "joulukuu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:144
|
||||
msgid "Jan"
|
||||
msgstr "tammi"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:145
|
||||
msgid "Feb"
|
||||
msgstr "helmi"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:146
|
||||
msgid "Mar"
|
||||
msgstr "maalis"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:147
|
||||
msgid "Apr"
|
||||
msgstr "huhti"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:149
|
||||
msgid "Jun"
|
||||
msgstr "touko"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:150
|
||||
msgid "Jul"
|
||||
msgstr "kesä"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:151
|
||||
msgid "Aug"
|
||||
msgstr "heinä"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:152
|
||||
msgid "Sep"
|
||||
msgstr "syys"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:153
|
||||
msgid "Oct"
|
||||
msgstr "loka"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:154
|
||||
msgid "Nov"
|
||||
msgstr "marras"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:155
|
||||
msgid "Dec"
|
||||
msgstr "joulu"
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:161
|
||||
msgid "{day} {month_short_str} {year}"
|
||||
msgstr "{day}.{month_no}.{year}"
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:44
|
||||
msgid "{h24_0}:{min_0}"
|
||||
msgstr "{h24_0}:{min_0}"
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:77
|
||||
msgid "<00> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:78
|
||||
msgid "<01> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:79
|
||||
msgid "<02> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:80
|
||||
msgid "<03> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:81
|
||||
msgid "<04> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:82
|
||||
msgid "<05> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:83
|
||||
msgid "<06> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:84
|
||||
msgid "<07> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:85
|
||||
msgid "<08> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:86
|
||||
msgid "<09> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:87
|
||||
msgid "<10> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:88
|
||||
msgid "<11> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:89
|
||||
msgid "<12> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:90
|
||||
msgid "<13> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:91
|
||||
msgid "<14> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:92
|
||||
msgid "<15> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:93
|
||||
msgid "<16> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:94
|
||||
msgid "<17> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:95
|
||||
msgid "<18> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:96
|
||||
msgid "<19> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:97
|
||||
msgid "<20> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:98
|
||||
msgid "<21> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:99
|
||||
msgid "<22> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:100
|
||||
msgid "<23> pm"
|
||||
msgstr ""
|
|
@ -1,263 +0,0 @@
|
|||
# SOME DESCRIPTIVE TITLE.
|
||||
# Copyright (C) YEAR Mikko Ahlroth
|
||||
# This file is distributed under the same license as the Scriptorium package.
|
||||
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
|
||||
#
|
||||
#, fuzzy
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Scriptorium 2.0.0\n"
|
||||
"Report-Msgid-Bugs-To: mikko@ahlroth.fi\n"
|
||||
"POT-Creation-Date: 2024-05-30 21:47+0300\n"
|
||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||
"Language: \n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
|
||||
#: src/scriptorium/parser.gleam:119
|
||||
msgid "Menu parsing failed: {err}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/renderer.gleam:177
|
||||
msgid "Archives for {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/renderer.gleam:206
|
||||
msgid "Archives for {month} {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:95
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:47
|
||||
msgid "Tags"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/base.gleam:103
|
||||
msgid "Archives"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/list_page.gleam:22
|
||||
msgid "Pages"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:20
|
||||
msgid "Previous"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:38
|
||||
msgid "current page"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/nav.gleam:64
|
||||
msgid "Next"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:70
|
||||
msgid "Read more…"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:88
|
||||
msgid "{date}, {time}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/rendering/views/single_post.gleam:102
|
||||
msgid "Posted on {datetime}."
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:126
|
||||
msgid "January"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:127
|
||||
msgid "February"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:128
|
||||
msgid "March"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:129
|
||||
msgid "April"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:130 src/scriptorium/utils/date.gleam:148
|
||||
msgid "May"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:131
|
||||
msgid "June"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:132
|
||||
msgid "July"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:133
|
||||
msgid "August"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:134
|
||||
msgid "September"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:135
|
||||
msgid "October"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:136
|
||||
msgid "November"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:137
|
||||
msgid "December"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:144
|
||||
msgid "Jan"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:145
|
||||
msgid "Feb"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:146
|
||||
msgid "Mar"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:147
|
||||
msgid "Apr"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:149
|
||||
msgid "Jun"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:150
|
||||
msgid "Jul"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:151
|
||||
msgid "Aug"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:152
|
||||
msgid "Sep"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:153
|
||||
msgid "Oct"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:154
|
||||
msgid "Nov"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:155
|
||||
msgid "Dec"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/date.gleam:161
|
||||
msgid "{day} {month_short_str} {year}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:44
|
||||
msgid "{h24_0}:{min_0}"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:77
|
||||
msgid "<00> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:78
|
||||
msgid "<01> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:79
|
||||
msgid "<02> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:80
|
||||
msgid "<03> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:81
|
||||
msgid "<04> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:82
|
||||
msgid "<05> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:83
|
||||
msgid "<06> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:84
|
||||
msgid "<07> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:85
|
||||
msgid "<08> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:86
|
||||
msgid "<09> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:87
|
||||
msgid "<10> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:88
|
||||
msgid "<11> am"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:89
|
||||
msgid "<12> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:90
|
||||
msgid "<13> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:91
|
||||
msgid "<14> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:92
|
||||
msgid "<15> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:93
|
||||
msgid "<16> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:94
|
||||
msgid "<17> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:95
|
||||
msgid "<18> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:96
|
||||
msgid "<19> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:97
|
||||
msgid "<20> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:98
|
||||
msgid "<21> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:99
|
||||
msgid "<22> pm"
|
||||
msgstr ""
|
||||
|
||||
#: src/scriptorium/utils/time.gleam:100
|
||||
msgid "<23> pm"
|
||||
msgstr ""
|
35
priv/templates/base.html.glemp
Normal file
35
priv/templates/base.html.glemp
Normal file
|
@ -0,0 +1,35 @@
|
|||
<!DOCTYPE html>
|
||||
<html>
|
||||
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<title>
|
||||
<%= title %>
|
||||
</title>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<header>
|
||||
<h1>
|
||||
<a href="<%= urls.base %><%= urls.index %>">
|
||||
<%= blog_name %>
|
||||
</a>
|
||||
</h1>
|
||||
</header>
|
||||
|
||||
<main>
|
||||
<%= raw inner_content %>
|
||||
</main>
|
||||
|
||||
<footer>
|
||||
<div>
|
||||
<%= copyright %>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
Powered by <a href="https://gitlab.com/Nicd/gloss">gloss</a>.
|
||||
</div>
|
||||
</footer>
|
||||
</body>
|
||||
|
||||
</html>
|
11
priv/templates/list.html.glemp
Normal file
11
priv/templates/list.html.glemp
Normal file
|
@ -0,0 +1,11 @@
|
|||
<section class="list">
|
||||
<% for post in posts %>
|
||||
<% render "list_post.html.glemp" post: post, urls: urls %>
|
||||
<% end %>
|
||||
|
||||
<footer>
|
||||
<nav>
|
||||
<% render "navigation.html.glemp" urls: urls, next: nav.next, previous: nav.previous %>
|
||||
</nav>
|
||||
</footer>
|
||||
</section>
|
21
priv/templates/list_post.html.glemp
Normal file
21
priv/templates/list_post.html.glemp
Normal file
|
@ -0,0 +1,21 @@
|
|||
<section class="post list-post">
|
||||
<header>
|
||||
<h2>
|
||||
<%= post.title %>
|
||||
</h2>
|
||||
</header>
|
||||
|
||||
<div>
|
||||
<% if post.short_content %>
|
||||
<%= raw post.short_content %>
|
||||
|
||||
<a href="<%= urls.base %><%= post.url %>">
|
||||
Read more…
|
||||
</a>
|
||||
<% else %>
|
||||
<%= raw post.content %>
|
||||
<% end %>
|
||||
</div>
|
||||
|
||||
<footer></footer>
|
||||
</section>
|
15
priv/templates/navigation.html.glemp
Normal file
15
priv/templates/navigation.html.glemp
Normal file
|
@ -0,0 +1,15 @@
|
|||
<div class="nav-previous">
|
||||
<% if previous %>
|
||||
<a href="<%= urls.base %><%= previous %>" aria-label="Previous">«</a>
|
||||
<% else %>
|
||||
<span class="nav-none" aria-hidden="true">«</span>
|
||||
<% end %>
|
||||
</div>
|
||||
|
||||
<div class="nav-next">
|
||||
<% if next %>
|
||||
<a href="<%= urls.base %><%= next %>" aria-label="Next">»</a>
|
||||
<% else %>
|
||||
<span class="nav-none" aria-hidden="true">»</span>
|
||||
<% end %>
|
||||
</div>
|
13
priv/templates/single_post.html.glemp
Normal file
13
priv/templates/single_post.html.glemp
Normal file
|
@ -0,0 +1,13 @@
|
|||
<section class="post single-post">
|
||||
<header>
|
||||
<h2>
|
||||
<%= title %>
|
||||
</h2>
|
||||
</header>
|
||||
|
||||
<div>
|
||||
<%= raw content %>
|
||||
</div>
|
||||
|
||||
<footer></footer>
|
||||
</section>
|
7
priv/vendor/luxon/LICENSE.md
vendored
7
priv/vendor/luxon/LICENSE.md
vendored
|
@ -1,7 +0,0 @@
|
|||
Copyright 2019 JS Foundation and other contributors
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
55
priv/vendor/luxon/README.md
vendored
55
priv/vendor/luxon/README.md
vendored
|
@ -1,55 +0,0 @@
|
|||
# Luxon
|
||||
|
||||
[![MIT License][license-image]][license] [![Build Status][github-action-image]][github-action-url] [![NPM version][npm-version-image]][npm-url] [![Coverage Status][test-coverage-image]][test-coverage-url] [![PRs welcome][contributing-image]][contributing-url]
|
||||
|
||||
Luxon is a library for working with dates and times in JavaScript.
|
||||
|
||||
```js
|
||||
DateTime.now().setZone("America/New_York").minus({ weeks: 1 }).endOf("day").toISO();
|
||||
```
|
||||
|
||||
## Upgrading to 3.0
|
||||
|
||||
[Guide](https://moment.github.io/luxon/#upgrading)
|
||||
|
||||
## Features
|
||||
* DateTime, Duration, and Interval types.
|
||||
* Immutable, chainable, unambiguous API.
|
||||
* Parsing and formatting for common and custom formats.
|
||||
* Native time zone and Intl support (no locale or tz files).
|
||||
|
||||
## Download/install
|
||||
|
||||
[Download/install instructions](https://moment.github.io/luxon/#/install)
|
||||
|
||||
## Documentation
|
||||
|
||||
* [General documentation](https://moment.github.io/luxon/#/?id=luxon)
|
||||
* [API docs](https://moment.github.io/luxon/api-docs/index.html)
|
||||
* [Quick tour](https://moment.github.io/luxon/#/tour)
|
||||
* [For Moment users](https://moment.github.io/luxon/#/moment)
|
||||
* [Why does Luxon exist?](https://moment.github.io/luxon/#/why)
|
||||
* [A quick demo](https://moment.github.io/luxon/demo/global.html)
|
||||
|
||||
## Development
|
||||
|
||||
See [contributing](CONTRIBUTING.md).
|
||||
|
||||
![Phasers to stun][phasers-image]
|
||||
|
||||
[license-image]: https://img.shields.io/badge/license-MIT-blue.svg
|
||||
[license]: LICENSE.md
|
||||
|
||||
[github-action-image]: https://github.com/moment/luxon/actions/workflows/test.yml/badge.svg
|
||||
[github-action-url]: https://github.com/moment/luxon/actions/workflows/test.yml
|
||||
|
||||
[npm-url]: https://npmjs.org/package/luxon
|
||||
[npm-version-image]: https://badge.fury.io/js/luxon.svg
|
||||
|
||||
[test-coverage-url]: https://codecov.io/gh/moment/luxon
|
||||
[test-coverage-image]: https://codecov.io/gh/moment/luxon/branch/master/graph/badge.svg
|
||||
|
||||
[contributing-url]: https://github.com/moment/luxon/blob/master/CONTRIBUTING.md
|
||||
[contributing-image]: https://img.shields.io/badge/PRs-welcome-brightgreen.svg
|
||||
|
||||
[phasers-image]: https://img.shields.io/badge/phasers-stun-brightgreen.svg
|
8311
priv/vendor/luxon/build/amd/luxon.js
vendored
8311
priv/vendor/luxon/build/amd/luxon.js
vendored
File diff suppressed because it is too large
Load diff
1
priv/vendor/luxon/build/amd/luxon.js.map
vendored
1
priv/vendor/luxon/build/amd/luxon.js.map
vendored
File diff suppressed because one or more lines are too long
8309
priv/vendor/luxon/build/cjs-browser/luxon.js
vendored
8309
priv/vendor/luxon/build/cjs-browser/luxon.js
vendored
File diff suppressed because it is too large
Load diff
File diff suppressed because one or more lines are too long
7695
priv/vendor/luxon/build/es6/luxon.js
vendored
7695
priv/vendor/luxon/build/es6/luxon.js
vendored
File diff suppressed because it is too large
Load diff
1
priv/vendor/luxon/build/es6/luxon.js.map
vendored
1
priv/vendor/luxon/build/es6/luxon.js.map
vendored
File diff suppressed because one or more lines are too long
8314
priv/vendor/luxon/build/global/luxon.js
vendored
8314
priv/vendor/luxon/build/global/luxon.js
vendored
File diff suppressed because it is too large
Load diff
1
priv/vendor/luxon/build/global/luxon.js.map
vendored
1
priv/vendor/luxon/build/global/luxon.js.map
vendored
File diff suppressed because one or more lines are too long
1
priv/vendor/luxon/build/global/luxon.min.js
vendored
1
priv/vendor/luxon/build/global/luxon.min.js
vendored
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
7384
priv/vendor/luxon/build/node/luxon.js
vendored
7384
priv/vendor/luxon/build/node/luxon.js
vendored
File diff suppressed because it is too large
Load diff
1
priv/vendor/luxon/build/node/luxon.js.map
vendored
1
priv/vendor/luxon/build/node/luxon.js.map
vendored
File diff suppressed because one or more lines are too long
87
priv/vendor/luxon/package.json
vendored
87
priv/vendor/luxon/package.json
vendored
|
@ -1,87 +0,0 @@
|
|||
{
|
||||
"name": "luxon",
|
||||
"version": "3.4.4",
|
||||
"description": "Immutable date wrapper",
|
||||
"author": "Isaac Cambron",
|
||||
"keywords": [
|
||||
"date",
|
||||
"immutable"
|
||||
],
|
||||
"repository": "https://github.com/moment/luxon",
|
||||
"exports": {
|
||||
".": {
|
||||
"import": "./src/luxon.js",
|
||||
"require": "./build/node/luxon.js"
|
||||
},
|
||||
"./package.json": "./package.json"
|
||||
},
|
||||
"scripts": {
|
||||
"build": "babel-node tasks/buildAll.js",
|
||||
"build-node": "babel-node tasks/buildNode.js",
|
||||
"build-global": "babel-node tasks/buildGlobal.js",
|
||||
"jest": "jest",
|
||||
"test": "jest --coverage",
|
||||
"api-docs": "mkdir -p build && documentation build src/luxon.js -f html -o build/api-docs && sed -i.bak 's/<\\/body>/<script src=\"\\..\\/global\\/luxon.js\"><\\/script><script>console.log(\"You can try Luxon right here using the `luxon` global, like `luxon.DateTime.now()`\");<\\/script><\\/body>/g' build/api-docs/index.html && rm build/api-docs/index.html.bak",
|
||||
"copy-site": "mkdir -p build && rsync -a docs/ build/docs && rsync -a site/ build",
|
||||
"site": "npm run api-docs && npm run copy-site",
|
||||
"format": "prettier --write 'src/**/*.js' 'test/**/*.js' 'benchmarks/*.js'",
|
||||
"format-check": "prettier --check 'src/**/*.js' 'test/**/*.js' 'benchmarks/*.js'",
|
||||
"benchmark": "babel-node benchmarks/index.js",
|
||||
"codecov": "codecov",
|
||||
"prepack": "babel-node tasks/buildAll.js",
|
||||
"prepare": "husky install",
|
||||
"show-site": "http-server build"
|
||||
},
|
||||
"lint-staged": {
|
||||
"*.{js,json}": [
|
||||
"prettier --write"
|
||||
]
|
||||
},
|
||||
"devDependencies": {
|
||||
"@babel/core": "^7.18.6",
|
||||
"@babel/node": "^7.18.6",
|
||||
"@babel/plugin-external-helpers": "^7.18.6",
|
||||
"@babel/preset-env": "^7.18.6",
|
||||
"@rollup/plugin-babel": "^5.3.0",
|
||||
"@rollup/plugin-commonjs": "^19.0.0",
|
||||
"@rollup/plugin-node-resolve": "^13.0.0",
|
||||
"babel-jest": "^28.1.2",
|
||||
"benchmark": "latest",
|
||||
"codecov": "latest",
|
||||
"documentation": "latest",
|
||||
"fs-extra": "^6.0.1",
|
||||
"http-server": "^14.1.1",
|
||||
"husky": "^7.0.0",
|
||||
"jest": "^29.4.3",
|
||||
"lint-staged": "^13.2.1",
|
||||
"prettier": "latest",
|
||||
"rollup": "^2.52.7",
|
||||
"rollup-plugin-terser": "^7.0.2",
|
||||
"uglify-js": "^3.13.10"
|
||||
},
|
||||
"main": "build/node/luxon.js",
|
||||
"module": "src/luxon.js",
|
||||
"browser": "build/cjs-browser/luxon.js",
|
||||
"jsdelivr": "build/global/luxon.min.js",
|
||||
"unpkg": "build/global/luxon.min.js",
|
||||
"engines": {
|
||||
"node": ">=12"
|
||||
},
|
||||
"files": [
|
||||
"build/node/luxon.js",
|
||||
"build/node/luxon.js.map",
|
||||
"build/cjs-browser/luxon.js",
|
||||
"build/cjs-browser/luxon.js.map",
|
||||
"build/amd/luxon.js",
|
||||
"build/amd/luxon.js.map",
|
||||
"build/global/luxon.js",
|
||||
"build/global/luxon.js.map",
|
||||
"build/global/luxon.min.js",
|
||||
"build/global/luxon.min.js.map",
|
||||
"build/es6/luxon.js",
|
||||
"build/es6/luxon.js.map",
|
||||
"src"
|
||||
],
|
||||
"license": "MIT",
|
||||
"sideEffects": false
|
||||
}
|
2422
priv/vendor/luxon/src/datetime.js
vendored
2422
priv/vendor/luxon/src/datetime.js
vendored
File diff suppressed because it is too large
Load diff
990
priv/vendor/luxon/src/duration.js
vendored
990
priv/vendor/luxon/src/duration.js
vendored
|
@ -1,990 +0,0 @@
|
|||
import { InvalidArgumentError, InvalidDurationError, InvalidUnitError } from "./errors.js";
|
||||
import Formatter from "./impl/formatter.js";
|
||||
import Invalid from "./impl/invalid.js";
|
||||
import Locale from "./impl/locale.js";
|
||||
import { parseISODuration, parseISOTimeOnly } from "./impl/regexParser.js";
|
||||
import {
|
||||
asNumber,
|
||||
hasOwnProperty,
|
||||
isNumber,
|
||||
isUndefined,
|
||||
normalizeObject,
|
||||
roundTo,
|
||||
} from "./impl/util.js";
|
||||
import Settings from "./settings.js";
|
||||
import DateTime from "./datetime.js";
|
||||
|
||||
const INVALID = "Invalid Duration";
|
||||
|
||||
// unit conversion constants
|
||||
export const lowOrderMatrix = {
|
||||
weeks: {
|
||||
days: 7,
|
||||
hours: 7 * 24,
|
||||
minutes: 7 * 24 * 60,
|
||||
seconds: 7 * 24 * 60 * 60,
|
||||
milliseconds: 7 * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
days: {
|
||||
hours: 24,
|
||||
minutes: 24 * 60,
|
||||
seconds: 24 * 60 * 60,
|
||||
milliseconds: 24 * 60 * 60 * 1000,
|
||||
},
|
||||
hours: { minutes: 60, seconds: 60 * 60, milliseconds: 60 * 60 * 1000 },
|
||||
minutes: { seconds: 60, milliseconds: 60 * 1000 },
|
||||
seconds: { milliseconds: 1000 },
|
||||
},
|
||||
casualMatrix = {
|
||||
years: {
|
||||
quarters: 4,
|
||||
months: 12,
|
||||
weeks: 52,
|
||||
days: 365,
|
||||
hours: 365 * 24,
|
||||
minutes: 365 * 24 * 60,
|
||||
seconds: 365 * 24 * 60 * 60,
|
||||
milliseconds: 365 * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
quarters: {
|
||||
months: 3,
|
||||
weeks: 13,
|
||||
days: 91,
|
||||
hours: 91 * 24,
|
||||
minutes: 91 * 24 * 60,
|
||||
seconds: 91 * 24 * 60 * 60,
|
||||
milliseconds: 91 * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
months: {
|
||||
weeks: 4,
|
||||
days: 30,
|
||||
hours: 30 * 24,
|
||||
minutes: 30 * 24 * 60,
|
||||
seconds: 30 * 24 * 60 * 60,
|
||||
milliseconds: 30 * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
|
||||
...lowOrderMatrix,
|
||||
},
|
||||
daysInYearAccurate = 146097.0 / 400,
|
||||
daysInMonthAccurate = 146097.0 / 4800,
|
||||
accurateMatrix = {
|
||||
years: {
|
||||
quarters: 4,
|
||||
months: 12,
|
||||
weeks: daysInYearAccurate / 7,
|
||||
days: daysInYearAccurate,
|
||||
hours: daysInYearAccurate * 24,
|
||||
minutes: daysInYearAccurate * 24 * 60,
|
||||
seconds: daysInYearAccurate * 24 * 60 * 60,
|
||||
milliseconds: daysInYearAccurate * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
quarters: {
|
||||
months: 3,
|
||||
weeks: daysInYearAccurate / 28,
|
||||
days: daysInYearAccurate / 4,
|
||||
hours: (daysInYearAccurate * 24) / 4,
|
||||
minutes: (daysInYearAccurate * 24 * 60) / 4,
|
||||
seconds: (daysInYearAccurate * 24 * 60 * 60) / 4,
|
||||
milliseconds: (daysInYearAccurate * 24 * 60 * 60 * 1000) / 4,
|
||||
},
|
||||
months: {
|
||||
weeks: daysInMonthAccurate / 7,
|
||||
days: daysInMonthAccurate,
|
||||
hours: daysInMonthAccurate * 24,
|
||||
minutes: daysInMonthAccurate * 24 * 60,
|
||||
seconds: daysInMonthAccurate * 24 * 60 * 60,
|
||||
milliseconds: daysInMonthAccurate * 24 * 60 * 60 * 1000,
|
||||
},
|
||||
...lowOrderMatrix,
|
||||
};
|
||||
|
||||
// units ordered by size
|
||||
const orderedUnits = [
|
||||
"years",
|
||||
"quarters",
|
||||
"months",
|
||||
"weeks",
|
||||
"days",
|
||||
"hours",
|
||||
"minutes",
|
||||
"seconds",
|
||||
"milliseconds",
|
||||
];
|
||||
|
||||
const reverseUnits = orderedUnits.slice(0).reverse();
|
||||
|
||||
// clone really means "create another instance just like this one, but with these changes"
|
||||
function clone(dur, alts, clear = false) {
|
||||
// deep merge for vals
|
||||
const conf = {
|
||||
values: clear ? alts.values : { ...dur.values, ...(alts.values || {}) },
|
||||
loc: dur.loc.clone(alts.loc),
|
||||
conversionAccuracy: alts.conversionAccuracy || dur.conversionAccuracy,
|
||||
matrix: alts.matrix || dur.matrix,
|
||||
};
|
||||
return new Duration(conf);
|
||||
}
|
||||
|
||||
function durationToMillis(matrix, vals) {
|
||||
let sum = vals.milliseconds ?? 0;
|
||||
for (const unit of reverseUnits.slice(1)) {
|
||||
if (vals[unit]) {
|
||||
sum += vals[unit] * matrix[unit]["milliseconds"];
|
||||
}
|
||||
}
|
||||
return sum;
|
||||
}
|
||||
|
||||
// NB: mutates parameters
|
||||
function normalizeValues(matrix, vals) {
|
||||
// the logic below assumes the overall value of the duration is positive
|
||||
// if this is not the case, factor is used to make it so
|
||||
const factor = durationToMillis(matrix, vals) < 0 ? -1 : 1;
|
||||
|
||||
orderedUnits.reduceRight((previous, current) => {
|
||||
if (!isUndefined(vals[current])) {
|
||||
if (previous) {
|
||||
const previousVal = vals[previous] * factor;
|
||||
const conv = matrix[current][previous];
|
||||
|
||||
// if (previousVal < 0):
|
||||
// lower order unit is negative (e.g. { years: 2, days: -2 })
|
||||
// normalize this by reducing the higher order unit by the appropriate amount
|
||||
// and increasing the lower order unit
|
||||
// this can never make the higher order unit negative, because this function only operates
|
||||
// on positive durations, so the amount of time represented by the lower order unit cannot
|
||||
// be larger than the higher order unit
|
||||
// else:
|
||||
// lower order unit is positive (e.g. { years: 2, days: 450 } or { years: -2, days: 450 })
|
||||
// in this case we attempt to convert as much as possible from the lower order unit into
|
||||
// the higher order one
|
||||
//
|
||||
// Math.floor takes care of both of these cases, rounding away from 0
|
||||
// if previousVal < 0 it makes the absolute value larger
|
||||
// if previousVal >= it makes the absolute value smaller
|
||||
const rollUp = Math.floor(previousVal / conv);
|
||||
vals[current] += rollUp * factor;
|
||||
vals[previous] -= rollUp * conv * factor;
|
||||
}
|
||||
return current;
|
||||
} else {
|
||||
return previous;
|
||||
}
|
||||
}, null);
|
||||
|
||||
// try to convert any decimals into smaller units if possible
|
||||
// for example for { years: 2.5, days: 0, seconds: 0 } we want to get { years: 2, days: 182, hours: 12 }
|
||||
orderedUnits.reduce((previous, current) => {
|
||||
if (!isUndefined(vals[current])) {
|
||||
if (previous) {
|
||||
const fraction = vals[previous] % 1;
|
||||
vals[previous] -= fraction;
|
||||
vals[current] += fraction * matrix[previous][current];
|
||||
}
|
||||
return current;
|
||||
} else {
|
||||
return previous;
|
||||
}
|
||||
}, null);
|
||||
}
|
||||
|
||||
// Remove all properties with a value of 0 from an object
|
||||
function removeZeroes(vals) {
|
||||
const newVals = {};
|
||||
for (const [key, value] of Object.entries(vals)) {
|
||||
if (value !== 0) {
|
||||
newVals[key] = value;
|
||||
}
|
||||
}
|
||||
return newVals;
|
||||
}
|
||||
|
||||
/**
|
||||
* A Duration object represents a period of time, like "2 months" or "1 day, 1 hour". Conceptually, it's just a map of units to their quantities, accompanied by some additional configuration and methods for creating, parsing, interrogating, transforming, and formatting them. They can be used on their own or in conjunction with other Luxon types; for example, you can use {@link DateTime#plus} to add a Duration object to a DateTime, producing another DateTime.
|
||||
*
|
||||
* Here is a brief overview of commonly used methods and getters in Duration:
|
||||
*
|
||||
* * **Creation** To create a Duration, use {@link Duration.fromMillis}, {@link Duration.fromObject}, or {@link Duration.fromISO}.
|
||||
* * **Unit values** See the {@link Duration#years}, {@link Duration#months}, {@link Duration#weeks}, {@link Duration#days}, {@link Duration#hours}, {@link Duration#minutes}, {@link Duration#seconds}, {@link Duration#milliseconds} accessors.
|
||||
* * **Configuration** See {@link Duration#locale} and {@link Duration#numberingSystem} accessors.
|
||||
* * **Transformation** To create new Durations out of old ones use {@link Duration#plus}, {@link Duration#minus}, {@link Duration#normalize}, {@link Duration#set}, {@link Duration#reconfigure}, {@link Duration#shiftTo}, and {@link Duration#negate}.
|
||||
* * **Output** To convert the Duration into other representations, see {@link Duration#as}, {@link Duration#toISO}, {@link Duration#toFormat}, and {@link Duration#toJSON}
|
||||
*
|
||||
* There's are more methods documented below. In addition, for more information on subtler topics like internationalization and validity, see the external documentation.
|
||||
*/
|
||||
export default class Duration {
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
constructor(config) {
|
||||
const accurate = config.conversionAccuracy === "longterm" || false;
|
||||
let matrix = accurate ? accurateMatrix : casualMatrix;
|
||||
|
||||
if (config.matrix) {
|
||||
matrix = config.matrix;
|
||||
}
|
||||
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.values = config.values;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.loc = config.loc || Locale.create();
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.conversionAccuracy = accurate ? "longterm" : "casual";
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.invalid = config.invalid || null;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.matrix = matrix;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.isLuxonDuration = true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create Duration from a number of milliseconds.
|
||||
* @param {number} count of milliseconds
|
||||
* @param {Object} opts - options for parsing
|
||||
* @param {string} [opts.locale='en-US'] - the locale to use
|
||||
* @param {string} opts.numberingSystem - the numbering system to use
|
||||
* @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use
|
||||
* @return {Duration}
|
||||
*/
|
||||
static fromMillis(count, opts) {
|
||||
return Duration.fromObject({ milliseconds: count }, opts);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Duration from a JavaScript object with keys like 'years' and 'hours'.
|
||||
* If this object is empty then a zero milliseconds duration is returned.
|
||||
* @param {Object} obj - the object to create the DateTime from
|
||||
* @param {number} obj.years
|
||||
* @param {number} obj.quarters
|
||||
* @param {number} obj.months
|
||||
* @param {number} obj.weeks
|
||||
* @param {number} obj.days
|
||||
* @param {number} obj.hours
|
||||
* @param {number} obj.minutes
|
||||
* @param {number} obj.seconds
|
||||
* @param {number} obj.milliseconds
|
||||
* @param {Object} [opts=[]] - options for creating this Duration
|
||||
* @param {string} [opts.locale='en-US'] - the locale to use
|
||||
* @param {string} opts.numberingSystem - the numbering system to use
|
||||
* @param {string} [opts.conversionAccuracy='casual'] - the preset conversion system to use
|
||||
* @param {string} [opts.matrix=Object] - the custom conversion system to use
|
||||
* @return {Duration}
|
||||
*/
|
||||
static fromObject(obj, opts = {}) {
|
||||
if (obj == null || typeof obj !== "object") {
|
||||
throw new InvalidArgumentError(
|
||||
`Duration.fromObject: argument expected to be an object, got ${
|
||||
obj === null ? "null" : typeof obj
|
||||
}`
|
||||
);
|
||||
}
|
||||
|
||||
return new Duration({
|
||||
values: normalizeObject(obj, Duration.normalizeUnit),
|
||||
loc: Locale.fromObject(opts),
|
||||
conversionAccuracy: opts.conversionAccuracy,
|
||||
matrix: opts.matrix,
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Duration from DurationLike.
|
||||
*
|
||||
* @param {Object | number | Duration} durationLike
|
||||
* One of:
|
||||
* - object with keys like 'years' and 'hours'.
|
||||
* - number representing milliseconds
|
||||
* - Duration instance
|
||||
* @return {Duration}
|
||||
*/
|
||||
static fromDurationLike(durationLike) {
|
||||
if (isNumber(durationLike)) {
|
||||
return Duration.fromMillis(durationLike);
|
||||
} else if (Duration.isDuration(durationLike)) {
|
||||
return durationLike;
|
||||
} else if (typeof durationLike === "object") {
|
||||
return Duration.fromObject(durationLike);
|
||||
} else {
|
||||
throw new InvalidArgumentError(
|
||||
`Unknown duration argument ${durationLike} of type ${typeof durationLike}`
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Duration from an ISO 8601 duration string.
|
||||
* @param {string} text - text to parse
|
||||
* @param {Object} opts - options for parsing
|
||||
* @param {string} [opts.locale='en-US'] - the locale to use
|
||||
* @param {string} opts.numberingSystem - the numbering system to use
|
||||
* @param {string} [opts.conversionAccuracy='casual'] - the preset conversion system to use
|
||||
* @param {string} [opts.matrix=Object] - the preset conversion system to use
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Durations
|
||||
* @example Duration.fromISO('P3Y6M1W4DT12H30M5S').toObject() //=> { years: 3, months: 6, weeks: 1, days: 4, hours: 12, minutes: 30, seconds: 5 }
|
||||
* @example Duration.fromISO('PT23H').toObject() //=> { hours: 23 }
|
||||
* @example Duration.fromISO('P5Y3M').toObject() //=> { years: 5, months: 3 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
static fromISO(text, opts) {
|
||||
const [parsed] = parseISODuration(text);
|
||||
if (parsed) {
|
||||
return Duration.fromObject(parsed, opts);
|
||||
} else {
|
||||
return Duration.invalid("unparsable", `the input "${text}" can't be parsed as ISO 8601`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Duration from an ISO 8601 time string.
|
||||
* @param {string} text - text to parse
|
||||
* @param {Object} opts - options for parsing
|
||||
* @param {string} [opts.locale='en-US'] - the locale to use
|
||||
* @param {string} opts.numberingSystem - the numbering system to use
|
||||
* @param {string} [opts.conversionAccuracy='casual'] - the preset conversion system to use
|
||||
* @param {string} [opts.matrix=Object] - the conversion system to use
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Times
|
||||
* @example Duration.fromISOTime('11:22:33.444').toObject() //=> { hours: 11, minutes: 22, seconds: 33, milliseconds: 444 }
|
||||
* @example Duration.fromISOTime('11:00').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }
|
||||
* @example Duration.fromISOTime('T11:00').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }
|
||||
* @example Duration.fromISOTime('1100').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }
|
||||
* @example Duration.fromISOTime('T1100').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
static fromISOTime(text, opts) {
|
||||
const [parsed] = parseISOTimeOnly(text);
|
||||
if (parsed) {
|
||||
return Duration.fromObject(parsed, opts);
|
||||
} else {
|
||||
return Duration.invalid("unparsable", `the input "${text}" can't be parsed as ISO 8601`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an invalid Duration.
|
||||
* @param {string} reason - simple string of why this datetime is invalid. Should not contain parameters or anything else data-dependent
|
||||
* @param {string} [explanation=null] - longer explanation, may include parameters and other useful debugging information
|
||||
* @return {Duration}
|
||||
*/
|
||||
static invalid(reason, explanation = null) {
|
||||
if (!reason) {
|
||||
throw new InvalidArgumentError("need to specify a reason the Duration is invalid");
|
||||
}
|
||||
|
||||
const invalid = reason instanceof Invalid ? reason : new Invalid(reason, explanation);
|
||||
|
||||
if (Settings.throwOnInvalid) {
|
||||
throw new InvalidDurationError(invalid);
|
||||
} else {
|
||||
return new Duration({ invalid });
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
static normalizeUnit(unit) {
|
||||
const normalized = {
|
||||
year: "years",
|
||||
years: "years",
|
||||
quarter: "quarters",
|
||||
quarters: "quarters",
|
||||
month: "months",
|
||||
months: "months",
|
||||
week: "weeks",
|
||||
weeks: "weeks",
|
||||
day: "days",
|
||||
days: "days",
|
||||
hour: "hours",
|
||||
hours: "hours",
|
||||
minute: "minutes",
|
||||
minutes: "minutes",
|
||||
second: "seconds",
|
||||
seconds: "seconds",
|
||||
millisecond: "milliseconds",
|
||||
milliseconds: "milliseconds",
|
||||
}[unit ? unit.toLowerCase() : unit];
|
||||
|
||||
if (!normalized) throw new InvalidUnitError(unit);
|
||||
|
||||
return normalized;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if an object is a Duration. Works across context boundaries
|
||||
* @param {object} o
|
||||
* @return {boolean}
|
||||
*/
|
||||
static isDuration(o) {
|
||||
return (o && o.isLuxonDuration) || false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the locale of a Duration, such 'en-GB'
|
||||
* @type {string}
|
||||
*/
|
||||
get locale() {
|
||||
return this.isValid ? this.loc.locale : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the numbering system of a Duration, such 'beng'. The numbering system is used when formatting the Duration
|
||||
*
|
||||
* @type {string}
|
||||
*/
|
||||
get numberingSystem() {
|
||||
return this.isValid ? this.loc.numberingSystem : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of this Duration formatted according to the specified format string. You may use these tokens:
|
||||
* * `S` for milliseconds
|
||||
* * `s` for seconds
|
||||
* * `m` for minutes
|
||||
* * `h` for hours
|
||||
* * `d` for days
|
||||
* * `w` for weeks
|
||||
* * `M` for months
|
||||
* * `y` for years
|
||||
* Notes:
|
||||
* * Add padding by repeating the token, e.g. "yy" pads the years to two digits, "hhhh" pads the hours out to four digits
|
||||
* * Tokens can be escaped by wrapping with single quotes.
|
||||
* * The duration will be converted to the set of units in the format string using {@link Duration#shiftTo} and the Durations's conversion accuracy setting.
|
||||
* @param {string} fmt - the format string
|
||||
* @param {Object} opts - options
|
||||
* @param {boolean} [opts.floor=true] - floor numerical values
|
||||
* @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat("y d s") //=> "1 6 2"
|
||||
* @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat("yy dd sss") //=> "01 06 002"
|
||||
* @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat("M S") //=> "12 518402000"
|
||||
* @return {string}
|
||||
*/
|
||||
toFormat(fmt, opts = {}) {
|
||||
// reverse-compat since 1.2; we always round down now, never up, and we do it by default
|
||||
const fmtOpts = {
|
||||
...opts,
|
||||
floor: opts.round !== false && opts.floor !== false,
|
||||
};
|
||||
return this.isValid
|
||||
? Formatter.create(this.loc, fmtOpts).formatDurationFromString(this, fmt)
|
||||
: INVALID;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of a Duration with all units included.
|
||||
* To modify its behavior, use `listStyle` and any Intl.NumberFormat option, though `unitDisplay` is especially relevant.
|
||||
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Intl/NumberFormat/NumberFormat#options
|
||||
* @param {Object} opts - Formatting options. Accepts the same keys as the options parameter of the native `Intl.NumberFormat` constructor, as well as `listStyle`.
|
||||
* @param {string} [opts.listStyle='narrow'] - How to format the merged list. Corresponds to the `style` property of the options parameter of the native `Intl.ListFormat` constructor.
|
||||
* @example
|
||||
* ```js
|
||||
* var dur = Duration.fromObject({ days: 1, hours: 5, minutes: 6 })
|
||||
* dur.toHuman() //=> '1 day, 5 hours, 6 minutes'
|
||||
* dur.toHuman({ listStyle: "long" }) //=> '1 day, 5 hours, and 6 minutes'
|
||||
* dur.toHuman({ unitDisplay: "short" }) //=> '1 day, 5 hr, 6 min'
|
||||
* ```
|
||||
*/
|
||||
toHuman(opts = {}) {
|
||||
if (!this.isValid) return INVALID;
|
||||
|
||||
const l = orderedUnits
|
||||
.map((unit) => {
|
||||
const val = this.values[unit];
|
||||
if (isUndefined(val)) {
|
||||
return null;
|
||||
}
|
||||
return this.loc
|
||||
.numberFormatter({ style: "unit", unitDisplay: "long", ...opts, unit: unit.slice(0, -1) })
|
||||
.format(val);
|
||||
})
|
||||
.filter((n) => n);
|
||||
|
||||
return this.loc
|
||||
.listFormatter({ type: "conjunction", style: opts.listStyle || "narrow", ...opts })
|
||||
.format(l);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a JavaScript object with this Duration's values.
|
||||
* @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toObject() //=> { years: 1, days: 6, seconds: 2 }
|
||||
* @return {Object}
|
||||
*/
|
||||
toObject() {
|
||||
if (!this.isValid) return {};
|
||||
return { ...this.values };
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601-compliant string representation of this Duration.
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Durations
|
||||
* @example Duration.fromObject({ years: 3, seconds: 45 }).toISO() //=> 'P3YT45S'
|
||||
* @example Duration.fromObject({ months: 4, seconds: 45 }).toISO() //=> 'P4MT45S'
|
||||
* @example Duration.fromObject({ months: 5 }).toISO() //=> 'P5M'
|
||||
* @example Duration.fromObject({ minutes: 5 }).toISO() //=> 'PT5M'
|
||||
* @example Duration.fromObject({ milliseconds: 6 }).toISO() //=> 'PT0.006S'
|
||||
* @return {string}
|
||||
*/
|
||||
toISO() {
|
||||
// we could use the formatter, but this is an easier way to get the minimum string
|
||||
if (!this.isValid) return null;
|
||||
|
||||
let s = "P";
|
||||
if (this.years !== 0) s += this.years + "Y";
|
||||
if (this.months !== 0 || this.quarters !== 0) s += this.months + this.quarters * 3 + "M";
|
||||
if (this.weeks !== 0) s += this.weeks + "W";
|
||||
if (this.days !== 0) s += this.days + "D";
|
||||
if (this.hours !== 0 || this.minutes !== 0 || this.seconds !== 0 || this.milliseconds !== 0)
|
||||
s += "T";
|
||||
if (this.hours !== 0) s += this.hours + "H";
|
||||
if (this.minutes !== 0) s += this.minutes + "M";
|
||||
if (this.seconds !== 0 || this.milliseconds !== 0)
|
||||
// this will handle "floating point madness" by removing extra decimal places
|
||||
// https://stackoverflow.com/questions/588004/is-floating-point-math-broken
|
||||
s += roundTo(this.seconds + this.milliseconds / 1000, 3) + "S";
|
||||
if (s === "P") s += "T0S";
|
||||
return s;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601-compliant string representation of this Duration, formatted as a time of day.
|
||||
* Note that this will return null if the duration is invalid, negative, or equal to or greater than 24 hours.
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Times
|
||||
* @param {Object} opts - options
|
||||
* @param {boolean} [opts.suppressMilliseconds=false] - exclude milliseconds from the format if they're 0
|
||||
* @param {boolean} [opts.suppressSeconds=false] - exclude seconds from the format if they're 0
|
||||
* @param {boolean} [opts.includePrefix=false] - include the `T` prefix
|
||||
* @param {string} [opts.format='extended'] - choose between the basic and extended format
|
||||
* @example Duration.fromObject({ hours: 11 }).toISOTime() //=> '11:00:00.000'
|
||||
* @example Duration.fromObject({ hours: 11 }).toISOTime({ suppressMilliseconds: true }) //=> '11:00:00'
|
||||
* @example Duration.fromObject({ hours: 11 }).toISOTime({ suppressSeconds: true }) //=> '11:00'
|
||||
* @example Duration.fromObject({ hours: 11 }).toISOTime({ includePrefix: true }) //=> 'T11:00:00.000'
|
||||
* @example Duration.fromObject({ hours: 11 }).toISOTime({ format: 'basic' }) //=> '110000.000'
|
||||
* @return {string}
|
||||
*/
|
||||
toISOTime(opts = {}) {
|
||||
if (!this.isValid) return null;
|
||||
|
||||
const millis = this.toMillis();
|
||||
if (millis < 0 || millis >= 86400000) return null;
|
||||
|
||||
opts = {
|
||||
suppressMilliseconds: false,
|
||||
suppressSeconds: false,
|
||||
includePrefix: false,
|
||||
format: "extended",
|
||||
...opts,
|
||||
includeOffset: false,
|
||||
};
|
||||
|
||||
const dateTime = DateTime.fromMillis(millis, { zone: "UTC" });
|
||||
return dateTime.toISOTime(opts);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601 representation of this Duration appropriate for use in JSON.
|
||||
* @return {string}
|
||||
*/
|
||||
toJSON() {
|
||||
return this.toISO();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601 representation of this Duration appropriate for use in debugging.
|
||||
* @return {string}
|
||||
*/
|
||||
toString() {
|
||||
return this.toISO();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of this Duration appropriate for the REPL.
|
||||
* @return {string}
|
||||
*/
|
||||
[Symbol.for("nodejs.util.inspect.custom")]() {
|
||||
if (this.isValid) {
|
||||
return `Duration { values: ${JSON.stringify(this.values)} }`;
|
||||
} else {
|
||||
return `Duration { Invalid, reason: ${this.invalidReason} }`;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an milliseconds value of this Duration.
|
||||
* @return {number}
|
||||
*/
|
||||
toMillis() {
|
||||
if (!this.isValid) return NaN;
|
||||
|
||||
return durationToMillis(this.matrix, this.values);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an milliseconds value of this Duration. Alias of {@link toMillis}
|
||||
* @return {number}
|
||||
*/
|
||||
valueOf() {
|
||||
return this.toMillis();
|
||||
}
|
||||
|
||||
/**
|
||||
* Make this Duration longer by the specified amount. Return a newly-constructed Duration.
|
||||
* @param {Duration|Object|number} duration - The amount to add. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()
|
||||
* @return {Duration}
|
||||
*/
|
||||
plus(duration) {
|
||||
if (!this.isValid) return this;
|
||||
|
||||
const dur = Duration.fromDurationLike(duration),
|
||||
result = {};
|
||||
|
||||
for (const k of orderedUnits) {
|
||||
if (hasOwnProperty(dur.values, k) || hasOwnProperty(this.values, k)) {
|
||||
result[k] = dur.get(k) + this.get(k);
|
||||
}
|
||||
}
|
||||
|
||||
return clone(this, { values: result }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Make this Duration shorter by the specified amount. Return a newly-constructed Duration.
|
||||
* @param {Duration|Object|number} duration - The amount to subtract. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()
|
||||
* @return {Duration}
|
||||
*/
|
||||
minus(duration) {
|
||||
if (!this.isValid) return this;
|
||||
|
||||
const dur = Duration.fromDurationLike(duration);
|
||||
return this.plus(dur.negate());
|
||||
}
|
||||
|
||||
/**
|
||||
* Scale this Duration by the specified amount. Return a newly-constructed Duration.
|
||||
* @param {function} fn - The function to apply to each unit. Arity is 1 or 2: the value of the unit and, optionally, the unit name. Must return a number.
|
||||
* @example Duration.fromObject({ hours: 1, minutes: 30 }).mapUnits(x => x * 2) //=> { hours: 2, minutes: 60 }
|
||||
* @example Duration.fromObject({ hours: 1, minutes: 30 }).mapUnits((x, u) => u === "hours" ? x * 2 : x) //=> { hours: 2, minutes: 30 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
mapUnits(fn) {
|
||||
if (!this.isValid) return this;
|
||||
const result = {};
|
||||
for (const k of Object.keys(this.values)) {
|
||||
result[k] = asNumber(fn(this.values[k], k));
|
||||
}
|
||||
return clone(this, { values: result }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the value of unit.
|
||||
* @param {string} unit - a unit such as 'minute' or 'day'
|
||||
* @example Duration.fromObject({years: 2, days: 3}).get('years') //=> 2
|
||||
* @example Duration.fromObject({years: 2, days: 3}).get('months') //=> 0
|
||||
* @example Duration.fromObject({years: 2, days: 3}).get('days') //=> 3
|
||||
* @return {number}
|
||||
*/
|
||||
get(unit) {
|
||||
return this[Duration.normalizeUnit(unit)];
|
||||
}
|
||||
|
||||
/**
|
||||
* "Set" the values of specified units. Return a newly-constructed Duration.
|
||||
* @param {Object} values - a mapping of units to numbers
|
||||
* @example dur.set({ years: 2017 })
|
||||
* @example dur.set({ hours: 8, minutes: 30 })
|
||||
* @return {Duration}
|
||||
*/
|
||||
set(values) {
|
||||
if (!this.isValid) return this;
|
||||
|
||||
const mixed = { ...this.values, ...normalizeObject(values, Duration.normalizeUnit) };
|
||||
return clone(this, { values: mixed });
|
||||
}
|
||||
|
||||
/**
|
||||
* "Set" the locale and/or numberingSystem. Returns a newly-constructed Duration.
|
||||
* @example dur.reconfigure({ locale: 'en-GB' })
|
||||
* @return {Duration}
|
||||
*/
|
||||
reconfigure({ locale, numberingSystem, conversionAccuracy, matrix } = {}) {
|
||||
const loc = this.loc.clone({ locale, numberingSystem });
|
||||
const opts = { loc, matrix, conversionAccuracy };
|
||||
return clone(this, opts);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the length of the duration in the specified unit.
|
||||
* @param {string} unit - a unit such as 'minutes' or 'days'
|
||||
* @example Duration.fromObject({years: 1}).as('days') //=> 365
|
||||
* @example Duration.fromObject({years: 1}).as('months') //=> 12
|
||||
* @example Duration.fromObject({hours: 60}).as('days') //=> 2.5
|
||||
* @return {number}
|
||||
*/
|
||||
as(unit) {
|
||||
return this.isValid ? this.shiftTo(unit).get(unit) : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Reduce this Duration to its canonical representation in its current units.
|
||||
* Assuming the overall value of the Duration is positive, this means:
|
||||
* - excessive values for lower-order units are converted to higher-order units (if possible, see first and second example)
|
||||
* - negative lower-order units are converted to higher order units (there must be such a higher order unit, otherwise
|
||||
* the overall value would be negative, see third example)
|
||||
* - fractional values for higher-order units are converted to lower-order units (if possible, see fourth example)
|
||||
*
|
||||
* If the overall value is negative, the result of this method is equivalent to `this.negate().normalize().negate()`.
|
||||
* @example Duration.fromObject({ years: 2, days: 5000 }).normalize().toObject() //=> { years: 15, days: 255 }
|
||||
* @example Duration.fromObject({ days: 5000 }).normalize().toObject() //=> { days: 5000 }
|
||||
* @example Duration.fromObject({ hours: 12, minutes: -45 }).normalize().toObject() //=> { hours: 11, minutes: 15 }
|
||||
* @example Duration.fromObject({ years: 2.5, days: 0, hours: 0 }).normalize().toObject() //=> { years: 2, days: 182, hours: 12 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
normalize() {
|
||||
if (!this.isValid) return this;
|
||||
const vals = this.toObject();
|
||||
normalizeValues(this.matrix, vals);
|
||||
return clone(this, { values: vals }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Rescale units to its largest representation
|
||||
* @example Duration.fromObject({ milliseconds: 90000 }).rescale().toObject() //=> { minutes: 1, seconds: 30 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
rescale() {
|
||||
if (!this.isValid) return this;
|
||||
const vals = removeZeroes(this.normalize().shiftToAll().toObject());
|
||||
return clone(this, { values: vals }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert this Duration into its representation in a different set of units.
|
||||
* @example Duration.fromObject({ hours: 1, seconds: 30 }).shiftTo('minutes', 'milliseconds').toObject() //=> { minutes: 60, milliseconds: 30000 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
shiftTo(...units) {
|
||||
if (!this.isValid) return this;
|
||||
|
||||
if (units.length === 0) {
|
||||
return this;
|
||||
}
|
||||
|
||||
units = units.map((u) => Duration.normalizeUnit(u));
|
||||
|
||||
const built = {},
|
||||
accumulated = {},
|
||||
vals = this.toObject();
|
||||
let lastUnit;
|
||||
|
||||
for (const k of orderedUnits) {
|
||||
if (units.indexOf(k) >= 0) {
|
||||
lastUnit = k;
|
||||
|
||||
let own = 0;
|
||||
|
||||
// anything we haven't boiled down yet should get boiled to this unit
|
||||
for (const ak in accumulated) {
|
||||
own += this.matrix[ak][k] * accumulated[ak];
|
||||
accumulated[ak] = 0;
|
||||
}
|
||||
|
||||
// plus anything that's already in this unit
|
||||
if (isNumber(vals[k])) {
|
||||
own += vals[k];
|
||||
}
|
||||
|
||||
// only keep the integer part for now in the hopes of putting any decimal part
|
||||
// into a smaller unit later
|
||||
const i = Math.trunc(own);
|
||||
built[k] = i;
|
||||
accumulated[k] = (own * 1000 - i * 1000) / 1000;
|
||||
|
||||
// otherwise, keep it in the wings to boil it later
|
||||
} else if (isNumber(vals[k])) {
|
||||
accumulated[k] = vals[k];
|
||||
}
|
||||
}
|
||||
|
||||
// anything leftover becomes the decimal for the last unit
|
||||
// lastUnit must be defined since units is not empty
|
||||
for (const key in accumulated) {
|
||||
if (accumulated[key] !== 0) {
|
||||
built[lastUnit] +=
|
||||
key === lastUnit ? accumulated[key] : accumulated[key] / this.matrix[lastUnit][key];
|
||||
}
|
||||
}
|
||||
|
||||
normalizeValues(this.matrix, built);
|
||||
return clone(this, { values: built }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Shift this Duration to all available units.
|
||||
* Same as shiftTo("years", "months", "weeks", "days", "hours", "minutes", "seconds", "milliseconds")
|
||||
* @return {Duration}
|
||||
*/
|
||||
shiftToAll() {
|
||||
if (!this.isValid) return this;
|
||||
return this.shiftTo(
|
||||
"years",
|
||||
"months",
|
||||
"weeks",
|
||||
"days",
|
||||
"hours",
|
||||
"minutes",
|
||||
"seconds",
|
||||
"milliseconds"
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the negative of this Duration.
|
||||
* @example Duration.fromObject({ hours: 1, seconds: 30 }).negate().toObject() //=> { hours: -1, seconds: -30 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
negate() {
|
||||
if (!this.isValid) return this;
|
||||
const negated = {};
|
||||
for (const k of Object.keys(this.values)) {
|
||||
negated[k] = this.values[k] === 0 ? 0 : -this.values[k];
|
||||
}
|
||||
return clone(this, { values: negated }, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the years.
|
||||
* @type {number}
|
||||
*/
|
||||
get years() {
|
||||
return this.isValid ? this.values.years || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the quarters.
|
||||
* @type {number}
|
||||
*/
|
||||
get quarters() {
|
||||
return this.isValid ? this.values.quarters || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the months.
|
||||
* @type {number}
|
||||
*/
|
||||
get months() {
|
||||
return this.isValid ? this.values.months || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the weeks
|
||||
* @type {number}
|
||||
*/
|
||||
get weeks() {
|
||||
return this.isValid ? this.values.weeks || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the days.
|
||||
* @type {number}
|
||||
*/
|
||||
get days() {
|
||||
return this.isValid ? this.values.days || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the hours.
|
||||
* @type {number}
|
||||
*/
|
||||
get hours() {
|
||||
return this.isValid ? this.values.hours || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the minutes.
|
||||
* @type {number}
|
||||
*/
|
||||
get minutes() {
|
||||
return this.isValid ? this.values.minutes || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the seconds.
|
||||
* @return {number}
|
||||
*/
|
||||
get seconds() {
|
||||
return this.isValid ? this.values.seconds || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the milliseconds.
|
||||
* @return {number}
|
||||
*/
|
||||
get milliseconds() {
|
||||
return this.isValid ? this.values.milliseconds || 0 : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether the Duration is invalid. Invalid durations are returned by diff operations
|
||||
* on invalid DateTimes or Intervals.
|
||||
* @return {boolean}
|
||||
*/
|
||||
get isValid() {
|
||||
return this.invalid === null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an error code if this Duration became invalid, or null if the Duration is valid
|
||||
* @return {string}
|
||||
*/
|
||||
get invalidReason() {
|
||||
return this.invalid ? this.invalid.reason : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an explanation of why this Duration became invalid, or null if the Duration is valid
|
||||
* @type {string}
|
||||
*/
|
||||
get invalidExplanation() {
|
||||
return this.invalid ? this.invalid.explanation : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Equality check
|
||||
* Two Durations are equal iff they have the same units and the same values for each unit.
|
||||
* @param {Duration} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
equals(other) {
|
||||
if (!this.isValid || !other.isValid) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!this.loc.equals(other.loc)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
function eq(v1, v2) {
|
||||
// Consider 0 and undefined as equal
|
||||
if (v1 === undefined || v1 === 0) return v2 === undefined || v2 === 0;
|
||||
return v1 === v2;
|
||||
}
|
||||
|
||||
for (const u of orderedUnits) {
|
||||
if (!eq(this.values[u], other.values[u])) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
61
priv/vendor/luxon/src/errors.js
vendored
61
priv/vendor/luxon/src/errors.js
vendored
|
@ -1,61 +0,0 @@
|
|||
// these aren't really private, but nor are they really useful to document
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
class LuxonError extends Error {}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class InvalidDateTimeError extends LuxonError {
|
||||
constructor(reason) {
|
||||
super(`Invalid DateTime: ${reason.toMessage()}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class InvalidIntervalError extends LuxonError {
|
||||
constructor(reason) {
|
||||
super(`Invalid Interval: ${reason.toMessage()}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class InvalidDurationError extends LuxonError {
|
||||
constructor(reason) {
|
||||
super(`Invalid Duration: ${reason.toMessage()}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class ConflictingSpecificationError extends LuxonError {}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class InvalidUnitError extends LuxonError {
|
||||
constructor(unit) {
|
||||
super(`Invalid unit ${unit}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class InvalidArgumentError extends LuxonError {}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
export class ZoneIsAbstractError extends LuxonError {
|
||||
constructor() {
|
||||
super("Zone is an abstract class");
|
||||
}
|
||||
}
|
206
priv/vendor/luxon/src/impl/conversions.js
vendored
206
priv/vendor/luxon/src/impl/conversions.js
vendored
|
@ -1,206 +0,0 @@
|
|||
import {
|
||||
integerBetween,
|
||||
isLeapYear,
|
||||
timeObject,
|
||||
daysInYear,
|
||||
daysInMonth,
|
||||
weeksInWeekYear,
|
||||
isInteger,
|
||||
isUndefined,
|
||||
} from "./util.js";
|
||||
import Invalid from "./invalid.js";
|
||||
import { ConflictingSpecificationError } from "../errors.js";
|
||||
|
||||
const nonLeapLadder = [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334],
|
||||
leapLadder = [0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335];
|
||||
|
||||
function unitOutOfRange(unit, value) {
|
||||
return new Invalid(
|
||||
"unit out of range",
|
||||
`you specified ${value} (of type ${typeof value}) as a ${unit}, which is invalid`
|
||||
);
|
||||
}
|
||||
|
||||
export function dayOfWeek(year, month, day) {
|
||||
const d = new Date(Date.UTC(year, month - 1, day));
|
||||
|
||||
if (year < 100 && year >= 0) {
|
||||
d.setUTCFullYear(d.getUTCFullYear() - 1900);
|
||||
}
|
||||
|
||||
const js = d.getUTCDay();
|
||||
|
||||
return js === 0 ? 7 : js;
|
||||
}
|
||||
|
||||
function computeOrdinal(year, month, day) {
|
||||
return day + (isLeapYear(year) ? leapLadder : nonLeapLadder)[month - 1];
|
||||
}
|
||||
|
||||
function uncomputeOrdinal(year, ordinal) {
|
||||
const table = isLeapYear(year) ? leapLadder : nonLeapLadder,
|
||||
month0 = table.findIndex((i) => i < ordinal),
|
||||
day = ordinal - table[month0];
|
||||
return { month: month0 + 1, day };
|
||||
}
|
||||
|
||||
export function isoWeekdayToLocal(isoWeekday, startOfWeek) {
|
||||
return ((isoWeekday - startOfWeek + 7) % 7) + 1;
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
export function gregorianToWeek(gregObj, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
||||
const { year, month, day } = gregObj,
|
||||
ordinal = computeOrdinal(year, month, day),
|
||||
weekday = isoWeekdayToLocal(dayOfWeek(year, month, day), startOfWeek);
|
||||
|
||||
let weekNumber = Math.floor((ordinal - weekday + 14 - minDaysInFirstWeek) / 7),
|
||||
weekYear;
|
||||
|
||||
if (weekNumber < 1) {
|
||||
weekYear = year - 1;
|
||||
weekNumber = weeksInWeekYear(weekYear, minDaysInFirstWeek, startOfWeek);
|
||||
} else if (weekNumber > weeksInWeekYear(year, minDaysInFirstWeek, startOfWeek)) {
|
||||
weekYear = year + 1;
|
||||
weekNumber = 1;
|
||||
} else {
|
||||
weekYear = year;
|
||||
}
|
||||
|
||||
return { weekYear, weekNumber, weekday, ...timeObject(gregObj) };
|
||||
}
|
||||
|
||||
export function weekToGregorian(weekData, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
||||
const { weekYear, weekNumber, weekday } = weekData,
|
||||
weekdayOfJan4 = isoWeekdayToLocal(dayOfWeek(weekYear, 1, minDaysInFirstWeek), startOfWeek),
|
||||
yearInDays = daysInYear(weekYear);
|
||||
|
||||
let ordinal = weekNumber * 7 + weekday - weekdayOfJan4 - 7 + minDaysInFirstWeek,
|
||||
year;
|
||||
|
||||
if (ordinal < 1) {
|
||||
year = weekYear - 1;
|
||||
ordinal += daysInYear(year);
|
||||
} else if (ordinal > yearInDays) {
|
||||
year = weekYear + 1;
|
||||
ordinal -= daysInYear(weekYear);
|
||||
} else {
|
||||
year = weekYear;
|
||||
}
|
||||
|
||||
const { month, day } = uncomputeOrdinal(year, ordinal);
|
||||
return { year, month, day, ...timeObject(weekData) };
|
||||
}
|
||||
|
||||
export function gregorianToOrdinal(gregData) {
|
||||
const { year, month, day } = gregData;
|
||||
const ordinal = computeOrdinal(year, month, day);
|
||||
return { year, ordinal, ...timeObject(gregData) };
|
||||
}
|
||||
|
||||
export function ordinalToGregorian(ordinalData) {
|
||||
const { year, ordinal } = ordinalData;
|
||||
const { month, day } = uncomputeOrdinal(year, ordinal);
|
||||
return { year, month, day, ...timeObject(ordinalData) };
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if local week units like localWeekday are used in obj.
|
||||
* If so, validates that they are not mixed with ISO week units and then copies them to the normal week unit properties.
|
||||
* Modifies obj in-place!
|
||||
* @param obj the object values
|
||||
*/
|
||||
export function usesLocalWeekValues(obj, loc) {
|
||||
const hasLocaleWeekData =
|
||||
!isUndefined(obj.localWeekday) ||
|
||||
!isUndefined(obj.localWeekNumber) ||
|
||||
!isUndefined(obj.localWeekYear);
|
||||
if (hasLocaleWeekData) {
|
||||
const hasIsoWeekData =
|
||||
!isUndefined(obj.weekday) || !isUndefined(obj.weekNumber) || !isUndefined(obj.weekYear);
|
||||
|
||||
if (hasIsoWeekData) {
|
||||
throw new ConflictingSpecificationError(
|
||||
"Cannot mix locale-based week fields with ISO-based week fields"
|
||||
);
|
||||
}
|
||||
if (!isUndefined(obj.localWeekday)) obj.weekday = obj.localWeekday;
|
||||
if (!isUndefined(obj.localWeekNumber)) obj.weekNumber = obj.localWeekNumber;
|
||||
if (!isUndefined(obj.localWeekYear)) obj.weekYear = obj.localWeekYear;
|
||||
delete obj.localWeekday;
|
||||
delete obj.localWeekNumber;
|
||||
delete obj.localWeekYear;
|
||||
return {
|
||||
minDaysInFirstWeek: loc.getMinDaysInFirstWeek(),
|
||||
startOfWeek: loc.getStartOfWeek(),
|
||||
};
|
||||
} else {
|
||||
return { minDaysInFirstWeek: 4, startOfWeek: 1 };
|
||||
}
|
||||
}
|
||||
|
||||
export function hasInvalidWeekData(obj, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
||||
const validYear = isInteger(obj.weekYear),
|
||||
validWeek = integerBetween(
|
||||
obj.weekNumber,
|
||||
1,
|
||||
weeksInWeekYear(obj.weekYear, minDaysInFirstWeek, startOfWeek)
|
||||
),
|
||||
validWeekday = integerBetween(obj.weekday, 1, 7);
|
||||
|
||||
if (!validYear) {
|
||||
return unitOutOfRange("weekYear", obj.weekYear);
|
||||
} else if (!validWeek) {
|
||||
return unitOutOfRange("week", obj.weekNumber);
|
||||
} else if (!validWeekday) {
|
||||
return unitOutOfRange("weekday", obj.weekday);
|
||||
} else return false;
|
||||
}
|
||||
|
||||
export function hasInvalidOrdinalData(obj) {
|
||||
const validYear = isInteger(obj.year),
|
||||
validOrdinal = integerBetween(obj.ordinal, 1, daysInYear(obj.year));
|
||||
|
||||
if (!validYear) {
|
||||
return unitOutOfRange("year", obj.year);
|
||||
} else if (!validOrdinal) {
|
||||
return unitOutOfRange("ordinal", obj.ordinal);
|
||||
} else return false;
|
||||
}
|
||||
|
||||
export function hasInvalidGregorianData(obj) {
|
||||
const validYear = isInteger(obj.year),
|
||||
validMonth = integerBetween(obj.month, 1, 12),
|
||||
validDay = integerBetween(obj.day, 1, daysInMonth(obj.year, obj.month));
|
||||
|
||||
if (!validYear) {
|
||||
return unitOutOfRange("year", obj.year);
|
||||
} else if (!validMonth) {
|
||||
return unitOutOfRange("month", obj.month);
|
||||
} else if (!validDay) {
|
||||
return unitOutOfRange("day", obj.day);
|
||||
} else return false;
|
||||
}
|
||||
|
||||
export function hasInvalidTimeData(obj) {
|
||||
const { hour, minute, second, millisecond } = obj;
|
||||
const validHour =
|
||||
integerBetween(hour, 0, 23) ||
|
||||
(hour === 24 && minute === 0 && second === 0 && millisecond === 0),
|
||||
validMinute = integerBetween(minute, 0, 59),
|
||||
validSecond = integerBetween(second, 0, 59),
|
||||
validMillisecond = integerBetween(millisecond, 0, 999);
|
||||
|
||||
if (!validHour) {
|
||||
return unitOutOfRange("hour", hour);
|
||||
} else if (!validMinute) {
|
||||
return unitOutOfRange("minute", minute);
|
||||
} else if (!validSecond) {
|
||||
return unitOutOfRange("second", second);
|
||||
} else if (!validMillisecond) {
|
||||
return unitOutOfRange("millisecond", millisecond);
|
||||
} else return false;
|
||||
}
|
95
priv/vendor/luxon/src/impl/diff.js
vendored
95
priv/vendor/luxon/src/impl/diff.js
vendored
|
@ -1,95 +0,0 @@
|
|||
import Duration from "../duration.js";
|
||||
|
||||
function dayDiff(earlier, later) {
|
||||
const utcDayStart = (dt) => dt.toUTC(0, { keepLocalTime: true }).startOf("day").valueOf(),
|
||||
ms = utcDayStart(later) - utcDayStart(earlier);
|
||||
return Math.floor(Duration.fromMillis(ms).as("days"));
|
||||
}
|
||||
|
||||
function highOrderDiffs(cursor, later, units) {
|
||||
const differs = [
|
||||
["years", (a, b) => b.year - a.year],
|
||||
["quarters", (a, b) => b.quarter - a.quarter + (b.year - a.year) * 4],
|
||||
["months", (a, b) => b.month - a.month + (b.year - a.year) * 12],
|
||||
[
|
||||
"weeks",
|
||||
(a, b) => {
|
||||
const days = dayDiff(a, b);
|
||||
return (days - (days % 7)) / 7;
|
||||
},
|
||||
],
|
||||
["days", dayDiff],
|
||||
];
|
||||
|
||||
const results = {};
|
||||
const earlier = cursor;
|
||||
let lowestOrder, highWater;
|
||||
|
||||
/* This loop tries to diff using larger units first.
|
||||
If we overshoot, we backtrack and try the next smaller unit.
|
||||
"cursor" starts out at the earlier timestamp and moves closer and closer to "later"
|
||||
as we use smaller and smaller units.
|
||||
highWater keeps track of where we would be if we added one more of the smallest unit,
|
||||
this is used later to potentially convert any difference smaller than the smallest higher order unit
|
||||
into a fraction of that smallest higher order unit
|
||||
*/
|
||||
for (const [unit, differ] of differs) {
|
||||
if (units.indexOf(unit) >= 0) {
|
||||
lowestOrder = unit;
|
||||
|
||||
results[unit] = differ(cursor, later);
|
||||
highWater = earlier.plus(results);
|
||||
|
||||
if (highWater > later) {
|
||||
// we overshot the end point, backtrack cursor by 1
|
||||
results[unit]--;
|
||||
cursor = earlier.plus(results);
|
||||
|
||||
// if we are still overshooting now, we need to backtrack again
|
||||
// this happens in certain situations when diffing times in different zones,
|
||||
// because this calculation ignores time zones
|
||||
if (cursor > later) {
|
||||
// keep the "overshot by 1" around as highWater
|
||||
highWater = cursor;
|
||||
// backtrack cursor by 1
|
||||
results[unit]--;
|
||||
cursor = earlier.plus(results);
|
||||
}
|
||||
} else {
|
||||
cursor = highWater;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return [cursor, results, highWater, lowestOrder];
|
||||
}
|
||||
|
||||
export default function (earlier, later, units, opts) {
|
||||
let [cursor, results, highWater, lowestOrder] = highOrderDiffs(earlier, later, units);
|
||||
|
||||
const remainingMillis = later - cursor;
|
||||
|
||||
const lowerOrderUnits = units.filter(
|
||||
(u) => ["hours", "minutes", "seconds", "milliseconds"].indexOf(u) >= 0
|
||||
);
|
||||
|
||||
if (lowerOrderUnits.length === 0) {
|
||||
if (highWater < later) {
|
||||
highWater = cursor.plus({ [lowestOrder]: 1 });
|
||||
}
|
||||
|
||||
if (highWater !== cursor) {
|
||||
results[lowestOrder] = (results[lowestOrder] || 0) + remainingMillis / (highWater - cursor);
|
||||
}
|
||||
}
|
||||
|
||||
const duration = Duration.fromObject(results, opts);
|
||||
|
||||
if (lowerOrderUnits.length > 0) {
|
||||
return Duration.fromMillis(remainingMillis, opts)
|
||||
.shiftTo(...lowerOrderUnits)
|
||||
.plus(duration);
|
||||
} else {
|
||||
return duration;
|
||||
}
|
||||
}
|
75
priv/vendor/luxon/src/impl/digits.js
vendored
75
priv/vendor/luxon/src/impl/digits.js
vendored
|
@ -1,75 +0,0 @@
|
|||
const numberingSystems = {
|
||||
arab: "[\u0660-\u0669]",
|
||||
arabext: "[\u06F0-\u06F9]",
|
||||
bali: "[\u1B50-\u1B59]",
|
||||
beng: "[\u09E6-\u09EF]",
|
||||
deva: "[\u0966-\u096F]",
|
||||
fullwide: "[\uFF10-\uFF19]",
|
||||
gujr: "[\u0AE6-\u0AEF]",
|
||||
hanidec: "[〇|一|二|三|四|五|六|七|八|九]",
|
||||
khmr: "[\u17E0-\u17E9]",
|
||||
knda: "[\u0CE6-\u0CEF]",
|
||||
laoo: "[\u0ED0-\u0ED9]",
|
||||
limb: "[\u1946-\u194F]",
|
||||
mlym: "[\u0D66-\u0D6F]",
|
||||
mong: "[\u1810-\u1819]",
|
||||
mymr: "[\u1040-\u1049]",
|
||||
orya: "[\u0B66-\u0B6F]",
|
||||
tamldec: "[\u0BE6-\u0BEF]",
|
||||
telu: "[\u0C66-\u0C6F]",
|
||||
thai: "[\u0E50-\u0E59]",
|
||||
tibt: "[\u0F20-\u0F29]",
|
||||
latn: "\\d",
|
||||
};
|
||||
|
||||
const numberingSystemsUTF16 = {
|
||||
arab: [1632, 1641],
|
||||
arabext: [1776, 1785],
|
||||
bali: [6992, 7001],
|
||||
beng: [2534, 2543],
|
||||
deva: [2406, 2415],
|
||||
fullwide: [65296, 65303],
|
||||
gujr: [2790, 2799],
|
||||
khmr: [6112, 6121],
|
||||
knda: [3302, 3311],
|
||||
laoo: [3792, 3801],
|
||||
limb: [6470, 6479],
|
||||
mlym: [3430, 3439],
|
||||
mong: [6160, 6169],
|
||||
mymr: [4160, 4169],
|
||||
orya: [2918, 2927],
|
||||
tamldec: [3046, 3055],
|
||||
telu: [3174, 3183],
|
||||
thai: [3664, 3673],
|
||||
tibt: [3872, 3881],
|
||||
};
|
||||
|
||||
const hanidecChars = numberingSystems.hanidec.replace(/[\[|\]]/g, "").split("");
|
||||
|
||||
export function parseDigits(str) {
|
||||
let value = parseInt(str, 10);
|
||||
if (isNaN(value)) {
|
||||
value = "";
|
||||
for (let i = 0; i < str.length; i++) {
|
||||
const code = str.charCodeAt(i);
|
||||
|
||||
if (str[i].search(numberingSystems.hanidec) !== -1) {
|
||||
value += hanidecChars.indexOf(str[i]);
|
||||
} else {
|
||||
for (const key in numberingSystemsUTF16) {
|
||||
const [min, max] = numberingSystemsUTF16[key];
|
||||
if (code >= min && code <= max) {
|
||||
value += code - min;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return parseInt(value, 10);
|
||||
} else {
|
||||
return value;
|
||||
}
|
||||
}
|
||||
|
||||
export function digitRegex({ numberingSystem }, append = "") {
|
||||
return new RegExp(`${numberingSystems[numberingSystem || "latn"]}${append}`);
|
||||
}
|
233
priv/vendor/luxon/src/impl/english.js
vendored
233
priv/vendor/luxon/src/impl/english.js
vendored
|
@ -1,233 +0,0 @@
|
|||
import * as Formats from "./formats.js";
|
||||
import { pick } from "./util.js";
|
||||
|
||||
function stringify(obj) {
|
||||
return JSON.stringify(obj, Object.keys(obj).sort());
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
export const monthsLong = [
|
||||
"January",
|
||||
"February",
|
||||
"March",
|
||||
"April",
|
||||
"May",
|
||||
"June",
|
||||
"July",
|
||||
"August",
|
||||
"September",
|
||||
"October",
|
||||
"November",
|
||||
"December",
|
||||
];
|
||||
|
||||
export const monthsShort = [
|
||||
"Jan",
|
||||
"Feb",
|
||||
"Mar",
|
||||
"Apr",
|
||||
"May",
|
||||
"Jun",
|
||||
"Jul",
|
||||
"Aug",
|
||||
"Sep",
|
||||
"Oct",
|
||||
"Nov",
|
||||
"Dec",
|
||||
];
|
||||
|
||||
export const monthsNarrow = ["J", "F", "M", "A", "M", "J", "J", "A", "S", "O", "N", "D"];
|
||||
|
||||
export function months(length) {
|
||||
switch (length) {
|
||||
case "narrow":
|
||||
return [...monthsNarrow];
|
||||
case "short":
|
||||
return [...monthsShort];
|
||||
case "long":
|
||||
return [...monthsLong];
|
||||
case "numeric":
|
||||
return ["1", "2", "3", "4", "5", "6", "7", "8", "9", "10", "11", "12"];
|
||||
case "2-digit":
|
||||
return ["01", "02", "03", "04", "05", "06", "07", "08", "09", "10", "11", "12"];
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export const weekdaysLong = [
|
||||
"Monday",
|
||||
"Tuesday",
|
||||
"Wednesday",
|
||||
"Thursday",
|
||||
"Friday",
|
||||
"Saturday",
|
||||
"Sunday",
|
||||
];
|
||||
|
||||
export const weekdaysShort = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"];
|
||||
|
||||
export const weekdaysNarrow = ["M", "T", "W", "T", "F", "S", "S"];
|
||||
|
||||
export function weekdays(length) {
|
||||
switch (length) {
|
||||
case "narrow":
|
||||
return [...weekdaysNarrow];
|
||||
case "short":
|
||||
return [...weekdaysShort];
|
||||
case "long":
|
||||
return [...weekdaysLong];
|
||||
case "numeric":
|
||||
return ["1", "2", "3", "4", "5", "6", "7"];
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export const meridiems = ["AM", "PM"];
|
||||
|
||||
export const erasLong = ["Before Christ", "Anno Domini"];
|
||||
|
||||
export const erasShort = ["BC", "AD"];
|
||||
|
||||
export const erasNarrow = ["B", "A"];
|
||||
|
||||
export function eras(length) {
|
||||
switch (length) {
|
||||
case "narrow":
|
||||
return [...erasNarrow];
|
||||
case "short":
|
||||
return [...erasShort];
|
||||
case "long":
|
||||
return [...erasLong];
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export function meridiemForDateTime(dt) {
|
||||
return meridiems[dt.hour < 12 ? 0 : 1];
|
||||
}
|
||||
|
||||
export function weekdayForDateTime(dt, length) {
|
||||
return weekdays(length)[dt.weekday - 1];
|
||||
}
|
||||
|
||||
export function monthForDateTime(dt, length) {
|
||||
return months(length)[dt.month - 1];
|
||||
}
|
||||
|
||||
export function eraForDateTime(dt, length) {
|
||||
return eras(length)[dt.year < 0 ? 0 : 1];
|
||||
}
|
||||
|
||||
export function formatRelativeTime(unit, count, numeric = "always", narrow = false) {
|
||||
const units = {
|
||||
years: ["year", "yr."],
|
||||
quarters: ["quarter", "qtr."],
|
||||
months: ["month", "mo."],
|
||||
weeks: ["week", "wk."],
|
||||
days: ["day", "day", "days"],
|
||||
hours: ["hour", "hr."],
|
||||
minutes: ["minute", "min."],
|
||||
seconds: ["second", "sec."],
|
||||
};
|
||||
|
||||
const lastable = ["hours", "minutes", "seconds"].indexOf(unit) === -1;
|
||||
|
||||
if (numeric === "auto" && lastable) {
|
||||
const isDay = unit === "days";
|
||||
switch (count) {
|
||||
case 1:
|
||||
return isDay ? "tomorrow" : `next ${units[unit][0]}`;
|
||||
case -1:
|
||||
return isDay ? "yesterday" : `last ${units[unit][0]}`;
|
||||
case 0:
|
||||
return isDay ? "today" : `this ${units[unit][0]}`;
|
||||
default: // fall through
|
||||
}
|
||||
}
|
||||
|
||||
const isInPast = Object.is(count, -0) || count < 0,
|
||||
fmtValue = Math.abs(count),
|
||||
singular = fmtValue === 1,
|
||||
lilUnits = units[unit],
|
||||
fmtUnit = narrow
|
||||
? singular
|
||||
? lilUnits[1]
|
||||
: lilUnits[2] || lilUnits[1]
|
||||
: singular
|
||||
? units[unit][0]
|
||||
: unit;
|
||||
return isInPast ? `${fmtValue} ${fmtUnit} ago` : `in ${fmtValue} ${fmtUnit}`;
|
||||
}
|
||||
|
||||
export function formatString(knownFormat) {
|
||||
// these all have the offsets removed because we don't have access to them
|
||||
// without all the intl stuff this is backfilling
|
||||
const filtered = pick(knownFormat, [
|
||||
"weekday",
|
||||
"era",
|
||||
"year",
|
||||
"month",
|
||||
"day",
|
||||
"hour",
|
||||
"minute",
|
||||
"second",
|
||||
"timeZoneName",
|
||||
"hourCycle",
|
||||
]),
|
||||
key = stringify(filtered),
|
||||
dateTimeHuge = "EEEE, LLLL d, yyyy, h:mm a";
|
||||
switch (key) {
|
||||
case stringify(Formats.DATE_SHORT):
|
||||
return "M/d/yyyy";
|
||||
case stringify(Formats.DATE_MED):
|
||||
return "LLL d, yyyy";
|
||||
case stringify(Formats.DATE_MED_WITH_WEEKDAY):
|
||||
return "EEE, LLL d, yyyy";
|
||||
case stringify(Formats.DATE_FULL):
|
||||
return "LLLL d, yyyy";
|
||||
case stringify(Formats.DATE_HUGE):
|
||||
return "EEEE, LLLL d, yyyy";
|
||||
case stringify(Formats.TIME_SIMPLE):
|
||||
return "h:mm a";
|
||||
case stringify(Formats.TIME_WITH_SECONDS):
|
||||
return "h:mm:ss a";
|
||||
case stringify(Formats.TIME_WITH_SHORT_OFFSET):
|
||||
return "h:mm a";
|
||||
case stringify(Formats.TIME_WITH_LONG_OFFSET):
|
||||
return "h:mm a";
|
||||
case stringify(Formats.TIME_24_SIMPLE):
|
||||
return "HH:mm";
|
||||
case stringify(Formats.TIME_24_WITH_SECONDS):
|
||||
return "HH:mm:ss";
|
||||
case stringify(Formats.TIME_24_WITH_SHORT_OFFSET):
|
||||
return "HH:mm";
|
||||
case stringify(Formats.TIME_24_WITH_LONG_OFFSET):
|
||||
return "HH:mm";
|
||||
case stringify(Formats.DATETIME_SHORT):
|
||||
return "M/d/yyyy, h:mm a";
|
||||
case stringify(Formats.DATETIME_MED):
|
||||
return "LLL d, yyyy, h:mm a";
|
||||
case stringify(Formats.DATETIME_FULL):
|
||||
return "LLLL d, yyyy, h:mm a";
|
||||
case stringify(Formats.DATETIME_HUGE):
|
||||
return dateTimeHuge;
|
||||
case stringify(Formats.DATETIME_SHORT_WITH_SECONDS):
|
||||
return "M/d/yyyy, h:mm:ss a";
|
||||
case stringify(Formats.DATETIME_MED_WITH_SECONDS):
|
||||
return "LLL d, yyyy, h:mm:ss a";
|
||||
case stringify(Formats.DATETIME_MED_WITH_WEEKDAY):
|
||||
return "EEE, d LLL yyyy, h:mm a";
|
||||
case stringify(Formats.DATETIME_FULL_WITH_SECONDS):
|
||||
return "LLLL d, yyyy, h:mm:ss a";
|
||||
case stringify(Formats.DATETIME_HUGE_WITH_SECONDS):
|
||||
return "EEEE, LLLL d, yyyy, h:mm:ss a";
|
||||
default:
|
||||
return dateTimeHuge;
|
||||
}
|
||||
}
|
176
priv/vendor/luxon/src/impl/formats.js
vendored
176
priv/vendor/luxon/src/impl/formats.js
vendored
|
@ -1,176 +0,0 @@
|
|||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
const n = "numeric",
|
||||
s = "short",
|
||||
l = "long";
|
||||
|
||||
export const DATE_SHORT = {
|
||||
year: n,
|
||||
month: n,
|
||||
day: n,
|
||||
};
|
||||
|
||||
export const DATE_MED = {
|
||||
year: n,
|
||||
month: s,
|
||||
day: n,
|
||||
};
|
||||
|
||||
export const DATE_MED_WITH_WEEKDAY = {
|
||||
year: n,
|
||||
month: s,
|
||||
day: n,
|
||||
weekday: s,
|
||||
};
|
||||
|
||||
export const DATE_FULL = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
};
|
||||
|
||||
export const DATE_HUGE = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
weekday: l,
|
||||
};
|
||||
|
||||
export const TIME_SIMPLE = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
};
|
||||
|
||||
export const TIME_WITH_SECONDS = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
};
|
||||
|
||||
export const TIME_WITH_SHORT_OFFSET = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
timeZoneName: s,
|
||||
};
|
||||
|
||||
export const TIME_WITH_LONG_OFFSET = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
timeZoneName: l,
|
||||
};
|
||||
|
||||
export const TIME_24_SIMPLE = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
hourCycle: "h23",
|
||||
};
|
||||
|
||||
export const TIME_24_WITH_SECONDS = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
hourCycle: "h23",
|
||||
};
|
||||
|
||||
export const TIME_24_WITH_SHORT_OFFSET = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
hourCycle: "h23",
|
||||
timeZoneName: s,
|
||||
};
|
||||
|
||||
export const TIME_24_WITH_LONG_OFFSET = {
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
hourCycle: "h23",
|
||||
timeZoneName: l,
|
||||
};
|
||||
|
||||
export const DATETIME_SHORT = {
|
||||
year: n,
|
||||
month: n,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
};
|
||||
|
||||
export const DATETIME_SHORT_WITH_SECONDS = {
|
||||
year: n,
|
||||
month: n,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
};
|
||||
|
||||
export const DATETIME_MED = {
|
||||
year: n,
|
||||
month: s,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
};
|
||||
|
||||
export const DATETIME_MED_WITH_SECONDS = {
|
||||
year: n,
|
||||
month: s,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
};
|
||||
|
||||
export const DATETIME_MED_WITH_WEEKDAY = {
|
||||
year: n,
|
||||
month: s,
|
||||
day: n,
|
||||
weekday: s,
|
||||
hour: n,
|
||||
minute: n,
|
||||
};
|
||||
|
||||
export const DATETIME_FULL = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
timeZoneName: s,
|
||||
};
|
||||
|
||||
export const DATETIME_FULL_WITH_SECONDS = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
timeZoneName: s,
|
||||
};
|
||||
|
||||
export const DATETIME_HUGE = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
weekday: l,
|
||||
hour: n,
|
||||
minute: n,
|
||||
timeZoneName: l,
|
||||
};
|
||||
|
||||
export const DATETIME_HUGE_WITH_SECONDS = {
|
||||
year: n,
|
||||
month: l,
|
||||
day: n,
|
||||
weekday: l,
|
||||
hour: n,
|
||||
minute: n,
|
||||
second: n,
|
||||
timeZoneName: l,
|
||||
};
|
409
priv/vendor/luxon/src/impl/formatter.js
vendored
409
priv/vendor/luxon/src/impl/formatter.js
vendored
|
@ -1,409 +0,0 @@
|
|||
import * as English from "./english.js";
|
||||
import * as Formats from "./formats.js";
|
||||
import { padStart } from "./util.js";
|
||||
|
||||
function stringifyTokens(splits, tokenToString) {
|
||||
let s = "";
|
||||
for (const token of splits) {
|
||||
if (token.literal) {
|
||||
s += token.val;
|
||||
} else {
|
||||
s += tokenToString(token.val);
|
||||
}
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
const macroTokenToFormatOpts = {
|
||||
D: Formats.DATE_SHORT,
|
||||
DD: Formats.DATE_MED,
|
||||
DDD: Formats.DATE_FULL,
|
||||
DDDD: Formats.DATE_HUGE,
|
||||
t: Formats.TIME_SIMPLE,
|
||||
tt: Formats.TIME_WITH_SECONDS,
|
||||
ttt: Formats.TIME_WITH_SHORT_OFFSET,
|
||||
tttt: Formats.TIME_WITH_LONG_OFFSET,
|
||||
T: Formats.TIME_24_SIMPLE,
|
||||
TT: Formats.TIME_24_WITH_SECONDS,
|
||||
TTT: Formats.TIME_24_WITH_SHORT_OFFSET,
|
||||
TTTT: Formats.TIME_24_WITH_LONG_OFFSET,
|
||||
f: Formats.DATETIME_SHORT,
|
||||
ff: Formats.DATETIME_MED,
|
||||
fff: Formats.DATETIME_FULL,
|
||||
ffff: Formats.DATETIME_HUGE,
|
||||
F: Formats.DATETIME_SHORT_WITH_SECONDS,
|
||||
FF: Formats.DATETIME_MED_WITH_SECONDS,
|
||||
FFF: Formats.DATETIME_FULL_WITH_SECONDS,
|
||||
FFFF: Formats.DATETIME_HUGE_WITH_SECONDS,
|
||||
};
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
export default class Formatter {
|
||||
static create(locale, opts = {}) {
|
||||
return new Formatter(locale, opts);
|
||||
}
|
||||
|
||||
static parseFormat(fmt) {
|
||||
// white-space is always considered a literal in user-provided formats
|
||||
// the " " token has a special meaning (see unitForToken)
|
||||
|
||||
let current = null,
|
||||
currentFull = "",
|
||||
bracketed = false;
|
||||
const splits = [];
|
||||
for (let i = 0; i < fmt.length; i++) {
|
||||
const c = fmt.charAt(i);
|
||||
if (c === "'") {
|
||||
if (currentFull.length > 0) {
|
||||
splits.push({ literal: bracketed || /^\s+$/.test(currentFull), val: currentFull });
|
||||
}
|
||||
current = null;
|
||||
currentFull = "";
|
||||
bracketed = !bracketed;
|
||||
} else if (bracketed) {
|
||||
currentFull += c;
|
||||
} else if (c === current) {
|
||||
currentFull += c;
|
||||
} else {
|
||||
if (currentFull.length > 0) {
|
||||
splits.push({ literal: /^\s+$/.test(currentFull), val: currentFull });
|
||||
}
|
||||
currentFull = c;
|
||||
current = c;
|
||||
}
|
||||
}
|
||||
|
||||
if (currentFull.length > 0) {
|
||||
splits.push({ literal: bracketed || /^\s+$/.test(currentFull), val: currentFull });
|
||||
}
|
||||
|
||||
return splits;
|
||||
}
|
||||
|
||||
static macroTokenToFormatOpts(token) {
|
||||
return macroTokenToFormatOpts[token];
|
||||
}
|
||||
|
||||
constructor(locale, formatOpts) {
|
||||
this.opts = formatOpts;
|
||||
this.loc = locale;
|
||||
this.systemLoc = null;
|
||||
}
|
||||
|
||||
formatWithSystemDefault(dt, opts) {
|
||||
if (this.systemLoc === null) {
|
||||
this.systemLoc = this.loc.redefaultToSystem();
|
||||
}
|
||||
const df = this.systemLoc.dtFormatter(dt, { ...this.opts, ...opts });
|
||||
return df.format();
|
||||
}
|
||||
|
||||
dtFormatter(dt, opts = {}) {
|
||||
return this.loc.dtFormatter(dt, { ...this.opts, ...opts });
|
||||
}
|
||||
|
||||
formatDateTime(dt, opts) {
|
||||
return this.dtFormatter(dt, opts).format();
|
||||
}
|
||||
|
||||
formatDateTimeParts(dt, opts) {
|
||||
return this.dtFormatter(dt, opts).formatToParts();
|
||||
}
|
||||
|
||||
formatInterval(interval, opts) {
|
||||
const df = this.dtFormatter(interval.start, opts);
|
||||
return df.dtf.formatRange(interval.start.toJSDate(), interval.end.toJSDate());
|
||||
}
|
||||
|
||||
resolvedOptions(dt, opts) {
|
||||
return this.dtFormatter(dt, opts).resolvedOptions();
|
||||
}
|
||||
|
||||
num(n, p = 0) {
|
||||
// we get some perf out of doing this here, annoyingly
|
||||
if (this.opts.forceSimple) {
|
||||
return padStart(n, p);
|
||||
}
|
||||
|
||||
const opts = { ...this.opts };
|
||||
|
||||
if (p > 0) {
|
||||
opts.padTo = p;
|
||||
}
|
||||
|
||||
return this.loc.numberFormatter(opts).format(n);
|
||||
}
|
||||
|
||||
formatDateTimeFromString(dt, fmt) {
|
||||
const knownEnglish = this.loc.listingMode() === "en",
|
||||
useDateTimeFormatter = this.loc.outputCalendar && this.loc.outputCalendar !== "gregory",
|
||||
string = (opts, extract) => this.loc.extract(dt, opts, extract),
|
||||
formatOffset = (opts) => {
|
||||
if (dt.isOffsetFixed && dt.offset === 0 && opts.allowZ) {
|
||||
return "Z";
|
||||
}
|
||||
|
||||
return dt.isValid ? dt.zone.formatOffset(dt.ts, opts.format) : "";
|
||||
},
|
||||
meridiem = () =>
|
||||
knownEnglish
|
||||
? English.meridiemForDateTime(dt)
|
||||
: string({ hour: "numeric", hourCycle: "h12" }, "dayperiod"),
|
||||
month = (length, standalone) =>
|
||||
knownEnglish
|
||||
? English.monthForDateTime(dt, length)
|
||||
: string(standalone ? { month: length } : { month: length, day: "numeric" }, "month"),
|
||||
weekday = (length, standalone) =>
|
||||
knownEnglish
|
||||
? English.weekdayForDateTime(dt, length)
|
||||
: string(
|
||||
standalone ? { weekday: length } : { weekday: length, month: "long", day: "numeric" },
|
||||
"weekday"
|
||||
),
|
||||
maybeMacro = (token) => {
|
||||
const formatOpts = Formatter.macroTokenToFormatOpts(token);
|
||||
if (formatOpts) {
|
||||
return this.formatWithSystemDefault(dt, formatOpts);
|
||||
} else {
|
||||
return token;
|
||||
}
|
||||
},
|
||||
era = (length) =>
|
||||
knownEnglish ? English.eraForDateTime(dt, length) : string({ era: length }, "era"),
|
||||
tokenToString = (token) => {
|
||||
// Where possible: https://cldr.unicode.org/translation/date-time/date-time-symbols
|
||||
switch (token) {
|
||||
// ms
|
||||
case "S":
|
||||
return this.num(dt.millisecond);
|
||||
case "u":
|
||||
// falls through
|
||||
case "SSS":
|
||||
return this.num(dt.millisecond, 3);
|
||||
// seconds
|
||||
case "s":
|
||||
return this.num(dt.second);
|
||||
case "ss":
|
||||
return this.num(dt.second, 2);
|
||||
// fractional seconds
|
||||
case "uu":
|
||||
return this.num(Math.floor(dt.millisecond / 10), 2);
|
||||
case "uuu":
|
||||
return this.num(Math.floor(dt.millisecond / 100));
|
||||
// minutes
|
||||
case "m":
|
||||
return this.num(dt.minute);
|
||||
case "mm":
|
||||
return this.num(dt.minute, 2);
|
||||
// hours
|
||||
case "h":
|
||||
return this.num(dt.hour % 12 === 0 ? 12 : dt.hour % 12);
|
||||
case "hh":
|
||||
return this.num(dt.hour % 12 === 0 ? 12 : dt.hour % 12, 2);
|
||||
case "H":
|
||||
return this.num(dt.hour);
|
||||
case "HH":
|
||||
return this.num(dt.hour, 2);
|
||||
// offset
|
||||
case "Z":
|
||||
// like +6
|
||||
return formatOffset({ format: "narrow", allowZ: this.opts.allowZ });
|
||||
case "ZZ":
|
||||
// like +06:00
|
||||
return formatOffset({ format: "short", allowZ: this.opts.allowZ });
|
||||
case "ZZZ":
|
||||
// like +0600
|
||||
return formatOffset({ format: "techie", allowZ: this.opts.allowZ });
|
||||
case "ZZZZ":
|
||||
// like EST
|
||||
return dt.zone.offsetName(dt.ts, { format: "short", locale: this.loc.locale });
|
||||
case "ZZZZZ":
|
||||
// like Eastern Standard Time
|
||||
return dt.zone.offsetName(dt.ts, { format: "long", locale: this.loc.locale });
|
||||
// zone
|
||||
case "z":
|
||||
// like America/New_York
|
||||
return dt.zoneName;
|
||||
// meridiems
|
||||
case "a":
|
||||
return meridiem();
|
||||
// dates
|
||||
case "d":
|
||||
return useDateTimeFormatter ? string({ day: "numeric" }, "day") : this.num(dt.day);
|
||||
case "dd":
|
||||
return useDateTimeFormatter ? string({ day: "2-digit" }, "day") : this.num(dt.day, 2);
|
||||
// weekdays - standalone
|
||||
case "c":
|
||||
// like 1
|
||||
return this.num(dt.weekday);
|
||||
case "ccc":
|
||||
// like 'Tues'
|
||||
return weekday("short", true);
|
||||
case "cccc":
|
||||
// like 'Tuesday'
|
||||
return weekday("long", true);
|
||||
case "ccccc":
|
||||
// like 'T'
|
||||
return weekday("narrow", true);
|
||||
// weekdays - format
|
||||
case "E":
|
||||
// like 1
|
||||
return this.num(dt.weekday);
|
||||
case "EEE":
|
||||
// like 'Tues'
|
||||
return weekday("short", false);
|
||||
case "EEEE":
|
||||
// like 'Tuesday'
|
||||
return weekday("long", false);
|
||||
case "EEEEE":
|
||||
// like 'T'
|
||||
return weekday("narrow", false);
|
||||
// months - standalone
|
||||
case "L":
|
||||
// like 1
|
||||
return useDateTimeFormatter
|
||||
? string({ month: "numeric", day: "numeric" }, "month")
|
||||
: this.num(dt.month);
|
||||
case "LL":
|
||||
// like 01, doesn't seem to work
|
||||
return useDateTimeFormatter
|
||||
? string({ month: "2-digit", day: "numeric" }, "month")
|
||||
: this.num(dt.month, 2);
|
||||
case "LLL":
|
||||
// like Jan
|
||||
return month("short", true);
|
||||
case "LLLL":
|
||||
// like January
|
||||
return month("long", true);
|
||||
case "LLLLL":
|
||||
// like J
|
||||
return month("narrow", true);
|
||||
// months - format
|
||||
case "M":
|
||||
// like 1
|
||||
return useDateTimeFormatter
|
||||
? string({ month: "numeric" }, "month")
|
||||
: this.num(dt.month);
|
||||
case "MM":
|
||||
// like 01
|
||||
return useDateTimeFormatter
|
||||
? string({ month: "2-digit" }, "month")
|
||||
: this.num(dt.month, 2);
|
||||
case "MMM":
|
||||
// like Jan
|
||||
return month("short", false);
|
||||
case "MMMM":
|
||||
// like January
|
||||
return month("long", false);
|
||||
case "MMMMM":
|
||||
// like J
|
||||
return month("narrow", false);
|
||||
// years
|
||||
case "y":
|
||||
// like 2014
|
||||
return useDateTimeFormatter ? string({ year: "numeric" }, "year") : this.num(dt.year);
|
||||
case "yy":
|
||||
// like 14
|
||||
return useDateTimeFormatter
|
||||
? string({ year: "2-digit" }, "year")
|
||||
: this.num(dt.year.toString().slice(-2), 2);
|
||||
case "yyyy":
|
||||
// like 0012
|
||||
return useDateTimeFormatter
|
||||
? string({ year: "numeric" }, "year")
|
||||
: this.num(dt.year, 4);
|
||||
case "yyyyyy":
|
||||
// like 000012
|
||||
return useDateTimeFormatter
|
||||
? string({ year: "numeric" }, "year")
|
||||
: this.num(dt.year, 6);
|
||||
// eras
|
||||
case "G":
|
||||
// like AD
|
||||
return era("short");
|
||||
case "GG":
|
||||
// like Anno Domini
|
||||
return era("long");
|
||||
case "GGGGG":
|
||||
return era("narrow");
|
||||
case "kk":
|
||||
return this.num(dt.weekYear.toString().slice(-2), 2);
|
||||
case "kkkk":
|
||||
return this.num(dt.weekYear, 4);
|
||||
case "W":
|
||||
return this.num(dt.weekNumber);
|
||||
case "WW":
|
||||
return this.num(dt.weekNumber, 2);
|
||||
case "n":
|
||||
return this.num(dt.localWeekNumber);
|
||||
case "nn":
|
||||
return this.num(dt.localWeekNumber, 2);
|
||||
case "ii":
|
||||
return this.num(dt.localWeekYear.toString().slice(-2), 2);
|
||||
case "iiii":
|
||||
return this.num(dt.localWeekYear, 4);
|
||||
case "o":
|
||||
return this.num(dt.ordinal);
|
||||
case "ooo":
|
||||
return this.num(dt.ordinal, 3);
|
||||
case "q":
|
||||
// like 1
|
||||
return this.num(dt.quarter);
|
||||
case "qq":
|
||||
// like 01
|
||||
return this.num(dt.quarter, 2);
|
||||
case "X":
|
||||
return this.num(Math.floor(dt.ts / 1000));
|
||||
case "x":
|
||||
return this.num(dt.ts);
|
||||
default:
|
||||
return maybeMacro(token);
|
||||
}
|
||||
};
|
||||
|
||||
return stringifyTokens(Formatter.parseFormat(fmt), tokenToString);
|
||||
}
|
||||
|
||||
formatDurationFromString(dur, fmt) {
|
||||
const tokenToField = (token) => {
|
||||
switch (token[0]) {
|
||||
case "S":
|
||||
return "millisecond";
|
||||
case "s":
|
||||
return "second";
|
||||
case "m":
|
||||
return "minute";
|
||||
case "h":
|
||||
return "hour";
|
||||
case "d":
|
||||
return "day";
|
||||
case "w":
|
||||
return "week";
|
||||
case "M":
|
||||
return "month";
|
||||
case "y":
|
||||
return "year";
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
},
|
||||
tokenToString = (lildur) => (token) => {
|
||||
const mapped = tokenToField(token);
|
||||
if (mapped) {
|
||||
return this.num(lildur.get(mapped), token.length);
|
||||
} else {
|
||||
return token;
|
||||
}
|
||||
},
|
||||
tokens = Formatter.parseFormat(fmt),
|
||||
realTokens = tokens.reduce(
|
||||
(found, { literal, val }) => (literal ? found : found.concat(val)),
|
||||
[]
|
||||
),
|
||||
collapsed = dur.shiftTo(...realTokens.map(tokenToField).filter((t) => t));
|
||||
return stringifyTokens(tokens, tokenToString(collapsed));
|
||||
}
|
||||
}
|
14
priv/vendor/luxon/src/impl/invalid.js
vendored
14
priv/vendor/luxon/src/impl/invalid.js
vendored
|
@ -1,14 +0,0 @@
|
|||
export default class Invalid {
|
||||
constructor(reason, explanation) {
|
||||
this.reason = reason;
|
||||
this.explanation = explanation;
|
||||
}
|
||||
|
||||
toMessage() {
|
||||
if (this.explanation) {
|
||||
return `${this.reason}: ${this.explanation}`;
|
||||
} else {
|
||||
return this.reason;
|
||||
}
|
||||
}
|
||||
}
|
542
priv/vendor/luxon/src/impl/locale.js
vendored
542
priv/vendor/luxon/src/impl/locale.js
vendored
|
@ -1,542 +0,0 @@
|
|||
import { hasLocaleWeekInfo, hasRelative, padStart, roundTo, validateWeekSettings } from "./util.js";
|
||||
import * as English from "./english.js";
|
||||
import Settings from "../settings.js";
|
||||
import DateTime from "../datetime.js";
|
||||
import IANAZone from "../zones/IANAZone.js";
|
||||
|
||||
// todo - remap caching
|
||||
|
||||
let intlLFCache = {};
|
||||
function getCachedLF(locString, opts = {}) {
|
||||
const key = JSON.stringify([locString, opts]);
|
||||
let dtf = intlLFCache[key];
|
||||
if (!dtf) {
|
||||
dtf = new Intl.ListFormat(locString, opts);
|
||||
intlLFCache[key] = dtf;
|
||||
}
|
||||
return dtf;
|
||||
}
|
||||
|
||||
let intlDTCache = {};
|
||||
function getCachedDTF(locString, opts = {}) {
|
||||
const key = JSON.stringify([locString, opts]);
|
||||
let dtf = intlDTCache[key];
|
||||
if (!dtf) {
|
||||
dtf = new Intl.DateTimeFormat(locString, opts);
|
||||
intlDTCache[key] = dtf;
|
||||
}
|
||||
return dtf;
|
||||
}
|
||||
|
||||
let intlNumCache = {};
|
||||
function getCachedINF(locString, opts = {}) {
|
||||
const key = JSON.stringify([locString, opts]);
|
||||
let inf = intlNumCache[key];
|
||||
if (!inf) {
|
||||
inf = new Intl.NumberFormat(locString, opts);
|
||||
intlNumCache[key] = inf;
|
||||
}
|
||||
return inf;
|
||||
}
|
||||
|
||||
let intlRelCache = {};
|
||||
function getCachedRTF(locString, opts = {}) {
|
||||
const { base, ...cacheKeyOpts } = opts; // exclude `base` from the options
|
||||
const key = JSON.stringify([locString, cacheKeyOpts]);
|
||||
let inf = intlRelCache[key];
|
||||
if (!inf) {
|
||||
inf = new Intl.RelativeTimeFormat(locString, opts);
|
||||
intlRelCache[key] = inf;
|
||||
}
|
||||
return inf;
|
||||
}
|
||||
|
||||
let sysLocaleCache = null;
|
||||
function systemLocale() {
|
||||
if (sysLocaleCache) {
|
||||
return sysLocaleCache;
|
||||
} else {
|
||||
sysLocaleCache = new Intl.DateTimeFormat().resolvedOptions().locale;
|
||||
return sysLocaleCache;
|
||||
}
|
||||
}
|
||||
|
||||
let weekInfoCache = {};
|
||||
function getCachedWeekInfo(locString) {
|
||||
let data = weekInfoCache[locString];
|
||||
if (!data) {
|
||||
const locale = new Intl.Locale(locString);
|
||||
// browsers currently implement this as a property, but spec says it should be a getter function
|
||||
data = "getWeekInfo" in locale ? locale.getWeekInfo() : locale.weekInfo;
|
||||
weekInfoCache[locString] = data;
|
||||
}
|
||||
return data;
|
||||
}
|
||||
|
||||
function parseLocaleString(localeStr) {
|
||||
// I really want to avoid writing a BCP 47 parser
|
||||
// see, e.g. https://github.com/wooorm/bcp-47
|
||||
// Instead, we'll do this:
|
||||
|
||||
// a) if the string has no -u extensions, just leave it alone
|
||||
// b) if it does, use Intl to resolve everything
|
||||
// c) if Intl fails, try again without the -u
|
||||
|
||||
// private subtags and unicode subtags have ordering requirements,
|
||||
// and we're not properly parsing this, so just strip out the
|
||||
// private ones if they exist.
|
||||
const xIndex = localeStr.indexOf("-x-");
|
||||
if (xIndex !== -1) {
|
||||
localeStr = localeStr.substring(0, xIndex);
|
||||
}
|
||||
|
||||
const uIndex = localeStr.indexOf("-u-");
|
||||
if (uIndex === -1) {
|
||||
return [localeStr];
|
||||
} else {
|
||||
let options;
|
||||
let selectedStr;
|
||||
try {
|
||||
options = getCachedDTF(localeStr).resolvedOptions();
|
||||
selectedStr = localeStr;
|
||||
} catch (e) {
|
||||
const smaller = localeStr.substring(0, uIndex);
|
||||
options = getCachedDTF(smaller).resolvedOptions();
|
||||
selectedStr = smaller;
|
||||
}
|
||||
|
||||
const { numberingSystem, calendar } = options;
|
||||
return [selectedStr, numberingSystem, calendar];
|
||||
}
|
||||
}
|
||||
|
||||
function intlConfigString(localeStr, numberingSystem, outputCalendar) {
|
||||
if (outputCalendar || numberingSystem) {
|
||||
if (!localeStr.includes("-u-")) {
|
||||
localeStr += "-u";
|
||||
}
|
||||
|
||||
if (outputCalendar) {
|
||||
localeStr += `-ca-${outputCalendar}`;
|
||||
}
|
||||
|
||||
if (numberingSystem) {
|
||||
localeStr += `-nu-${numberingSystem}`;
|
||||
}
|
||||
return localeStr;
|
||||
} else {
|
||||
return localeStr;
|
||||
}
|
||||
}
|
||||
|
||||
function mapMonths(f) {
|
||||
const ms = [];
|
||||
for (let i = 1; i <= 12; i++) {
|
||||
const dt = DateTime.utc(2009, i, 1);
|
||||
ms.push(f(dt));
|
||||
}
|
||||
return ms;
|
||||
}
|
||||
|
||||
function mapWeekdays(f) {
|
||||
const ms = [];
|
||||
for (let i = 1; i <= 7; i++) {
|
||||
const dt = DateTime.utc(2016, 11, 13 + i);
|
||||
ms.push(f(dt));
|
||||
}
|
||||
return ms;
|
||||
}
|
||||
|
||||
function listStuff(loc, length, englishFn, intlFn) {
|
||||
const mode = loc.listingMode();
|
||||
|
||||
if (mode === "error") {
|
||||
return null;
|
||||
} else if (mode === "en") {
|
||||
return englishFn(length);
|
||||
} else {
|
||||
return intlFn(length);
|
||||
}
|
||||
}
|
||||
|
||||
function supportsFastNumbers(loc) {
|
||||
if (loc.numberingSystem && loc.numberingSystem !== "latn") {
|
||||
return false;
|
||||
} else {
|
||||
return (
|
||||
loc.numberingSystem === "latn" ||
|
||||
!loc.locale ||
|
||||
loc.locale.startsWith("en") ||
|
||||
new Intl.DateTimeFormat(loc.intl).resolvedOptions().numberingSystem === "latn"
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
class PolyNumberFormatter {
|
||||
constructor(intl, forceSimple, opts) {
|
||||
this.padTo = opts.padTo || 0;
|
||||
this.floor = opts.floor || false;
|
||||
|
||||
const { padTo, floor, ...otherOpts } = opts;
|
||||
|
||||
if (!forceSimple || Object.keys(otherOpts).length > 0) {
|
||||
const intlOpts = { useGrouping: false, ...opts };
|
||||
if (opts.padTo > 0) intlOpts.minimumIntegerDigits = opts.padTo;
|
||||
this.inf = getCachedINF(intl, intlOpts);
|
||||
}
|
||||
}
|
||||
|
||||
format(i) {
|
||||
if (this.inf) {
|
||||
const fixed = this.floor ? Math.floor(i) : i;
|
||||
return this.inf.format(fixed);
|
||||
} else {
|
||||
// to match the browser's numberformatter defaults
|
||||
const fixed = this.floor ? Math.floor(i) : roundTo(i, 3);
|
||||
return padStart(fixed, this.padTo);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
class PolyDateFormatter {
|
||||
constructor(dt, intl, opts) {
|
||||
this.opts = opts;
|
||||
this.originalZone = undefined;
|
||||
|
||||
let z = undefined;
|
||||
if (this.opts.timeZone) {
|
||||
// Don't apply any workarounds if a timeZone is explicitly provided in opts
|
||||
this.dt = dt;
|
||||
} else if (dt.zone.type === "fixed") {
|
||||
// UTC-8 or Etc/UTC-8 are not part of tzdata, only Etc/GMT+8 and the like.
|
||||
// That is why fixed-offset TZ is set to that unless it is:
|
||||
// 1. Representing offset 0 when UTC is used to maintain previous behavior and does not become GMT.
|
||||
// 2. Unsupported by the browser:
|
||||
// - some do not support Etc/
|
||||
// - < Etc/GMT-14, > Etc/GMT+12, and 30-minute or 45-minute offsets are not part of tzdata
|
||||
const gmtOffset = -1 * (dt.offset / 60);
|
||||
const offsetZ = gmtOffset >= 0 ? `Etc/GMT+${gmtOffset}` : `Etc/GMT${gmtOffset}`;
|
||||
if (dt.offset !== 0 && IANAZone.create(offsetZ).valid) {
|
||||
z = offsetZ;
|
||||
this.dt = dt;
|
||||
} else {
|
||||
// Not all fixed-offset zones like Etc/+4:30 are present in tzdata so
|
||||
// we manually apply the offset and substitute the zone as needed.
|
||||
z = "UTC";
|
||||
this.dt = dt.offset === 0 ? dt : dt.setZone("UTC").plus({ minutes: dt.offset });
|
||||
this.originalZone = dt.zone;
|
||||
}
|
||||
} else if (dt.zone.type === "system") {
|
||||
this.dt = dt;
|
||||
} else if (dt.zone.type === "iana") {
|
||||
this.dt = dt;
|
||||
z = dt.zone.name;
|
||||
} else {
|
||||
// Custom zones can have any offset / offsetName so we just manually
|
||||
// apply the offset and substitute the zone as needed.
|
||||
z = "UTC";
|
||||
this.dt = dt.setZone("UTC").plus({ minutes: dt.offset });
|
||||
this.originalZone = dt.zone;
|
||||
}
|
||||
|
||||
const intlOpts = { ...this.opts };
|
||||
intlOpts.timeZone = intlOpts.timeZone || z;
|
||||
this.dtf = getCachedDTF(intl, intlOpts);
|
||||
}
|
||||
|
||||
format() {
|
||||
if (this.originalZone) {
|
||||
// If we have to substitute in the actual zone name, we have to use
|
||||
// formatToParts so that the timezone can be replaced.
|
||||
return this.formatToParts()
|
||||
.map(({ value }) => value)
|
||||
.join("");
|
||||
}
|
||||
return this.dtf.format(this.dt.toJSDate());
|
||||
}
|
||||
|
||||
formatToParts() {
|
||||
const parts = this.dtf.formatToParts(this.dt.toJSDate());
|
||||
if (this.originalZone) {
|
||||
return parts.map((part) => {
|
||||
if (part.type === "timeZoneName") {
|
||||
const offsetName = this.originalZone.offsetName(this.dt.ts, {
|
||||
locale: this.dt.locale,
|
||||
format: this.opts.timeZoneName,
|
||||
});
|
||||
return {
|
||||
...part,
|
||||
value: offsetName,
|
||||
};
|
||||
} else {
|
||||
return part;
|
||||
}
|
||||
});
|
||||
}
|
||||
return parts;
|
||||
}
|
||||
|
||||
resolvedOptions() {
|
||||
return this.dtf.resolvedOptions();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
class PolyRelFormatter {
|
||||
constructor(intl, isEnglish, opts) {
|
||||
this.opts = { style: "long", ...opts };
|
||||
if (!isEnglish && hasRelative()) {
|
||||
this.rtf = getCachedRTF(intl, opts);
|
||||
}
|
||||
}
|
||||
|
||||
format(count, unit) {
|
||||
if (this.rtf) {
|
||||
return this.rtf.format(count, unit);
|
||||
} else {
|
||||
return English.formatRelativeTime(unit, count, this.opts.numeric, this.opts.style !== "long");
|
||||
}
|
||||
}
|
||||
|
||||
formatToParts(count, unit) {
|
||||
if (this.rtf) {
|
||||
return this.rtf.formatToParts(count, unit);
|
||||
} else {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const fallbackWeekSettings = {
|
||||
firstDay: 1,
|
||||
minimalDays: 4,
|
||||
weekend: [6, 7],
|
||||
};
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
export default class Locale {
|
||||
static fromOpts(opts) {
|
||||
return Locale.create(
|
||||
opts.locale,
|
||||
opts.numberingSystem,
|
||||
opts.outputCalendar,
|
||||
opts.weekSettings,
|
||||
opts.defaultToEN
|
||||
);
|
||||
}
|
||||
|
||||
static create(locale, numberingSystem, outputCalendar, weekSettings, defaultToEN = false) {
|
||||
const specifiedLocale = locale || Settings.defaultLocale;
|
||||
// the system locale is useful for human readable strings but annoying for parsing/formatting known formats
|
||||
const localeR = specifiedLocale || (defaultToEN ? "en-US" : systemLocale());
|
||||
const numberingSystemR = numberingSystem || Settings.defaultNumberingSystem;
|
||||
const outputCalendarR = outputCalendar || Settings.defaultOutputCalendar;
|
||||
const weekSettingsR = validateWeekSettings(weekSettings) || Settings.defaultWeekSettings;
|
||||
return new Locale(localeR, numberingSystemR, outputCalendarR, weekSettingsR, specifiedLocale);
|
||||
}
|
||||
|
||||
static resetCache() {
|
||||
sysLocaleCache = null;
|
||||
intlDTCache = {};
|
||||
intlNumCache = {};
|
||||
intlRelCache = {};
|
||||
}
|
||||
|
||||
static fromObject({ locale, numberingSystem, outputCalendar, weekSettings } = {}) {
|
||||
return Locale.create(locale, numberingSystem, outputCalendar, weekSettings);
|
||||
}
|
||||
|
||||
constructor(locale, numbering, outputCalendar, weekSettings, specifiedLocale) {
|
||||
const [parsedLocale, parsedNumberingSystem, parsedOutputCalendar] = parseLocaleString(locale);
|
||||
|
||||
this.locale = parsedLocale;
|
||||
this.numberingSystem = numbering || parsedNumberingSystem || null;
|
||||
this.outputCalendar = outputCalendar || parsedOutputCalendar || null;
|
||||
this.weekSettings = weekSettings;
|
||||
this.intl = intlConfigString(this.locale, this.numberingSystem, this.outputCalendar);
|
||||
|
||||
this.weekdaysCache = { format: {}, standalone: {} };
|
||||
this.monthsCache = { format: {}, standalone: {} };
|
||||
this.meridiemCache = null;
|
||||
this.eraCache = {};
|
||||
|
||||
this.specifiedLocale = specifiedLocale;
|
||||
this.fastNumbersCached = null;
|
||||
}
|
||||
|
||||
get fastNumbers() {
|
||||
if (this.fastNumbersCached == null) {
|
||||
this.fastNumbersCached = supportsFastNumbers(this);
|
||||
}
|
||||
|
||||
return this.fastNumbersCached;
|
||||
}
|
||||
|
||||
listingMode() {
|
||||
const isActuallyEn = this.isEnglish();
|
||||
const hasNoWeirdness =
|
||||
(this.numberingSystem === null || this.numberingSystem === "latn") &&
|
||||
(this.outputCalendar === null || this.outputCalendar === "gregory");
|
||||
return isActuallyEn && hasNoWeirdness ? "en" : "intl";
|
||||
}
|
||||
|
||||
clone(alts) {
|
||||
if (!alts || Object.getOwnPropertyNames(alts).length === 0) {
|
||||
return this;
|
||||
} else {
|
||||
return Locale.create(
|
||||
alts.locale || this.specifiedLocale,
|
||||
alts.numberingSystem || this.numberingSystem,
|
||||
alts.outputCalendar || this.outputCalendar,
|
||||
validateWeekSettings(alts.weekSettings) || this.weekSettings,
|
||||
alts.defaultToEN || false
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
redefaultToEN(alts = {}) {
|
||||
return this.clone({ ...alts, defaultToEN: true });
|
||||
}
|
||||
|
||||
redefaultToSystem(alts = {}) {
|
||||
return this.clone({ ...alts, defaultToEN: false });
|
||||
}
|
||||
|
||||
months(length, format = false) {
|
||||
return listStuff(this, length, English.months, () => {
|
||||
const intl = format ? { month: length, day: "numeric" } : { month: length },
|
||||
formatStr = format ? "format" : "standalone";
|
||||
if (!this.monthsCache[formatStr][length]) {
|
||||
this.monthsCache[formatStr][length] = mapMonths((dt) => this.extract(dt, intl, "month"));
|
||||
}
|
||||
return this.monthsCache[formatStr][length];
|
||||
});
|
||||
}
|
||||
|
||||
weekdays(length, format = false) {
|
||||
return listStuff(this, length, English.weekdays, () => {
|
||||
const intl = format
|
||||
? { weekday: length, year: "numeric", month: "long", day: "numeric" }
|
||||
: { weekday: length },
|
||||
formatStr = format ? "format" : "standalone";
|
||||
if (!this.weekdaysCache[formatStr][length]) {
|
||||
this.weekdaysCache[formatStr][length] = mapWeekdays((dt) =>
|
||||
this.extract(dt, intl, "weekday")
|
||||
);
|
||||
}
|
||||
return this.weekdaysCache[formatStr][length];
|
||||
});
|
||||
}
|
||||
|
||||
meridiems() {
|
||||
return listStuff(
|
||||
this,
|
||||
undefined,
|
||||
() => English.meridiems,
|
||||
() => {
|
||||
// In theory there could be aribitrary day periods. We're gonna assume there are exactly two
|
||||
// for AM and PM. This is probably wrong, but it's makes parsing way easier.
|
||||
if (!this.meridiemCache) {
|
||||
const intl = { hour: "numeric", hourCycle: "h12" };
|
||||
this.meridiemCache = [DateTime.utc(2016, 11, 13, 9), DateTime.utc(2016, 11, 13, 19)].map(
|
||||
(dt) => this.extract(dt, intl, "dayperiod")
|
||||
);
|
||||
}
|
||||
|
||||
return this.meridiemCache;
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
eras(length) {
|
||||
return listStuff(this, length, English.eras, () => {
|
||||
const intl = { era: length };
|
||||
|
||||
// This is problematic. Different calendars are going to define eras totally differently. What I need is the minimum set of dates
|
||||
// to definitely enumerate them.
|
||||
if (!this.eraCache[length]) {
|
||||
this.eraCache[length] = [DateTime.utc(-40, 1, 1), DateTime.utc(2017, 1, 1)].map((dt) =>
|
||||
this.extract(dt, intl, "era")
|
||||
);
|
||||
}
|
||||
|
||||
return this.eraCache[length];
|
||||
});
|
||||
}
|
||||
|
||||
extract(dt, intlOpts, field) {
|
||||
const df = this.dtFormatter(dt, intlOpts),
|
||||
results = df.formatToParts(),
|
||||
matching = results.find((m) => m.type.toLowerCase() === field);
|
||||
return matching ? matching.value : null;
|
||||
}
|
||||
|
||||
numberFormatter(opts = {}) {
|
||||
// this forcesimple option is never used (the only caller short-circuits on it, but it seems safer to leave)
|
||||
// (in contrast, the rest of the condition is used heavily)
|
||||
return new PolyNumberFormatter(this.intl, opts.forceSimple || this.fastNumbers, opts);
|
||||
}
|
||||
|
||||
dtFormatter(dt, intlOpts = {}) {
|
||||
return new PolyDateFormatter(dt, this.intl, intlOpts);
|
||||
}
|
||||
|
||||
relFormatter(opts = {}) {
|
||||
return new PolyRelFormatter(this.intl, this.isEnglish(), opts);
|
||||
}
|
||||
|
||||
listFormatter(opts = {}) {
|
||||
return getCachedLF(this.intl, opts);
|
||||
}
|
||||
|
||||
isEnglish() {
|
||||
return (
|
||||
this.locale === "en" ||
|
||||
this.locale.toLowerCase() === "en-us" ||
|
||||
new Intl.DateTimeFormat(this.intl).resolvedOptions().locale.startsWith("en-us")
|
||||
);
|
||||
}
|
||||
|
||||
getWeekSettings() {
|
||||
if (this.weekSettings) {
|
||||
return this.weekSettings;
|
||||
} else if (!hasLocaleWeekInfo()) {
|
||||
return fallbackWeekSettings;
|
||||
} else {
|
||||
return getCachedWeekInfo(this.locale);
|
||||
}
|
||||
}
|
||||
|
||||
getStartOfWeek() {
|
||||
return this.getWeekSettings().firstDay;
|
||||
}
|
||||
|
||||
getMinDaysInFirstWeek() {
|
||||
return this.getWeekSettings().minimalDays;
|
||||
}
|
||||
|
||||
getWeekendDays() {
|
||||
return this.getWeekSettings().weekend;
|
||||
}
|
||||
|
||||
equals(other) {
|
||||
return (
|
||||
this.locale === other.locale &&
|
||||
this.numberingSystem === other.numberingSystem &&
|
||||
this.outputCalendar === other.outputCalendar
|
||||
);
|
||||
}
|
||||
}
|
335
priv/vendor/luxon/src/impl/regexParser.js
vendored
335
priv/vendor/luxon/src/impl/regexParser.js
vendored
|
@ -1,335 +0,0 @@
|
|||
import {
|
||||
untruncateYear,
|
||||
signedOffset,
|
||||
parseInteger,
|
||||
parseMillis,
|
||||
isUndefined,
|
||||
parseFloating,
|
||||
} from "./util.js";
|
||||
import * as English from "./english.js";
|
||||
import FixedOffsetZone from "../zones/fixedOffsetZone.js";
|
||||
import IANAZone from "../zones/IANAZone.js";
|
||||
|
||||
/*
|
||||
* This file handles parsing for well-specified formats. Here's how it works:
|
||||
* Two things go into parsing: a regex to match with and an extractor to take apart the groups in the match.
|
||||
* An extractor is just a function that takes a regex match array and returns a { year: ..., month: ... } object
|
||||
* parse() does the work of executing the regex and applying the extractor. It takes multiple regex/extractor pairs to try in sequence.
|
||||
* Extractors can take a "cursor" representing the offset in the match to look at. This makes it easy to combine extractors.
|
||||
* combineExtractors() does the work of combining them, keeping track of the cursor through multiple extractions.
|
||||
* Some extractions are super dumb and simpleParse and fromStrings help DRY them.
|
||||
*/
|
||||
|
||||
const ianaRegex = /[A-Za-z_+-]{1,256}(?::?\/[A-Za-z0-9_+-]{1,256}(?:\/[A-Za-z0-9_+-]{1,256})?)?/;
|
||||
|
||||
function combineRegexes(...regexes) {
|
||||
const full = regexes.reduce((f, r) => f + r.source, "");
|
||||
return RegExp(`^${full}$`);
|
||||
}
|
||||
|
||||
function combineExtractors(...extractors) {
|
||||
return (m) =>
|
||||
extractors
|
||||
.reduce(
|
||||
([mergedVals, mergedZone, cursor], ex) => {
|
||||
const [val, zone, next] = ex(m, cursor);
|
||||
return [{ ...mergedVals, ...val }, zone || mergedZone, next];
|
||||
},
|
||||
[{}, null, 1]
|
||||
)
|
||||
.slice(0, 2);
|
||||
}
|
||||
|
||||
function parse(s, ...patterns) {
|
||||
if (s == null) {
|
||||
return [null, null];
|
||||
}
|
||||
|
||||
for (const [regex, extractor] of patterns) {
|
||||
const m = regex.exec(s);
|
||||
if (m) {
|
||||
return extractor(m);
|
||||
}
|
||||
}
|
||||
return [null, null];
|
||||
}
|
||||
|
||||
function simpleParse(...keys) {
|
||||
return (match, cursor) => {
|
||||
const ret = {};
|
||||
let i;
|
||||
|
||||
for (i = 0; i < keys.length; i++) {
|
||||
ret[keys[i]] = parseInteger(match[cursor + i]);
|
||||
}
|
||||
return [ret, null, cursor + i];
|
||||
};
|
||||
}
|
||||
|
||||
// ISO and SQL parsing
|
||||
const offsetRegex = /(?:(Z)|([+-]\d\d)(?::?(\d\d))?)/;
|
||||
const isoExtendedZone = `(?:${offsetRegex.source}?(?:\\[(${ianaRegex.source})\\])?)?`;
|
||||
const isoTimeBaseRegex = /(\d\d)(?::?(\d\d)(?::?(\d\d)(?:[.,](\d{1,30}))?)?)?/;
|
||||
const isoTimeRegex = RegExp(`${isoTimeBaseRegex.source}${isoExtendedZone}`);
|
||||
const isoTimeExtensionRegex = RegExp(`(?:T${isoTimeRegex.source})?`);
|
||||
const isoYmdRegex = /([+-]\d{6}|\d{4})(?:-?(\d\d)(?:-?(\d\d))?)?/;
|
||||
const isoWeekRegex = /(\d{4})-?W(\d\d)(?:-?(\d))?/;
|
||||
const isoOrdinalRegex = /(\d{4})-?(\d{3})/;
|
||||
const extractISOWeekData = simpleParse("weekYear", "weekNumber", "weekDay");
|
||||
const extractISOOrdinalData = simpleParse("year", "ordinal");
|
||||
const sqlYmdRegex = /(\d{4})-(\d\d)-(\d\d)/; // dumbed-down version of the ISO one
|
||||
const sqlTimeRegex = RegExp(
|
||||
`${isoTimeBaseRegex.source} ?(?:${offsetRegex.source}|(${ianaRegex.source}))?`
|
||||
);
|
||||
const sqlTimeExtensionRegex = RegExp(`(?: ${sqlTimeRegex.source})?`);
|
||||
|
||||
function int(match, pos, fallback) {
|
||||
const m = match[pos];
|
||||
return isUndefined(m) ? fallback : parseInteger(m);
|
||||
}
|
||||
|
||||
function extractISOYmd(match, cursor) {
|
||||
const item = {
|
||||
year: int(match, cursor),
|
||||
month: int(match, cursor + 1, 1),
|
||||
day: int(match, cursor + 2, 1),
|
||||
};
|
||||
|
||||
return [item, null, cursor + 3];
|
||||
}
|
||||
|
||||
function extractISOTime(match, cursor) {
|
||||
const item = {
|
||||
hours: int(match, cursor, 0),
|
||||
minutes: int(match, cursor + 1, 0),
|
||||
seconds: int(match, cursor + 2, 0),
|
||||
milliseconds: parseMillis(match[cursor + 3]),
|
||||
};
|
||||
|
||||
return [item, null, cursor + 4];
|
||||
}
|
||||
|
||||
function extractISOOffset(match, cursor) {
|
||||
const local = !match[cursor] && !match[cursor + 1],
|
||||
fullOffset = signedOffset(match[cursor + 1], match[cursor + 2]),
|
||||
zone = local ? null : FixedOffsetZone.instance(fullOffset);
|
||||
return [{}, zone, cursor + 3];
|
||||
}
|
||||
|
||||
function extractIANAZone(match, cursor) {
|
||||
const zone = match[cursor] ? IANAZone.create(match[cursor]) : null;
|
||||
return [{}, zone, cursor + 1];
|
||||
}
|
||||
|
||||
// ISO time parsing
|
||||
|
||||
const isoTimeOnly = RegExp(`^T?${isoTimeBaseRegex.source}$`);
|
||||
|
||||
// ISO duration parsing
|
||||
|
||||
const isoDuration =
|
||||
/^-?P(?:(?:(-?\d{1,20}(?:\.\d{1,20})?)Y)?(?:(-?\d{1,20}(?:\.\d{1,20})?)M)?(?:(-?\d{1,20}(?:\.\d{1,20})?)W)?(?:(-?\d{1,20}(?:\.\d{1,20})?)D)?(?:T(?:(-?\d{1,20}(?:\.\d{1,20})?)H)?(?:(-?\d{1,20}(?:\.\d{1,20})?)M)?(?:(-?\d{1,20})(?:[.,](-?\d{1,20}))?S)?)?)$/;
|
||||
|
||||
function extractISODuration(match) {
|
||||
const [s, yearStr, monthStr, weekStr, dayStr, hourStr, minuteStr, secondStr, millisecondsStr] =
|
||||
match;
|
||||
|
||||
const hasNegativePrefix = s[0] === "-";
|
||||
const negativeSeconds = secondStr && secondStr[0] === "-";
|
||||
|
||||
const maybeNegate = (num, force = false) =>
|
||||
num !== undefined && (force || (num && hasNegativePrefix)) ? -num : num;
|
||||
|
||||
return [
|
||||
{
|
||||
years: maybeNegate(parseFloating(yearStr)),
|
||||
months: maybeNegate(parseFloating(monthStr)),
|
||||
weeks: maybeNegate(parseFloating(weekStr)),
|
||||
days: maybeNegate(parseFloating(dayStr)),
|
||||
hours: maybeNegate(parseFloating(hourStr)),
|
||||
minutes: maybeNegate(parseFloating(minuteStr)),
|
||||
seconds: maybeNegate(parseFloating(secondStr), secondStr === "-0"),
|
||||
milliseconds: maybeNegate(parseMillis(millisecondsStr), negativeSeconds),
|
||||
},
|
||||
];
|
||||
}
|
||||
|
||||
// These are a little braindead. EDT *should* tell us that we're in, say, America/New_York
|
||||
// and not just that we're in -240 *right now*. But since I don't think these are used that often
|
||||
// I'm just going to ignore that
|
||||
const obsOffsets = {
|
||||
GMT: 0,
|
||||
EDT: -4 * 60,
|
||||
EST: -5 * 60,
|
||||
CDT: -5 * 60,
|
||||
CST: -6 * 60,
|
||||
MDT: -6 * 60,
|
||||
MST: -7 * 60,
|
||||
PDT: -7 * 60,
|
||||
PST: -8 * 60,
|
||||
};
|
||||
|
||||
function fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr) {
|
||||
const result = {
|
||||
year: yearStr.length === 2 ? untruncateYear(parseInteger(yearStr)) : parseInteger(yearStr),
|
||||
month: English.monthsShort.indexOf(monthStr) + 1,
|
||||
day: parseInteger(dayStr),
|
||||
hour: parseInteger(hourStr),
|
||||
minute: parseInteger(minuteStr),
|
||||
};
|
||||
|
||||
if (secondStr) result.second = parseInteger(secondStr);
|
||||
if (weekdayStr) {
|
||||
result.weekday =
|
||||
weekdayStr.length > 3
|
||||
? English.weekdaysLong.indexOf(weekdayStr) + 1
|
||||
: English.weekdaysShort.indexOf(weekdayStr) + 1;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
// RFC 2822/5322
|
||||
const rfc2822 =
|
||||
/^(?:(Mon|Tue|Wed|Thu|Fri|Sat|Sun),\s)?(\d{1,2})\s(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)\s(\d{2,4})\s(\d\d):(\d\d)(?::(\d\d))?\s(?:(UT|GMT|[ECMP][SD]T)|([Zz])|(?:([+-]\d\d)(\d\d)))$/;
|
||||
|
||||
function extractRFC2822(match) {
|
||||
const [
|
||||
,
|
||||
weekdayStr,
|
||||
dayStr,
|
||||
monthStr,
|
||||
yearStr,
|
||||
hourStr,
|
||||
minuteStr,
|
||||
secondStr,
|
||||
obsOffset,
|
||||
milOffset,
|
||||
offHourStr,
|
||||
offMinuteStr,
|
||||
] = match,
|
||||
result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);
|
||||
|
||||
let offset;
|
||||
if (obsOffset) {
|
||||
offset = obsOffsets[obsOffset];
|
||||
} else if (milOffset) {
|
||||
offset = 0;
|
||||
} else {
|
||||
offset = signedOffset(offHourStr, offMinuteStr);
|
||||
}
|
||||
|
||||
return [result, new FixedOffsetZone(offset)];
|
||||
}
|
||||
|
||||
function preprocessRFC2822(s) {
|
||||
// Remove comments and folding whitespace and replace multiple-spaces with a single space
|
||||
return s
|
||||
.replace(/\([^()]*\)|[\n\t]/g, " ")
|
||||
.replace(/(\s\s+)/g, " ")
|
||||
.trim();
|
||||
}
|
||||
|
||||
// http date
|
||||
|
||||
const rfc1123 =
|
||||
/^(Mon|Tue|Wed|Thu|Fri|Sat|Sun), (\d\d) (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec) (\d{4}) (\d\d):(\d\d):(\d\d) GMT$/,
|
||||
rfc850 =
|
||||
/^(Monday|Tuesday|Wednesday|Thursday|Friday|Saturday|Sunday), (\d\d)-(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)-(\d\d) (\d\d):(\d\d):(\d\d) GMT$/,
|
||||
ascii =
|
||||
/^(Mon|Tue|Wed|Thu|Fri|Sat|Sun) (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec) ( \d|\d\d) (\d\d):(\d\d):(\d\d) (\d{4})$/;
|
||||
|
||||
function extractRFC1123Or850(match) {
|
||||
const [, weekdayStr, dayStr, monthStr, yearStr, hourStr, minuteStr, secondStr] = match,
|
||||
result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);
|
||||
return [result, FixedOffsetZone.utcInstance];
|
||||
}
|
||||
|
||||
function extractASCII(match) {
|
||||
const [, weekdayStr, monthStr, dayStr, hourStr, minuteStr, secondStr, yearStr] = match,
|
||||
result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);
|
||||
return [result, FixedOffsetZone.utcInstance];
|
||||
}
|
||||
|
||||
const isoYmdWithTimeExtensionRegex = combineRegexes(isoYmdRegex, isoTimeExtensionRegex);
|
||||
const isoWeekWithTimeExtensionRegex = combineRegexes(isoWeekRegex, isoTimeExtensionRegex);
|
||||
const isoOrdinalWithTimeExtensionRegex = combineRegexes(isoOrdinalRegex, isoTimeExtensionRegex);
|
||||
const isoTimeCombinedRegex = combineRegexes(isoTimeRegex);
|
||||
|
||||
const extractISOYmdTimeAndOffset = combineExtractors(
|
||||
extractISOYmd,
|
||||
extractISOTime,
|
||||
extractISOOffset,
|
||||
extractIANAZone
|
||||
);
|
||||
const extractISOWeekTimeAndOffset = combineExtractors(
|
||||
extractISOWeekData,
|
||||
extractISOTime,
|
||||
extractISOOffset,
|
||||
extractIANAZone
|
||||
);
|
||||
const extractISOOrdinalDateAndTime = combineExtractors(
|
||||
extractISOOrdinalData,
|
||||
extractISOTime,
|
||||
extractISOOffset,
|
||||
extractIANAZone
|
||||
);
|
||||
const extractISOTimeAndOffset = combineExtractors(
|
||||
extractISOTime,
|
||||
extractISOOffset,
|
||||
extractIANAZone
|
||||
);
|
||||
|
||||
/*
|
||||
* @private
|
||||
*/
|
||||
|
||||
export function parseISODate(s) {
|
||||
return parse(
|
||||
s,
|
||||
[isoYmdWithTimeExtensionRegex, extractISOYmdTimeAndOffset],
|
||||
[isoWeekWithTimeExtensionRegex, extractISOWeekTimeAndOffset],
|
||||
[isoOrdinalWithTimeExtensionRegex, extractISOOrdinalDateAndTime],
|
||||
[isoTimeCombinedRegex, extractISOTimeAndOffset]
|
||||
);
|
||||
}
|
||||
|
||||
export function parseRFC2822Date(s) {
|
||||
return parse(preprocessRFC2822(s), [rfc2822, extractRFC2822]);
|
||||
}
|
||||
|
||||
export function parseHTTPDate(s) {
|
||||
return parse(
|
||||
s,
|
||||
[rfc1123, extractRFC1123Or850],
|
||||
[rfc850, extractRFC1123Or850],
|
||||
[ascii, extractASCII]
|
||||
);
|
||||
}
|
||||
|
||||
export function parseISODuration(s) {
|
||||
return parse(s, [isoDuration, extractISODuration]);
|
||||
}
|
||||
|
||||
const extractISOTimeOnly = combineExtractors(extractISOTime);
|
||||
|
||||
export function parseISOTimeOnly(s) {
|
||||
return parse(s, [isoTimeOnly, extractISOTimeOnly]);
|
||||
}
|
||||
|
||||
const sqlYmdWithTimeExtensionRegex = combineRegexes(sqlYmdRegex, sqlTimeExtensionRegex);
|
||||
const sqlTimeCombinedRegex = combineRegexes(sqlTimeRegex);
|
||||
|
||||
const extractISOTimeOffsetAndIANAZone = combineExtractors(
|
||||
extractISOTime,
|
||||
extractISOOffset,
|
||||
extractIANAZone
|
||||
);
|
||||
|
||||
export function parseSQL(s) {
|
||||
return parse(
|
||||
s,
|
||||
[sqlYmdWithTimeExtensionRegex, extractISOYmdTimeAndOffset],
|
||||
[sqlTimeCombinedRegex, extractISOTimeOffsetAndIANAZone]
|
||||
);
|
||||
}
|
473
priv/vendor/luxon/src/impl/tokenParser.js
vendored
473
priv/vendor/luxon/src/impl/tokenParser.js
vendored
|
@ -1,473 +0,0 @@
|
|||
import { parseMillis, isUndefined, untruncateYear, signedOffset, hasOwnProperty } from "./util.js";
|
||||
import Formatter from "./formatter.js";
|
||||
import FixedOffsetZone from "../zones/fixedOffsetZone.js";
|
||||
import IANAZone from "../zones/IANAZone.js";
|
||||
import DateTime from "../datetime.js";
|
||||
import { digitRegex, parseDigits } from "./digits.js";
|
||||
import { ConflictingSpecificationError } from "../errors.js";
|
||||
|
||||
const MISSING_FTP = "missing Intl.DateTimeFormat.formatToParts support";
|
||||
|
||||
function intUnit(regex, post = (i) => i) {
|
||||
return { regex, deser: ([s]) => post(parseDigits(s)) };
|
||||
}
|
||||
|
||||
const NBSP = String.fromCharCode(160);
|
||||
const spaceOrNBSP = `[ ${NBSP}]`;
|
||||
const spaceOrNBSPRegExp = new RegExp(spaceOrNBSP, "g");
|
||||
|
||||
function fixListRegex(s) {
|
||||
// make dots optional and also make them literal
|
||||
// make space and non breakable space characters interchangeable
|
||||
return s.replace(/\./g, "\\.?").replace(spaceOrNBSPRegExp, spaceOrNBSP);
|
||||
}
|
||||
|
||||
function stripInsensitivities(s) {
|
||||
return s
|
||||
.replace(/\./g, "") // ignore dots that were made optional
|
||||
.replace(spaceOrNBSPRegExp, " ") // interchange space and nbsp
|
||||
.toLowerCase();
|
||||
}
|
||||
|
||||
function oneOf(strings, startIndex) {
|
||||
if (strings === null) {
|
||||
return null;
|
||||
} else {
|
||||
return {
|
||||
regex: RegExp(strings.map(fixListRegex).join("|")),
|
||||
deser: ([s]) =>
|
||||
strings.findIndex((i) => stripInsensitivities(s) === stripInsensitivities(i)) + startIndex,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
function offset(regex, groups) {
|
||||
return { regex, deser: ([, h, m]) => signedOffset(h, m), groups };
|
||||
}
|
||||
|
||||
function simple(regex) {
|
||||
return { regex, deser: ([s]) => s };
|
||||
}
|
||||
|
||||
function escapeToken(value) {
|
||||
return value.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&");
|
||||
}
|
||||
|
||||
/**
|
||||
* @param token
|
||||
* @param {Locale} loc
|
||||
*/
|
||||
function unitForToken(token, loc) {
|
||||
const one = digitRegex(loc),
|
||||
two = digitRegex(loc, "{2}"),
|
||||
three = digitRegex(loc, "{3}"),
|
||||
four = digitRegex(loc, "{4}"),
|
||||
six = digitRegex(loc, "{6}"),
|
||||
oneOrTwo = digitRegex(loc, "{1,2}"),
|
||||
oneToThree = digitRegex(loc, "{1,3}"),
|
||||
oneToSix = digitRegex(loc, "{1,6}"),
|
||||
oneToNine = digitRegex(loc, "{1,9}"),
|
||||
twoToFour = digitRegex(loc, "{2,4}"),
|
||||
fourToSix = digitRegex(loc, "{4,6}"),
|
||||
literal = (t) => ({ regex: RegExp(escapeToken(t.val)), deser: ([s]) => s, literal: true }),
|
||||
unitate = (t) => {
|
||||
if (token.literal) {
|
||||
return literal(t);
|
||||
}
|
||||
switch (t.val) {
|
||||
// era
|
||||
case "G":
|
||||
return oneOf(loc.eras("short"), 0);
|
||||
case "GG":
|
||||
return oneOf(loc.eras("long"), 0);
|
||||
// years
|
||||
case "y":
|
||||
return intUnit(oneToSix);
|
||||
case "yy":
|
||||
return intUnit(twoToFour, untruncateYear);
|
||||
case "yyyy":
|
||||
return intUnit(four);
|
||||
case "yyyyy":
|
||||
return intUnit(fourToSix);
|
||||
case "yyyyyy":
|
||||
return intUnit(six);
|
||||
// months
|
||||
case "M":
|
||||
return intUnit(oneOrTwo);
|
||||
case "MM":
|
||||
return intUnit(two);
|
||||
case "MMM":
|
||||
return oneOf(loc.months("short", true), 1);
|
||||
case "MMMM":
|
||||
return oneOf(loc.months("long", true), 1);
|
||||
case "L":
|
||||
return intUnit(oneOrTwo);
|
||||
case "LL":
|
||||
return intUnit(two);
|
||||
case "LLL":
|
||||
return oneOf(loc.months("short", false), 1);
|
||||
case "LLLL":
|
||||
return oneOf(loc.months("long", false), 1);
|
||||
// dates
|
||||
case "d":
|
||||
return intUnit(oneOrTwo);
|
||||
case "dd":
|
||||
return intUnit(two);
|
||||
// ordinals
|
||||
case "o":
|
||||
return intUnit(oneToThree);
|
||||
case "ooo":
|
||||
return intUnit(three);
|
||||
// time
|
||||
case "HH":
|
||||
return intUnit(two);
|
||||
case "H":
|
||||
return intUnit(oneOrTwo);
|
||||
case "hh":
|
||||
return intUnit(two);
|
||||
case "h":
|
||||
return intUnit(oneOrTwo);
|
||||
case "mm":
|
||||
return intUnit(two);
|
||||
case "m":
|
||||
return intUnit(oneOrTwo);
|
||||
case "q":
|
||||
return intUnit(oneOrTwo);
|
||||
case "qq":
|
||||
return intUnit(two);
|
||||
case "s":
|
||||
return intUnit(oneOrTwo);
|
||||
case "ss":
|
||||
return intUnit(two);
|
||||
case "S":
|
||||
return intUnit(oneToThree);
|
||||
case "SSS":
|
||||
return intUnit(three);
|
||||
case "u":
|
||||
return simple(oneToNine);
|
||||
case "uu":
|
||||
return simple(oneOrTwo);
|
||||
case "uuu":
|
||||
return intUnit(one);
|
||||
// meridiem
|
||||
case "a":
|
||||
return oneOf(loc.meridiems(), 0);
|
||||
// weekYear (k)
|
||||
case "kkkk":
|
||||
return intUnit(four);
|
||||
case "kk":
|
||||
return intUnit(twoToFour, untruncateYear);
|
||||
// weekNumber (W)
|
||||
case "W":
|
||||
return intUnit(oneOrTwo);
|
||||
case "WW":
|
||||
return intUnit(two);
|
||||
// weekdays
|
||||
case "E":
|
||||
case "c":
|
||||
return intUnit(one);
|
||||
case "EEE":
|
||||
return oneOf(loc.weekdays("short", false), 1);
|
||||
case "EEEE":
|
||||
return oneOf(loc.weekdays("long", false), 1);
|
||||
case "ccc":
|
||||
return oneOf(loc.weekdays("short", true), 1);
|
||||
case "cccc":
|
||||
return oneOf(loc.weekdays("long", true), 1);
|
||||
// offset/zone
|
||||
case "Z":
|
||||
case "ZZ":
|
||||
return offset(new RegExp(`([+-]${oneOrTwo.source})(?::(${two.source}))?`), 2);
|
||||
case "ZZZ":
|
||||
return offset(new RegExp(`([+-]${oneOrTwo.source})(${two.source})?`), 2);
|
||||
// we don't support ZZZZ (PST) or ZZZZZ (Pacific Standard Time) in parsing
|
||||
// because we don't have any way to figure out what they are
|
||||
case "z":
|
||||
return simple(/[a-z_+-/]{1,256}?/i);
|
||||
// this special-case "token" represents a place where a macro-token expanded into a white-space literal
|
||||
// in this case we accept any non-newline white-space
|
||||
case " ":
|
||||
return simple(/[^\S\n\r]/);
|
||||
default:
|
||||
return literal(t);
|
||||
}
|
||||
};
|
||||
|
||||
const unit = unitate(token) || {
|
||||
invalidReason: MISSING_FTP,
|
||||
};
|
||||
|
||||
unit.token = token;
|
||||
|
||||
return unit;
|
||||
}
|
||||
|
||||
const partTypeStyleToTokenVal = {
|
||||
year: {
|
||||
"2-digit": "yy",
|
||||
numeric: "yyyyy",
|
||||
},
|
||||
month: {
|
||||
numeric: "M",
|
||||
"2-digit": "MM",
|
||||
short: "MMM",
|
||||
long: "MMMM",
|
||||
},
|
||||
day: {
|
||||
numeric: "d",
|
||||
"2-digit": "dd",
|
||||
},
|
||||
weekday: {
|
||||
short: "EEE",
|
||||
long: "EEEE",
|
||||
},
|
||||
dayperiod: "a",
|
||||
dayPeriod: "a",
|
||||
hour12: {
|
||||
numeric: "h",
|
||||
"2-digit": "hh",
|
||||
},
|
||||
hour24: {
|
||||
numeric: "H",
|
||||
"2-digit": "HH",
|
||||
},
|
||||
minute: {
|
||||
numeric: "m",
|
||||
"2-digit": "mm",
|
||||
},
|
||||
second: {
|
||||
numeric: "s",
|
||||
"2-digit": "ss",
|
||||
},
|
||||
timeZoneName: {
|
||||
long: "ZZZZZ",
|
||||
short: "ZZZ",
|
||||
},
|
||||
};
|
||||
|
||||
function tokenForPart(part, formatOpts, resolvedOpts) {
|
||||
const { type, value } = part;
|
||||
|
||||
if (type === "literal") {
|
||||
const isSpace = /^\s+$/.test(value);
|
||||
return {
|
||||
literal: !isSpace,
|
||||
val: isSpace ? " " : value,
|
||||
};
|
||||
}
|
||||
|
||||
const style = formatOpts[type];
|
||||
|
||||
// The user might have explicitly specified hour12 or hourCycle
|
||||
// if so, respect their decision
|
||||
// if not, refer back to the resolvedOpts, which are based on the locale
|
||||
let actualType = type;
|
||||
if (type === "hour") {
|
||||
if (formatOpts.hour12 != null) {
|
||||
actualType = formatOpts.hour12 ? "hour12" : "hour24";
|
||||
} else if (formatOpts.hourCycle != null) {
|
||||
if (formatOpts.hourCycle === "h11" || formatOpts.hourCycle === "h12") {
|
||||
actualType = "hour12";
|
||||
} else {
|
||||
actualType = "hour24";
|
||||
}
|
||||
} else {
|
||||
// tokens only differentiate between 24 hours or not,
|
||||
// so we do not need to check hourCycle here, which is less supported anyways
|
||||
actualType = resolvedOpts.hour12 ? "hour12" : "hour24";
|
||||
}
|
||||
}
|
||||
let val = partTypeStyleToTokenVal[actualType];
|
||||
if (typeof val === "object") {
|
||||
val = val[style];
|
||||
}
|
||||
|
||||
if (val) {
|
||||
return {
|
||||
literal: false,
|
||||
val,
|
||||
};
|
||||
}
|
||||
|
||||
return undefined;
|
||||
}
|
||||
|
||||
function buildRegex(units) {
|
||||
const re = units.map((u) => u.regex).reduce((f, r) => `${f}(${r.source})`, "");
|
||||
return [`^${re}$`, units];
|
||||
}
|
||||
|
||||
function match(input, regex, handlers) {
|
||||
const matches = input.match(regex);
|
||||
|
||||
if (matches) {
|
||||
const all = {};
|
||||
let matchIndex = 1;
|
||||
for (const i in handlers) {
|
||||
if (hasOwnProperty(handlers, i)) {
|
||||
const h = handlers[i],
|
||||
groups = h.groups ? h.groups + 1 : 1;
|
||||
if (!h.literal && h.token) {
|
||||
all[h.token.val[0]] = h.deser(matches.slice(matchIndex, matchIndex + groups));
|
||||
}
|
||||
matchIndex += groups;
|
||||
}
|
||||
}
|
||||
return [matches, all];
|
||||
} else {
|
||||
return [matches, {}];
|
||||
}
|
||||
}
|
||||
|
||||
function dateTimeFromMatches(matches) {
|
||||
const toField = (token) => {
|
||||
switch (token) {
|
||||
case "S":
|
||||
return "millisecond";
|
||||
case "s":
|
||||
return "second";
|
||||
case "m":
|
||||
return "minute";
|
||||
case "h":
|
||||
case "H":
|
||||
return "hour";
|
||||
case "d":
|
||||
return "day";
|
||||
case "o":
|
||||
return "ordinal";
|
||||
case "L":
|
||||
case "M":
|
||||
return "month";
|
||||
case "y":
|
||||
return "year";
|
||||
case "E":
|
||||
case "c":
|
||||
return "weekday";
|
||||
case "W":
|
||||
return "weekNumber";
|
||||
case "k":
|
||||
return "weekYear";
|
||||
case "q":
|
||||
return "quarter";
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
let zone = null;
|
||||
let specificOffset;
|
||||
if (!isUndefined(matches.z)) {
|
||||
zone = IANAZone.create(matches.z);
|
||||
}
|
||||
|
||||
if (!isUndefined(matches.Z)) {
|
||||
if (!zone) {
|
||||
zone = new FixedOffsetZone(matches.Z);
|
||||
}
|
||||
specificOffset = matches.Z;
|
||||
}
|
||||
|
||||
if (!isUndefined(matches.q)) {
|
||||
matches.M = (matches.q - 1) * 3 + 1;
|
||||
}
|
||||
|
||||
if (!isUndefined(matches.h)) {
|
||||
if (matches.h < 12 && matches.a === 1) {
|
||||
matches.h += 12;
|
||||
} else if (matches.h === 12 && matches.a === 0) {
|
||||
matches.h = 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (matches.G === 0 && matches.y) {
|
||||
matches.y = -matches.y;
|
||||
}
|
||||
|
||||
if (!isUndefined(matches.u)) {
|
||||
matches.S = parseMillis(matches.u);
|
||||
}
|
||||
|
||||
const vals = Object.keys(matches).reduce((r, k) => {
|
||||
const f = toField(k);
|
||||
if (f) {
|
||||
r[f] = matches[k];
|
||||
}
|
||||
|
||||
return r;
|
||||
}, {});
|
||||
|
||||
return [vals, zone, specificOffset];
|
||||
}
|
||||
|
||||
let dummyDateTimeCache = null;
|
||||
|
||||
function getDummyDateTime() {
|
||||
if (!dummyDateTimeCache) {
|
||||
dummyDateTimeCache = DateTime.fromMillis(1555555555555);
|
||||
}
|
||||
|
||||
return dummyDateTimeCache;
|
||||
}
|
||||
|
||||
function maybeExpandMacroToken(token, locale) {
|
||||
if (token.literal) {
|
||||
return token;
|
||||
}
|
||||
|
||||
const formatOpts = Formatter.macroTokenToFormatOpts(token.val);
|
||||
const tokens = formatOptsToTokens(formatOpts, locale);
|
||||
|
||||
if (tokens == null || tokens.includes(undefined)) {
|
||||
return token;
|
||||
}
|
||||
|
||||
return tokens;
|
||||
}
|
||||
|
||||
export function expandMacroTokens(tokens, locale) {
|
||||
return Array.prototype.concat(...tokens.map((t) => maybeExpandMacroToken(t, locale)));
|
||||
}
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
export function explainFromTokens(locale, input, format) {
|
||||
const tokens = expandMacroTokens(Formatter.parseFormat(format), locale),
|
||||
units = tokens.map((t) => unitForToken(t, locale)),
|
||||
disqualifyingUnit = units.find((t) => t.invalidReason);
|
||||
|
||||
if (disqualifyingUnit) {
|
||||
return { input, tokens, invalidReason: disqualifyingUnit.invalidReason };
|
||||
} else {
|
||||
const [regexString, handlers] = buildRegex(units),
|
||||
regex = RegExp(regexString, "i"),
|
||||
[rawMatches, matches] = match(input, regex, handlers),
|
||||
[result, zone, specificOffset] = matches
|
||||
? dateTimeFromMatches(matches)
|
||||
: [null, null, undefined];
|
||||
if (hasOwnProperty(matches, "a") && hasOwnProperty(matches, "H")) {
|
||||
throw new ConflictingSpecificationError(
|
||||
"Can't include meridiem when specifying 24-hour format"
|
||||
);
|
||||
}
|
||||
return { input, tokens, regex, rawMatches, matches, result, zone, specificOffset };
|
||||
}
|
||||
}
|
||||
|
||||
export function parseFromTokens(locale, input, format) {
|
||||
const { result, zone, specificOffset, invalidReason } = explainFromTokens(locale, input, format);
|
||||
return [result, zone, specificOffset, invalidReason];
|
||||
}
|
||||
|
||||
export function formatOptsToTokens(formatOpts, locale) {
|
||||
if (!formatOpts) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const formatter = Formatter.create(locale, formatOpts);
|
||||
const df = formatter.dtFormatter(getDummyDateTime());
|
||||
const parts = df.formatToParts();
|
||||
const resolvedOpts = df.resolvedOptions();
|
||||
return parts.map((p) => tokenForPart(p, formatOpts, resolvedOpts));
|
||||
}
|
309
priv/vendor/luxon/src/impl/util.js
vendored
309
priv/vendor/luxon/src/impl/util.js
vendored
|
@ -1,309 +0,0 @@
|
|||
/*
|
||||
This is just a junk drawer, containing anything used across multiple classes.
|
||||
Because Luxon is small(ish), this should stay small and we won't worry about splitting
|
||||
it up into, say, parsingUtil.js and basicUtil.js and so on. But they are divided up by feature area.
|
||||
*/
|
||||
|
||||
import { InvalidArgumentError } from "../errors.js";
|
||||
import Settings from "../settings.js";
|
||||
import { dayOfWeek, isoWeekdayToLocal } from "./conversions.js";
|
||||
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
// TYPES
|
||||
|
||||
export function isUndefined(o) {
|
||||
return typeof o === "undefined";
|
||||
}
|
||||
|
||||
export function isNumber(o) {
|
||||
return typeof o === "number";
|
||||
}
|
||||
|
||||
export function isInteger(o) {
|
||||
return typeof o === "number" && o % 1 === 0;
|
||||
}
|
||||
|
||||
export function isString(o) {
|
||||
return typeof o === "string";
|
||||
}
|
||||
|
||||
export function isDate(o) {
|
||||
return Object.prototype.toString.call(o) === "[object Date]";
|
||||
}
|
||||
|
||||
// CAPABILITIES
|
||||
|
||||
export function hasRelative() {
|
||||
try {
|
||||
return typeof Intl !== "undefined" && !!Intl.RelativeTimeFormat;
|
||||
} catch (e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
export function hasLocaleWeekInfo() {
|
||||
try {
|
||||
return (
|
||||
typeof Intl !== "undefined" &&
|
||||
!!Intl.Locale &&
|
||||
("weekInfo" in Intl.Locale.prototype || "getWeekInfo" in Intl.Locale.prototype)
|
||||
);
|
||||
} catch (e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// OBJECTS AND ARRAYS
|
||||
|
||||
export function maybeArray(thing) {
|
||||
return Array.isArray(thing) ? thing : [thing];
|
||||
}
|
||||
|
||||
export function bestBy(arr, by, compare) {
|
||||
if (arr.length === 0) {
|
||||
return undefined;
|
||||
}
|
||||
return arr.reduce((best, next) => {
|
||||
const pair = [by(next), next];
|
||||
if (!best) {
|
||||
return pair;
|
||||
} else if (compare(best[0], pair[0]) === best[0]) {
|
||||
return best;
|
||||
} else {
|
||||
return pair;
|
||||
}
|
||||
}, null)[1];
|
||||
}
|
||||
|
||||
export function pick(obj, keys) {
|
||||
return keys.reduce((a, k) => {
|
||||
a[k] = obj[k];
|
||||
return a;
|
||||
}, {});
|
||||
}
|
||||
|
||||
export function hasOwnProperty(obj, prop) {
|
||||
return Object.prototype.hasOwnProperty.call(obj, prop);
|
||||
}
|
||||
|
||||
export function validateWeekSettings(settings) {
|
||||
if (settings == null) {
|
||||
return null;
|
||||
} else if (typeof settings !== "object") {
|
||||
throw new InvalidArgumentError("Week settings must be an object");
|
||||
} else {
|
||||
if (
|
||||
!integerBetween(settings.firstDay, 1, 7) ||
|
||||
!integerBetween(settings.minimalDays, 1, 7) ||
|
||||
!Array.isArray(settings.weekend) ||
|
||||
settings.weekend.some((v) => !integerBetween(v, 1, 7))
|
||||
) {
|
||||
throw new InvalidArgumentError("Invalid week settings");
|
||||
}
|
||||
return {
|
||||
firstDay: settings.firstDay,
|
||||
minimalDays: settings.minimalDays,
|
||||
weekend: Array.from(settings.weekend),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
// NUMBERS AND STRINGS
|
||||
|
||||
export function integerBetween(thing, bottom, top) {
|
||||
return isInteger(thing) && thing >= bottom && thing <= top;
|
||||
}
|
||||
|
||||
// x % n but takes the sign of n instead of x
|
||||
export function floorMod(x, n) {
|
||||
return x - n * Math.floor(x / n);
|
||||
}
|
||||
|
||||
export function padStart(input, n = 2) {
|
||||
const isNeg = input < 0;
|
||||
let padded;
|
||||
if (isNeg) {
|
||||
padded = "-" + ("" + -input).padStart(n, "0");
|
||||
} else {
|
||||
padded = ("" + input).padStart(n, "0");
|
||||
}
|
||||
return padded;
|
||||
}
|
||||
|
||||
export function parseInteger(string) {
|
||||
if (isUndefined(string) || string === null || string === "") {
|
||||
return undefined;
|
||||
} else {
|
||||
return parseInt(string, 10);
|
||||
}
|
||||
}
|
||||
|
||||
export function parseFloating(string) {
|
||||
if (isUndefined(string) || string === null || string === "") {
|
||||
return undefined;
|
||||
} else {
|
||||
return parseFloat(string);
|
||||
}
|
||||
}
|
||||
|
||||
export function parseMillis(fraction) {
|
||||
// Return undefined (instead of 0) in these cases, where fraction is not set
|
||||
if (isUndefined(fraction) || fraction === null || fraction === "") {
|
||||
return undefined;
|
||||
} else {
|
||||
const f = parseFloat("0." + fraction) * 1000;
|
||||
return Math.floor(f);
|
||||
}
|
||||
}
|
||||
|
||||
export function roundTo(number, digits, towardZero = false) {
|
||||
const factor = 10 ** digits,
|
||||
rounder = towardZero ? Math.trunc : Math.round;
|
||||
return rounder(number * factor) / factor;
|
||||
}
|
||||
|
||||
// DATE BASICS
|
||||
|
||||
export function isLeapYear(year) {
|
||||
return year % 4 === 0 && (year % 100 !== 0 || year % 400 === 0);
|
||||
}
|
||||
|
||||
export function daysInYear(year) {
|
||||
return isLeapYear(year) ? 366 : 365;
|
||||
}
|
||||
|
||||
export function daysInMonth(year, month) {
|
||||
const modMonth = floorMod(month - 1, 12) + 1,
|
||||
modYear = year + (month - modMonth) / 12;
|
||||
|
||||
if (modMonth === 2) {
|
||||
return isLeapYear(modYear) ? 29 : 28;
|
||||
} else {
|
||||
return [31, null, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31][modMonth - 1];
|
||||
}
|
||||
}
|
||||
|
||||
// convert a calendar object to a local timestamp (epoch, but with the offset baked in)
|
||||
export function objToLocalTS(obj) {
|
||||
let d = Date.UTC(
|
||||
obj.year,
|
||||
obj.month - 1,
|
||||
obj.day,
|
||||
obj.hour,
|
||||
obj.minute,
|
||||
obj.second,
|
||||
obj.millisecond
|
||||
);
|
||||
|
||||
// for legacy reasons, years between 0 and 99 are interpreted as 19XX; revert that
|
||||
if (obj.year < 100 && obj.year >= 0) {
|
||||
d = new Date(d);
|
||||
// set the month and day again, this is necessary because year 2000 is a leap year, but year 100 is not
|
||||
// so if obj.year is in 99, but obj.day makes it roll over into year 100,
|
||||
// the calculations done by Date.UTC are using year 2000 - which is incorrect
|
||||
d.setUTCFullYear(obj.year, obj.month - 1, obj.day);
|
||||
}
|
||||
return +d;
|
||||
}
|
||||
|
||||
// adapted from moment.js: https://github.com/moment/moment/blob/000ac1800e620f770f4eb31b5ae908f6167b0ab2/src/lib/units/week-calendar-utils.js
|
||||
function firstWeekOffset(year, minDaysInFirstWeek, startOfWeek) {
|
||||
const fwdlw = isoWeekdayToLocal(dayOfWeek(year, 1, minDaysInFirstWeek), startOfWeek);
|
||||
return -fwdlw + minDaysInFirstWeek - 1;
|
||||
}
|
||||
|
||||
export function weeksInWeekYear(weekYear, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
||||
const weekOffset = firstWeekOffset(weekYear, minDaysInFirstWeek, startOfWeek);
|
||||
const weekOffsetNext = firstWeekOffset(weekYear + 1, minDaysInFirstWeek, startOfWeek);
|
||||
return (daysInYear(weekYear) - weekOffset + weekOffsetNext) / 7;
|
||||
}
|
||||
|
||||
export function untruncateYear(year) {
|
||||
if (year > 99) {
|
||||
return year;
|
||||
} else return year > Settings.twoDigitCutoffYear ? 1900 + year : 2000 + year;
|
||||
}
|
||||
|
||||
// PARSING
|
||||
|
||||
export function parseZoneInfo(ts, offsetFormat, locale, timeZone = null) {
|
||||
const date = new Date(ts),
|
||||
intlOpts = {
|
||||
hourCycle: "h23",
|
||||
year: "numeric",
|
||||
month: "2-digit",
|
||||
day: "2-digit",
|
||||
hour: "2-digit",
|
||||
minute: "2-digit",
|
||||
};
|
||||
|
||||
if (timeZone) {
|
||||
intlOpts.timeZone = timeZone;
|
||||
}
|
||||
|
||||
const modified = { timeZoneName: offsetFormat, ...intlOpts };
|
||||
|
||||
const parsed = new Intl.DateTimeFormat(locale, modified)
|
||||
.formatToParts(date)
|
||||
.find((m) => m.type.toLowerCase() === "timezonename");
|
||||
return parsed ? parsed.value : null;
|
||||
}
|
||||
|
||||
// signedOffset('-5', '30') -> -330
|
||||
export function signedOffset(offHourStr, offMinuteStr) {
|
||||
let offHour = parseInt(offHourStr, 10);
|
||||
|
||||
// don't || this because we want to preserve -0
|
||||
if (Number.isNaN(offHour)) {
|
||||
offHour = 0;
|
||||
}
|
||||
|
||||
const offMin = parseInt(offMinuteStr, 10) || 0,
|
||||
offMinSigned = offHour < 0 || Object.is(offHour, -0) ? -offMin : offMin;
|
||||
return offHour * 60 + offMinSigned;
|
||||
}
|
||||
|
||||
// COERCION
|
||||
|
||||
export function asNumber(value) {
|
||||
const numericValue = Number(value);
|
||||
if (typeof value === "boolean" || value === "" || Number.isNaN(numericValue))
|
||||
throw new InvalidArgumentError(`Invalid unit value ${value}`);
|
||||
return numericValue;
|
||||
}
|
||||
|
||||
export function normalizeObject(obj, normalizer) {
|
||||
const normalized = {};
|
||||
for (const u in obj) {
|
||||
if (hasOwnProperty(obj, u)) {
|
||||
const v = obj[u];
|
||||
if (v === undefined || v === null) continue;
|
||||
normalized[normalizer(u)] = asNumber(v);
|
||||
}
|
||||
}
|
||||
return normalized;
|
||||
}
|
||||
|
||||
export function formatOffset(offset, format) {
|
||||
const hours = Math.trunc(Math.abs(offset / 60)),
|
||||
minutes = Math.trunc(Math.abs(offset % 60)),
|
||||
sign = offset >= 0 ? "+" : "-";
|
||||
|
||||
switch (format) {
|
||||
case "short":
|
||||
return `${sign}${padStart(hours, 2)}:${padStart(minutes, 2)}`;
|
||||
case "narrow":
|
||||
return `${sign}${hours}${minutes > 0 ? `:${minutes}` : ""}`;
|
||||
case "techie":
|
||||
return `${sign}${padStart(hours, 2)}${padStart(minutes, 2)}`;
|
||||
default:
|
||||
throw new RangeError(`Value format ${format} is out of range for property format`);
|
||||
}
|
||||
}
|
||||
|
||||
export function timeObject(obj) {
|
||||
return pick(obj, ["hour", "minute", "second", "millisecond"]);
|
||||
}
|
34
priv/vendor/luxon/src/impl/zoneUtil.js
vendored
34
priv/vendor/luxon/src/impl/zoneUtil.js
vendored
|
@ -1,34 +0,0 @@
|
|||
/**
|
||||
* @private
|
||||
*/
|
||||
|
||||
import Zone from "../zone.js";
|
||||
import IANAZone from "../zones/IANAZone.js";
|
||||
import FixedOffsetZone from "../zones/fixedOffsetZone.js";
|
||||
import InvalidZone from "../zones/invalidZone.js";
|
||||
|
||||
import { isUndefined, isString, isNumber } from "./util.js";
|
||||
import SystemZone from "../zones/systemZone.js";
|
||||
|
||||
export function normalizeZone(input, defaultZone) {
|
||||
let offset;
|
||||
if (isUndefined(input) || input === null) {
|
||||
return defaultZone;
|
||||
} else if (input instanceof Zone) {
|
||||
return input;
|
||||
} else if (isString(input)) {
|
||||
const lowered = input.toLowerCase();
|
||||
if (lowered === "default") return defaultZone;
|
||||
else if (lowered === "local" || lowered === "system") return SystemZone.instance;
|
||||
else if (lowered === "utc" || lowered === "gmt") return FixedOffsetZone.utcInstance;
|
||||
else return FixedOffsetZone.parseSpecifier(lowered) || IANAZone.create(input);
|
||||
} else if (isNumber(input)) {
|
||||
return FixedOffsetZone.instance(input);
|
||||
} else if (typeof input === "object" && "offset" in input && typeof input.offset === "function") {
|
||||
// This is dumb, but the instanceof check above doesn't seem to really work
|
||||
// so we're duck checking it
|
||||
return input;
|
||||
} else {
|
||||
return new InvalidZone(input);
|
||||
}
|
||||
}
|
205
priv/vendor/luxon/src/info.js
vendored
205
priv/vendor/luxon/src/info.js
vendored
|
@ -1,205 +0,0 @@
|
|||
import DateTime from "./datetime.js";
|
||||
import Settings from "./settings.js";
|
||||
import Locale from "./impl/locale.js";
|
||||
import IANAZone from "./zones/IANAZone.js";
|
||||
import { normalizeZone } from "./impl/zoneUtil.js";
|
||||
|
||||
import { hasLocaleWeekInfo, hasRelative } from "./impl/util.js";
|
||||
|
||||
/**
|
||||
* The Info class contains static methods for retrieving general time and date related data. For example, it has methods for finding out if a time zone has a DST, for listing the months in any supported locale, and for discovering which of Luxon features are available in the current environment.
|
||||
*/
|
||||
export default class Info {
|
||||
/**
|
||||
* Return whether the specified zone contains a DST.
|
||||
* @param {string|Zone} [zone='local'] - Zone to check. Defaults to the environment's local zone.
|
||||
* @return {boolean}
|
||||
*/
|
||||
static hasDST(zone = Settings.defaultZone) {
|
||||
const proto = DateTime.now().setZone(zone).set({ month: 12 });
|
||||
|
||||
return !zone.isUniversal && proto.offset !== proto.set({ month: 6 }).offset;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether the specified zone is a valid IANA specifier.
|
||||
* @param {string} zone - Zone to check
|
||||
* @return {boolean}
|
||||
*/
|
||||
static isValidIANAZone(zone) {
|
||||
return IANAZone.isValidZone(zone);
|
||||
}
|
||||
|
||||
/**
|
||||
* Converts the input into a {@link Zone} instance.
|
||||
*
|
||||
* * If `input` is already a Zone instance, it is returned unchanged.
|
||||
* * If `input` is a string containing a valid time zone name, a Zone instance
|
||||
* with that name is returned.
|
||||
* * If `input` is a string that doesn't refer to a known time zone, a Zone
|
||||
* instance with {@link Zone#isValid} == false is returned.
|
||||
* * If `input is a number, a Zone instance with the specified fixed offset
|
||||
* in minutes is returned.
|
||||
* * If `input` is `null` or `undefined`, the default zone is returned.
|
||||
* @param {string|Zone|number} [input] - the value to be converted
|
||||
* @return {Zone}
|
||||
*/
|
||||
static normalizeZone(input) {
|
||||
return normalizeZone(input, Settings.defaultZone);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the weekday on which the week starts according to the given locale.
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @returns {number} the start of the week, 1 for Monday through 7 for Sunday
|
||||
*/
|
||||
static getStartOfWeek({ locale = null, locObj = null } = {}) {
|
||||
return (locObj || Locale.create(locale)).getStartOfWeek();
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the minimum number of days necessary in a week before it is considered part of the next year according
|
||||
* to the given locale.
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @returns {number}
|
||||
*/
|
||||
static getMinimumDaysInFirstWeek({ locale = null, locObj = null } = {}) {
|
||||
return (locObj || Locale.create(locale)).getMinDaysInFirstWeek();
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the weekdays, which are considered the weekend according to the given locale
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @returns {number[]} an array of weekdays, 1 for Monday through 7 for Sunday
|
||||
*/
|
||||
static getWeekendWeekdays({ locale = null, locObj = null } = {}) {
|
||||
// copy the array, because we cache it internally
|
||||
return (locObj || Locale.create(locale)).getWeekendDays().slice();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of standalone month names.
|
||||
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat
|
||||
* @param {string} [length='long'] - the length of the month representation, such as "numeric", "2-digit", "narrow", "short", "long"
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.numberingSystem=null] - the numbering system
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @param {string} [opts.outputCalendar='gregory'] - the calendar
|
||||
* @example Info.months()[0] //=> 'January'
|
||||
* @example Info.months('short')[0] //=> 'Jan'
|
||||
* @example Info.months('numeric')[0] //=> '1'
|
||||
* @example Info.months('short', { locale: 'fr-CA' } )[0] //=> 'janv.'
|
||||
* @example Info.months('numeric', { locale: 'ar' })[0] //=> '١'
|
||||
* @example Info.months('long', { outputCalendar: 'islamic' })[0] //=> 'Rabiʻ I'
|
||||
* @return {Array}
|
||||
*/
|
||||
static months(
|
||||
length = "long",
|
||||
{ locale = null, numberingSystem = null, locObj = null, outputCalendar = "gregory" } = {}
|
||||
) {
|
||||
return (locObj || Locale.create(locale, numberingSystem, outputCalendar)).months(length);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of format month names.
|
||||
* Format months differ from standalone months in that they're meant to appear next to the day of the month. In some languages, that
|
||||
* changes the string.
|
||||
* See {@link Info#months}
|
||||
* @param {string} [length='long'] - the length of the month representation, such as "numeric", "2-digit", "narrow", "short", "long"
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.numberingSystem=null] - the numbering system
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @param {string} [opts.outputCalendar='gregory'] - the calendar
|
||||
* @return {Array}
|
||||
*/
|
||||
static monthsFormat(
|
||||
length = "long",
|
||||
{ locale = null, numberingSystem = null, locObj = null, outputCalendar = "gregory" } = {}
|
||||
) {
|
||||
return (locObj || Locale.create(locale, numberingSystem, outputCalendar)).months(length, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of standalone week names.
|
||||
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat
|
||||
* @param {string} [length='long'] - the length of the weekday representation, such as "narrow", "short", "long".
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @param {string} [opts.numberingSystem=null] - the numbering system
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @example Info.weekdays()[0] //=> 'Monday'
|
||||
* @example Info.weekdays('short')[0] //=> 'Mon'
|
||||
* @example Info.weekdays('short', { locale: 'fr-CA' })[0] //=> 'lun.'
|
||||
* @example Info.weekdays('short', { locale: 'ar' })[0] //=> 'الاثنين'
|
||||
* @return {Array}
|
||||
*/
|
||||
static weekdays(length = "long", { locale = null, numberingSystem = null, locObj = null } = {}) {
|
||||
return (locObj || Locale.create(locale, numberingSystem, null)).weekdays(length);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of format week names.
|
||||
* Format weekdays differ from standalone weekdays in that they're meant to appear next to more date information. In some languages, that
|
||||
* changes the string.
|
||||
* See {@link Info#weekdays}
|
||||
* @param {string} [length='long'] - the length of the month representation, such as "narrow", "short", "long".
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale=null] - the locale code
|
||||
* @param {string} [opts.numberingSystem=null] - the numbering system
|
||||
* @param {string} [opts.locObj=null] - an existing locale object to use
|
||||
* @return {Array}
|
||||
*/
|
||||
static weekdaysFormat(
|
||||
length = "long",
|
||||
{ locale = null, numberingSystem = null, locObj = null } = {}
|
||||
) {
|
||||
return (locObj || Locale.create(locale, numberingSystem, null)).weekdays(length, true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of meridiems.
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @example Info.meridiems() //=> [ 'AM', 'PM' ]
|
||||
* @example Info.meridiems({ locale: 'my' }) //=> [ 'နံနက်', 'ညနေ' ]
|
||||
* @return {Array}
|
||||
*/
|
||||
static meridiems({ locale = null } = {}) {
|
||||
return Locale.create(locale).meridiems();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of eras, such as ['BC', 'AD']. The locale can be specified, but the calendar system is always Gregorian.
|
||||
* @param {string} [length='short'] - the length of the era representation, such as "short" or "long".
|
||||
* @param {Object} opts - options
|
||||
* @param {string} [opts.locale] - the locale code
|
||||
* @example Info.eras() //=> [ 'BC', 'AD' ]
|
||||
* @example Info.eras('long') //=> [ 'Before Christ', 'Anno Domini' ]
|
||||
* @example Info.eras('long', { locale: 'fr' }) //=> [ 'avant Jésus-Christ', 'après Jésus-Christ' ]
|
||||
* @return {Array}
|
||||
*/
|
||||
static eras(length = "short", { locale = null } = {}) {
|
||||
return Locale.create(locale, null, "gregory").eras(length);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the set of available features in this environment.
|
||||
* Some features of Luxon are not available in all environments. For example, on older browsers, relative time formatting support is not available. Use this function to figure out if that's the case.
|
||||
* Keys:
|
||||
* * `relative`: whether this environment supports relative time formatting
|
||||
* * `localeWeek`: whether this environment supports different weekdays for the start of the week based on the locale
|
||||
* @example Info.features() //=> { relative: false, localeWeek: true }
|
||||
* @return {Object}
|
||||
*/
|
||||
static features() {
|
||||
return { relative: hasRelative(), localeWeek: hasLocaleWeekInfo() };
|
||||
}
|
||||
}
|
657
priv/vendor/luxon/src/interval.js
vendored
657
priv/vendor/luxon/src/interval.js
vendored
|
@ -1,657 +0,0 @@
|
|||
import DateTime, { friendlyDateTime } from "./datetime.js";
|
||||
import Duration from "./duration.js";
|
||||
import Settings from "./settings.js";
|
||||
import { InvalidArgumentError, InvalidIntervalError } from "./errors.js";
|
||||
import Invalid from "./impl/invalid.js";
|
||||
import Formatter from "./impl/formatter.js";
|
||||
import * as Formats from "./impl/formats.js";
|
||||
|
||||
const INVALID = "Invalid Interval";
|
||||
|
||||
// checks if the start is equal to or before the end
|
||||
function validateStartEnd(start, end) {
|
||||
if (!start || !start.isValid) {
|
||||
return Interval.invalid("missing or invalid start");
|
||||
} else if (!end || !end.isValid) {
|
||||
return Interval.invalid("missing or invalid end");
|
||||
} else if (end < start) {
|
||||
return Interval.invalid(
|
||||
"end before start",
|
||||
`The end of an interval must be after its start, but you had start=${start.toISO()} and end=${end.toISO()}`
|
||||
);
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* An Interval object represents a half-open interval of time, where each endpoint is a {@link DateTime}. Conceptually, it's a container for those two endpoints, accompanied by methods for creating, parsing, interrogating, comparing, transforming, and formatting them.
|
||||
*
|
||||
* Here is a brief overview of the most commonly used methods and getters in Interval:
|
||||
*
|
||||
* * **Creation** To create an Interval, use {@link Interval.fromDateTimes}, {@link Interval.after}, {@link Interval.before}, or {@link Interval.fromISO}.
|
||||
* * **Accessors** Use {@link Interval#start} and {@link Interval#end} to get the start and end.
|
||||
* * **Interrogation** To analyze the Interval, use {@link Interval#count}, {@link Interval#length}, {@link Interval#hasSame}, {@link Interval#contains}, {@link Interval#isAfter}, or {@link Interval#isBefore}.
|
||||
* * **Transformation** To create other Intervals out of this one, use {@link Interval#set}, {@link Interval#splitAt}, {@link Interval#splitBy}, {@link Interval#divideEqually}, {@link Interval.merge}, {@link Interval.xor}, {@link Interval#union}, {@link Interval#intersection}, or {@link Interval#difference}.
|
||||
* * **Comparison** To compare this Interval to another one, use {@link Interval#equals}, {@link Interval#overlaps}, {@link Interval#abutsStart}, {@link Interval#abutsEnd}, {@link Interval#engulfs}
|
||||
* * **Output** To convert the Interval into other representations, see {@link Interval#toString}, {@link Interval#toLocaleString}, {@link Interval#toISO}, {@link Interval#toISODate}, {@link Interval#toISOTime}, {@link Interval#toFormat}, and {@link Interval#toDuration}.
|
||||
*/
|
||||
export default class Interval {
|
||||
/**
|
||||
* @private
|
||||
*/
|
||||
constructor(config) {
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.s = config.start;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.e = config.end;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.invalid = config.invalid || null;
|
||||
/**
|
||||
* @access private
|
||||
*/
|
||||
this.isLuxonInterval = true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an invalid Interval.
|
||||
* @param {string} reason - simple string of why this Interval is invalid. Should not contain parameters or anything else data-dependent
|
||||
* @param {string} [explanation=null] - longer explanation, may include parameters and other useful debugging information
|
||||
* @return {Interval}
|
||||
*/
|
||||
static invalid(reason, explanation = null) {
|
||||
if (!reason) {
|
||||
throw new InvalidArgumentError("need to specify a reason the Interval is invalid");
|
||||
}
|
||||
|
||||
const invalid = reason instanceof Invalid ? reason : new Invalid(reason, explanation);
|
||||
|
||||
if (Settings.throwOnInvalid) {
|
||||
throw new InvalidIntervalError(invalid);
|
||||
} else {
|
||||
return new Interval({ invalid });
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an Interval from a start DateTime and an end DateTime. Inclusive of the start but not the end.
|
||||
* @param {DateTime|Date|Object} start
|
||||
* @param {DateTime|Date|Object} end
|
||||
* @return {Interval}
|
||||
*/
|
||||
static fromDateTimes(start, end) {
|
||||
const builtStart = friendlyDateTime(start),
|
||||
builtEnd = friendlyDateTime(end);
|
||||
|
||||
const validateError = validateStartEnd(builtStart, builtEnd);
|
||||
|
||||
if (validateError == null) {
|
||||
return new Interval({
|
||||
start: builtStart,
|
||||
end: builtEnd,
|
||||
});
|
||||
} else {
|
||||
return validateError;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an Interval from a start DateTime and a Duration to extend to.
|
||||
* @param {DateTime|Date|Object} start
|
||||
* @param {Duration|Object|number} duration - the length of the Interval.
|
||||
* @return {Interval}
|
||||
*/
|
||||
static after(start, duration) {
|
||||
const dur = Duration.fromDurationLike(duration),
|
||||
dt = friendlyDateTime(start);
|
||||
return Interval.fromDateTimes(dt, dt.plus(dur));
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an Interval from an end DateTime and a Duration to extend backwards to.
|
||||
* @param {DateTime|Date|Object} end
|
||||
* @param {Duration|Object|number} duration - the length of the Interval.
|
||||
* @return {Interval}
|
||||
*/
|
||||
static before(end, duration) {
|
||||
const dur = Duration.fromDurationLike(duration),
|
||||
dt = friendlyDateTime(end);
|
||||
return Interval.fromDateTimes(dt.minus(dur), dt);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an Interval from an ISO 8601 string.
|
||||
* Accepts `<start>/<end>`, `<start>/<duration>`, and `<duration>/<end>` formats.
|
||||
* @param {string} text - the ISO string to parse
|
||||
* @param {Object} [opts] - options to pass {@link DateTime#fromISO} and optionally {@link Duration#fromISO}
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals
|
||||
* @return {Interval}
|
||||
*/
|
||||
static fromISO(text, opts) {
|
||||
const [s, e] = (text || "").split("/", 2);
|
||||
if (s && e) {
|
||||
let start, startIsValid;
|
||||
try {
|
||||
start = DateTime.fromISO(s, opts);
|
||||
startIsValid = start.isValid;
|
||||
} catch (e) {
|
||||
startIsValid = false;
|
||||
}
|
||||
|
||||
let end, endIsValid;
|
||||
try {
|
||||
end = DateTime.fromISO(e, opts);
|
||||
endIsValid = end.isValid;
|
||||
} catch (e) {
|
||||
endIsValid = false;
|
||||
}
|
||||
|
||||
if (startIsValid && endIsValid) {
|
||||
return Interval.fromDateTimes(start, end);
|
||||
}
|
||||
|
||||
if (startIsValid) {
|
||||
const dur = Duration.fromISO(e, opts);
|
||||
if (dur.isValid) {
|
||||
return Interval.after(start, dur);
|
||||
}
|
||||
} else if (endIsValid) {
|
||||
const dur = Duration.fromISO(s, opts);
|
||||
if (dur.isValid) {
|
||||
return Interval.before(end, dur);
|
||||
}
|
||||
}
|
||||
}
|
||||
return Interval.invalid("unparsable", `the input "${text}" can't be parsed as ISO 8601`);
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if an object is an Interval. Works across context boundaries
|
||||
* @param {object} o
|
||||
* @return {boolean}
|
||||
*/
|
||||
static isInterval(o) {
|
||||
return (o && o.isLuxonInterval) || false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the start of the Interval
|
||||
* @type {DateTime}
|
||||
*/
|
||||
get start() {
|
||||
return this.isValid ? this.s : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the end of the Interval
|
||||
* @type {DateTime}
|
||||
*/
|
||||
get end() {
|
||||
return this.isValid ? this.e : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether this Interval's end is at least its start, meaning that the Interval isn't 'backwards'.
|
||||
* @type {boolean}
|
||||
*/
|
||||
get isValid() {
|
||||
return this.invalidReason === null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an error code if this Interval is invalid, or null if the Interval is valid
|
||||
* @type {string}
|
||||
*/
|
||||
get invalidReason() {
|
||||
return this.invalid ? this.invalid.reason : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an explanation of why this Interval became invalid, or null if the Interval is valid
|
||||
* @type {string}
|
||||
*/
|
||||
get invalidExplanation() {
|
||||
return this.invalid ? this.invalid.explanation : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the length of the Interval in the specified unit.
|
||||
* @param {string} unit - the unit (such as 'hours' or 'days') to return the length in.
|
||||
* @return {number}
|
||||
*/
|
||||
length(unit = "milliseconds") {
|
||||
return this.isValid ? this.toDuration(...[unit]).get(unit) : NaN;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the count of minutes, hours, days, months, or years included in the Interval, even in part.
|
||||
* Unlike {@link Interval#length} this counts sections of the calendar, not periods of time, e.g. specifying 'day'
|
||||
* asks 'what dates are included in this interval?', not 'how many days long is this interval?'
|
||||
* @param {string} [unit='milliseconds'] - the unit of time to count.
|
||||
* @param {Object} opts - options
|
||||
* @param {boolean} [opts.useLocaleWeeks=false] - If true, use weeks based on the locale, i.e. use the locale-dependent start of the week; this operation will always use the locale of the start DateTime
|
||||
* @return {number}
|
||||
*/
|
||||
count(unit = "milliseconds", opts) {
|
||||
if (!this.isValid) return NaN;
|
||||
const start = this.start.startOf(unit, opts);
|
||||
let end;
|
||||
if (opts?.useLocaleWeeks) {
|
||||
end = this.end.reconfigure({ locale: start.locale });
|
||||
} else {
|
||||
end = this.end;
|
||||
}
|
||||
end = end.startOf(unit, opts);
|
||||
return Math.floor(end.diff(start, unit).get(unit)) + (end.valueOf() !== this.end.valueOf());
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether this Interval's start and end are both in the same unit of time
|
||||
* @param {string} unit - the unit of time to check sameness on
|
||||
* @return {boolean}
|
||||
*/
|
||||
hasSame(unit) {
|
||||
return this.isValid ? this.isEmpty() || this.e.minus(1).hasSame(this.s, unit) : false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval has the same start and end DateTimes.
|
||||
* @return {boolean}
|
||||
*/
|
||||
isEmpty() {
|
||||
return this.s.valueOf() === this.e.valueOf();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval's start is after the specified DateTime.
|
||||
* @param {DateTime} dateTime
|
||||
* @return {boolean}
|
||||
*/
|
||||
isAfter(dateTime) {
|
||||
if (!this.isValid) return false;
|
||||
return this.s > dateTime;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval's end is before the specified DateTime.
|
||||
* @param {DateTime} dateTime
|
||||
* @return {boolean}
|
||||
*/
|
||||
isBefore(dateTime) {
|
||||
if (!this.isValid) return false;
|
||||
return this.e <= dateTime;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval contains the specified DateTime.
|
||||
* @param {DateTime} dateTime
|
||||
* @return {boolean}
|
||||
*/
|
||||
contains(dateTime) {
|
||||
if (!this.isValid) return false;
|
||||
return this.s <= dateTime && this.e > dateTime;
|
||||
}
|
||||
|
||||
/**
|
||||
* "Sets" the start and/or end dates. Returns a newly-constructed Interval.
|
||||
* @param {Object} values - the values to set
|
||||
* @param {DateTime} values.start - the starting DateTime
|
||||
* @param {DateTime} values.end - the ending DateTime
|
||||
* @return {Interval}
|
||||
*/
|
||||
set({ start, end } = {}) {
|
||||
if (!this.isValid) return this;
|
||||
return Interval.fromDateTimes(start || this.s, end || this.e);
|
||||
}
|
||||
|
||||
/**
|
||||
* Split this Interval at each of the specified DateTimes
|
||||
* @param {...DateTime} dateTimes - the unit of time to count.
|
||||
* @return {Array}
|
||||
*/
|
||||
splitAt(...dateTimes) {
|
||||
if (!this.isValid) return [];
|
||||
const sorted = dateTimes
|
||||
.map(friendlyDateTime)
|
||||
.filter((d) => this.contains(d))
|
||||
.sort((a, b) => a.toMillis() - b.toMillis()),
|
||||
results = [];
|
||||
let { s } = this,
|
||||
i = 0;
|
||||
|
||||
while (s < this.e) {
|
||||
const added = sorted[i] || this.e,
|
||||
next = +added > +this.e ? this.e : added;
|
||||
results.push(Interval.fromDateTimes(s, next));
|
||||
s = next;
|
||||
i += 1;
|
||||
}
|
||||
|
||||
return results;
|
||||
}
|
||||
|
||||
/**
|
||||
* Split this Interval into smaller Intervals, each of the specified length.
|
||||
* Left over time is grouped into a smaller interval
|
||||
* @param {Duration|Object|number} duration - The length of each resulting interval.
|
||||
* @return {Array}
|
||||
*/
|
||||
splitBy(duration) {
|
||||
const dur = Duration.fromDurationLike(duration);
|
||||
|
||||
if (!this.isValid || !dur.isValid || dur.as("milliseconds") === 0) {
|
||||
return [];
|
||||
}
|
||||
|
||||
let { s } = this,
|
||||
idx = 1,
|
||||
next;
|
||||
|
||||
const results = [];
|
||||
while (s < this.e) {
|
||||
const added = this.start.plus(dur.mapUnits((x) => x * idx));
|
||||
next = +added > +this.e ? this.e : added;
|
||||
results.push(Interval.fromDateTimes(s, next));
|
||||
s = next;
|
||||
idx += 1;
|
||||
}
|
||||
|
||||
return results;
|
||||
}
|
||||
|
||||
/**
|
||||
* Split this Interval into the specified number of smaller intervals.
|
||||
* @param {number} numberOfParts - The number of Intervals to divide the Interval into.
|
||||
* @return {Array}
|
||||
*/
|
||||
divideEqually(numberOfParts) {
|
||||
if (!this.isValid) return [];
|
||||
return this.splitBy(this.length() / numberOfParts).slice(0, numberOfParts);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval overlaps with the specified Interval
|
||||
* @param {Interval} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
overlaps(other) {
|
||||
return this.e > other.s && this.s < other.e;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval's end is adjacent to the specified Interval's start.
|
||||
* @param {Interval} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
abutsStart(other) {
|
||||
if (!this.isValid) return false;
|
||||
return +this.e === +other.s;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval's start is adjacent to the specified Interval's end.
|
||||
* @param {Interval} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
abutsEnd(other) {
|
||||
if (!this.isValid) return false;
|
||||
return +other.e === +this.s;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval engulfs the start and end of the specified Interval.
|
||||
* @param {Interval} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
engulfs(other) {
|
||||
if (!this.isValid) return false;
|
||||
return this.s <= other.s && this.e >= other.e;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Interval has the same start and end as the specified Interval.
|
||||
* @param {Interval} other
|
||||
* @return {boolean}
|
||||
*/
|
||||
equals(other) {
|
||||
if (!this.isValid || !other.isValid) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return this.s.equals(other.s) && this.e.equals(other.e);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an Interval representing the intersection of this Interval and the specified Interval.
|
||||
* Specifically, the resulting Interval has the maximum start time and the minimum end time of the two Intervals.
|
||||
* Returns null if the intersection is empty, meaning, the intervals don't intersect.
|
||||
* @param {Interval} other
|
||||
* @return {Interval}
|
||||
*/
|
||||
intersection(other) {
|
||||
if (!this.isValid) return this;
|
||||
const s = this.s > other.s ? this.s : other.s,
|
||||
e = this.e < other.e ? this.e : other.e;
|
||||
|
||||
if (s >= e) {
|
||||
return null;
|
||||
} else {
|
||||
return Interval.fromDateTimes(s, e);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an Interval representing the union of this Interval and the specified Interval.
|
||||
* Specifically, the resulting Interval has the minimum start time and the maximum end time of the two Intervals.
|
||||
* @param {Interval} other
|
||||
* @return {Interval}
|
||||
*/
|
||||
union(other) {
|
||||
if (!this.isValid) return this;
|
||||
const s = this.s < other.s ? this.s : other.s,
|
||||
e = this.e > other.e ? this.e : other.e;
|
||||
return Interval.fromDateTimes(s, e);
|
||||
}
|
||||
|
||||
/**
|
||||
* Merge an array of Intervals into a equivalent minimal set of Intervals.
|
||||
* Combines overlapping and adjacent Intervals.
|
||||
* @param {Array} intervals
|
||||
* @return {Array}
|
||||
*/
|
||||
static merge(intervals) {
|
||||
const [found, final] = intervals
|
||||
.sort((a, b) => a.s - b.s)
|
||||
.reduce(
|
||||
([sofar, current], item) => {
|
||||
if (!current) {
|
||||
return [sofar, item];
|
||||
} else if (current.overlaps(item) || current.abutsStart(item)) {
|
||||
return [sofar, current.union(item)];
|
||||
} else {
|
||||
return [sofar.concat([current]), item];
|
||||
}
|
||||
},
|
||||
[[], null]
|
||||
);
|
||||
if (final) {
|
||||
found.push(final);
|
||||
}
|
||||
return found;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an array of Intervals representing the spans of time that only appear in one of the specified Intervals.
|
||||
* @param {Array} intervals
|
||||
* @return {Array}
|
||||
*/
|
||||
static xor(intervals) {
|
||||
let start = null,
|
||||
currentCount = 0;
|
||||
const results = [],
|
||||
ends = intervals.map((i) => [
|
||||
{ time: i.s, type: "s" },
|
||||
{ time: i.e, type: "e" },
|
||||
]),
|
||||
flattened = Array.prototype.concat(...ends),
|
||||
arr = flattened.sort((a, b) => a.time - b.time);
|
||||
|
||||
for (const i of arr) {
|
||||
currentCount += i.type === "s" ? 1 : -1;
|
||||
|
||||
if (currentCount === 1) {
|
||||
start = i.time;
|
||||
} else {
|
||||
if (start && +start !== +i.time) {
|
||||
results.push(Interval.fromDateTimes(start, i.time));
|
||||
}
|
||||
|
||||
start = null;
|
||||
}
|
||||
}
|
||||
|
||||
return Interval.merge(results);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an Interval representing the span of time in this Interval that doesn't overlap with any of the specified Intervals.
|
||||
* @param {...Interval} intervals
|
||||
* @return {Array}
|
||||
*/
|
||||
difference(...intervals) {
|
||||
return Interval.xor([this].concat(intervals))
|
||||
.map((i) => this.intersection(i))
|
||||
.filter((i) => i && !i.isEmpty());
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of this Interval appropriate for debugging.
|
||||
* @return {string}
|
||||
*/
|
||||
toString() {
|
||||
if (!this.isValid) return INVALID;
|
||||
return `[${this.s.toISO()} – ${this.e.toISO()})`;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of this Interval appropriate for the REPL.
|
||||
* @return {string}
|
||||
*/
|
||||
[Symbol.for("nodejs.util.inspect.custom")]() {
|
||||
if (this.isValid) {
|
||||
return `Interval { start: ${this.s.toISO()}, end: ${this.e.toISO()} }`;
|
||||
} else {
|
||||
return `Interval { Invalid, reason: ${this.invalidReason} }`;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a localized string representing this Interval. Accepts the same options as the
|
||||
* Intl.DateTimeFormat constructor and any presets defined by Luxon, such as
|
||||
* {@link DateTime.DATE_FULL} or {@link DateTime.TIME_SIMPLE}. The exact behavior of this method
|
||||
* is browser-specific, but in general it will return an appropriate representation of the
|
||||
* Interval in the assigned locale. Defaults to the system's locale if no locale has been
|
||||
* specified.
|
||||
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat
|
||||
* @param {Object} [formatOpts=DateTime.DATE_SHORT] - Either a DateTime preset or
|
||||
* Intl.DateTimeFormat constructor options.
|
||||
* @param {Object} opts - Options to override the configuration of the start DateTime.
|
||||
* @example Interval.fromISO('2022-11-07T09:00Z/2022-11-08T09:00Z').toLocaleString(); //=> 11/7/2022 – 11/8/2022
|
||||
* @example Interval.fromISO('2022-11-07T09:00Z/2022-11-08T09:00Z').toLocaleString(DateTime.DATE_FULL); //=> November 7 – 8, 2022
|
||||
* @example Interval.fromISO('2022-11-07T09:00Z/2022-11-08T09:00Z').toLocaleString(DateTime.DATE_FULL, { locale: 'fr-FR' }); //=> 7–8 novembre 2022
|
||||
* @example Interval.fromISO('2022-11-07T17:00Z/2022-11-07T19:00Z').toLocaleString(DateTime.TIME_SIMPLE); //=> 6:00 – 8:00 PM
|
||||
* @example Interval.fromISO('2022-11-07T17:00Z/2022-11-07T19:00Z').toLocaleString({ weekday: 'short', month: 'short', day: '2-digit', hour: '2-digit', minute: '2-digit' }); //=> Mon, Nov 07, 6:00 – 8:00 p
|
||||
* @return {string}
|
||||
*/
|
||||
toLocaleString(formatOpts = Formats.DATE_SHORT, opts = {}) {
|
||||
return this.isValid
|
||||
? Formatter.create(this.s.loc.clone(opts), formatOpts).formatInterval(this)
|
||||
: INVALID;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601-compliant string representation of this Interval.
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals
|
||||
* @param {Object} opts - The same options as {@link DateTime#toISO}
|
||||
* @return {string}
|
||||
*/
|
||||
toISO(opts) {
|
||||
if (!this.isValid) return INVALID;
|
||||
return `${this.s.toISO(opts)}/${this.e.toISO(opts)}`;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601-compliant string representation of date of this Interval.
|
||||
* The time components are ignored.
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals
|
||||
* @return {string}
|
||||
*/
|
||||
toISODate() {
|
||||
if (!this.isValid) return INVALID;
|
||||
return `${this.s.toISODate()}/${this.e.toISODate()}`;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an ISO 8601-compliant string representation of time of this Interval.
|
||||
* The date components are ignored.
|
||||
* @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals
|
||||
* @param {Object} opts - The same options as {@link DateTime#toISO}
|
||||
* @return {string}
|
||||
*/
|
||||
toISOTime(opts) {
|
||||
if (!this.isValid) return INVALID;
|
||||
return `${this.s.toISOTime(opts)}/${this.e.toISOTime(opts)}`;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a string representation of this Interval formatted according to the specified format
|
||||
* string. **You may not want this.** See {@link Interval#toLocaleString} for a more flexible
|
||||
* formatting tool.
|
||||
* @param {string} dateFormat - The format string. This string formats the start and end time.
|
||||
* See {@link DateTime#toFormat} for details.
|
||||
* @param {Object} opts - Options.
|
||||
* @param {string} [opts.separator = ' – '] - A separator to place between the start and end
|
||||
* representations.
|
||||
* @return {string}
|
||||
*/
|
||||
toFormat(dateFormat, { separator = " – " } = {}) {
|
||||
if (!this.isValid) return INVALID;
|
||||
return `${this.s.toFormat(dateFormat)}${separator}${this.e.toFormat(dateFormat)}`;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a Duration representing the time spanned by this interval.
|
||||
* @param {string|string[]} [unit=['milliseconds']] - the unit or units (such as 'hours' or 'days') to include in the duration.
|
||||
* @param {Object} opts - options that affect the creation of the Duration
|
||||
* @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use
|
||||
* @example Interval.fromDateTimes(dt1, dt2).toDuration().toObject() //=> { milliseconds: 88489257 }
|
||||
* @example Interval.fromDateTimes(dt1, dt2).toDuration('days').toObject() //=> { days: 1.0241812152777778 }
|
||||
* @example Interval.fromDateTimes(dt1, dt2).toDuration(['hours', 'minutes']).toObject() //=> { hours: 24, minutes: 34.82095 }
|
||||
* @example Interval.fromDateTimes(dt1, dt2).toDuration(['hours', 'minutes', 'seconds']).toObject() //=> { hours: 24, minutes: 34, seconds: 49.257 }
|
||||
* @example Interval.fromDateTimes(dt1, dt2).toDuration('seconds').toObject() //=> { seconds: 88489.257 }
|
||||
* @return {Duration}
|
||||
*/
|
||||
toDuration(unit, opts) {
|
||||
if (!this.isValid) {
|
||||
return Duration.invalid(this.invalidReason);
|
||||
}
|
||||
return this.e.diff(this.s, unit, opts);
|
||||
}
|
||||
|
||||
/**
|
||||
* Run mapFn on the interval start and end, returning a new Interval from the resulting DateTimes
|
||||
* @param {function} mapFn
|
||||
* @return {Interval}
|
||||
* @example Interval.fromDateTimes(dt1, dt2).mapEndpoints(endpoint => endpoint.toUTC())
|
||||
* @example Interval.fromDateTimes(dt1, dt2).mapEndpoints(endpoint => endpoint.plus({ hours: 2 }))
|
||||
*/
|
||||
mapEndpoints(mapFn) {
|
||||
return Interval.fromDateTimes(mapFn(this.s), mapFn(this.e));
|
||||
}
|
||||
}
|
26
priv/vendor/luxon/src/luxon.js
vendored
26
priv/vendor/luxon/src/luxon.js
vendored
|
@ -1,26 +0,0 @@
|
|||
import DateTime from "./datetime.js";
|
||||
import Duration from "./duration.js";
|
||||
import Interval from "./interval.js";
|
||||
import Info from "./info.js";
|
||||
import Zone from "./zone.js";
|
||||
import FixedOffsetZone from "./zones/fixedOffsetZone.js";
|
||||
import IANAZone from "./zones/IANAZone.js";
|
||||
import InvalidZone from "./zones/invalidZone.js";
|
||||
import SystemZone from "./zones/systemZone.js";
|
||||
import Settings from "./settings.js";
|
||||
|
||||
const VERSION = "3.4.4";
|
||||
|
||||
export {
|
||||
VERSION,
|
||||
DateTime,
|
||||
Duration,
|
||||
Interval,
|
||||
Info,
|
||||
Zone,
|
||||
FixedOffsetZone,
|
||||
IANAZone,
|
||||
InvalidZone,
|
||||
SystemZone,
|
||||
Settings,
|
||||
};
|
4
priv/vendor/luxon/src/package.json
vendored
4
priv/vendor/luxon/src/package.json
vendored
|
@ -1,4 +0,0 @@
|
|||
{
|
||||
"type": "module",
|
||||
"version": "3.4.4"
|
||||
}
|
175
priv/vendor/luxon/src/settings.js
vendored
175
priv/vendor/luxon/src/settings.js
vendored
|
@ -1,175 +0,0 @@
|
|||
import SystemZone from "./zones/systemZone.js";
|
||||
import IANAZone from "./zones/IANAZone.js";
|
||||
import Locale from "./impl/locale.js";
|
||||
|
||||
import { normalizeZone } from "./impl/zoneUtil.js";
|
||||
import { validateWeekSettings } from "./impl/util.js";
|
||||
|
||||
let now = () => Date.now(),
|
||||
defaultZone = "system",
|
||||
defaultLocale = null,
|
||||
defaultNumberingSystem = null,
|
||||
defaultOutputCalendar = null,
|
||||
twoDigitCutoffYear = 60,
|
||||
throwOnInvalid,
|
||||
defaultWeekSettings = null;
|
||||
|
||||
/**
|
||||
* Settings contains static getters and setters that control Luxon's overall behavior. Luxon is a simple library with few options, but the ones it does have live here.
|
||||
*/
|
||||
export default class Settings {
|
||||
/**
|
||||
* Get the callback for returning the current timestamp.
|
||||
* @type {function}
|
||||
*/
|
||||
static get now() {
|
||||
return now;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the callback for returning the current timestamp.
|
||||
* The function should return a number, which will be interpreted as an Epoch millisecond count
|
||||
* @type {function}
|
||||
* @example Settings.now = () => Date.now() + 3000 // pretend it is 3 seconds in the future
|
||||
* @example Settings.now = () => 0 // always pretend it's Jan 1, 1970 at midnight in UTC time
|
||||
*/
|
||||
static set now(n) {
|
||||
now = n;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the default time zone to create DateTimes in. Does not affect existing instances.
|
||||
* Use the value "system" to reset this value to the system's time zone.
|
||||
* @type {string}
|
||||
*/
|
||||
static set defaultZone(zone) {
|
||||
defaultZone = zone;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the default time zone object currently used to create DateTimes. Does not affect existing instances.
|
||||
* The default value is the system's time zone (the one set on the machine that runs this code).
|
||||
* @type {Zone}
|
||||
*/
|
||||
static get defaultZone() {
|
||||
return normalizeZone(defaultZone, SystemZone.instance);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the default locale to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static get defaultLocale() {
|
||||
return defaultLocale;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the default locale to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static set defaultLocale(locale) {
|
||||
defaultLocale = locale;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the default numbering system to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static get defaultNumberingSystem() {
|
||||
return defaultNumberingSystem;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the default numbering system to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static set defaultNumberingSystem(numberingSystem) {
|
||||
defaultNumberingSystem = numberingSystem;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the default output calendar to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static get defaultOutputCalendar() {
|
||||
return defaultOutputCalendar;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the default output calendar to create DateTimes with. Does not affect existing instances.
|
||||
* @type {string}
|
||||
*/
|
||||
static set defaultOutputCalendar(outputCalendar) {
|
||||
defaultOutputCalendar = outputCalendar;
|
||||
}
|
||||
|
||||
/**
|
||||
* @typedef {Object} WeekSettings
|
||||
* @property {number} firstDay
|
||||
* @property {number} minimalDays
|
||||
* @property {number[]} weekend
|
||||
*/
|
||||
|
||||
/**
|
||||
* @return {WeekSettings|null}
|
||||
*/
|
||||
static get defaultWeekSettings() {
|
||||
return defaultWeekSettings;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allows overriding the default locale week settings, i.e. the start of the week, the weekend and
|
||||
* how many days are required in the first week of a year.
|
||||
* Does not affect existing instances.
|
||||
*
|
||||
* @param {WeekSettings|null} weekSettings
|
||||
*/
|
||||
static set defaultWeekSettings(weekSettings) {
|
||||
defaultWeekSettings = validateWeekSettings(weekSettings);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the cutoff year after which a string encoding a year as two digits is interpreted to occur in the current century.
|
||||
* @type {number}
|
||||
*/
|
||||
static get twoDigitCutoffYear() {
|
||||
return twoDigitCutoffYear;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the cutoff year after which a string encoding a year as two digits is interpreted to occur in the current century.
|
||||
* @type {number}
|
||||
* @example Settings.twoDigitCutoffYear = 0 // cut-off year is 0, so all 'yy' are interpreted as current century
|
||||
* @example Settings.twoDigitCutoffYear = 50 // '49' -> 1949; '50' -> 2050
|
||||
* @example Settings.twoDigitCutoffYear = 1950 // interpreted as 50
|
||||
* @example Settings.twoDigitCutoffYear = 2050 // ALSO interpreted as 50
|
||||
*/
|
||||
static set twoDigitCutoffYear(cutoffYear) {
|
||||
twoDigitCutoffYear = cutoffYear % 100;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get whether Luxon will throw when it encounters invalid DateTimes, Durations, or Intervals
|
||||
* @type {boolean}
|
||||
*/
|
||||
static get throwOnInvalid() {
|
||||
return throwOnInvalid;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set whether Luxon will throw when it encounters invalid DateTimes, Durations, or Intervals
|
||||
* @type {boolean}
|
||||
*/
|
||||
static set throwOnInvalid(t) {
|
||||
throwOnInvalid = t;
|
||||
}
|
||||
|
||||
/**
|
||||
* Reset Luxon's global caches. Should only be necessary in testing scenarios.
|
||||
* @return {void}
|
||||
*/
|
||||
static resetCaches() {
|
||||
Locale.resetCache();
|
||||
IANAZone.resetCache();
|
||||
}
|
||||
}
|
91
priv/vendor/luxon/src/zone.js
vendored
91
priv/vendor/luxon/src/zone.js
vendored
|
@ -1,91 +0,0 @@
|
|||
import { ZoneIsAbstractError } from "./errors.js";
|
||||
|
||||
/**
|
||||
* @interface
|
||||
*/
|
||||
export default class Zone {
|
||||
/**
|
||||
* The type of zone
|
||||
* @abstract
|
||||
* @type {string}
|
||||
*/
|
||||
get type() {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* The name of this zone.
|
||||
* @abstract
|
||||
* @type {string}
|
||||
*/
|
||||
get name() {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
get ianaName() {
|
||||
return this.name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether the offset is known to be fixed for the whole year.
|
||||
* @abstract
|
||||
* @type {boolean}
|
||||
*/
|
||||
get isUniversal() {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the offset's common name (such as EST) at the specified timestamp
|
||||
* @abstract
|
||||
* @param {number} ts - Epoch milliseconds for which to get the name
|
||||
* @param {Object} opts - Options to affect the format
|
||||
* @param {string} opts.format - What style of offset to return. Accepts 'long' or 'short'.
|
||||
* @param {string} opts.locale - What locale to return the offset name in.
|
||||
* @return {string}
|
||||
*/
|
||||
offsetName(ts, opts) {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the offset's value as a string
|
||||
* @abstract
|
||||
* @param {number} ts - Epoch milliseconds for which to get the offset
|
||||
* @param {string} format - What style of offset to return.
|
||||
* Accepts 'narrow', 'short', or 'techie'. Returning '+6', '+06:00', or '+0600' respectively
|
||||
* @return {string}
|
||||
*/
|
||||
formatOffset(ts, format) {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the offset in minutes for this zone at the specified timestamp.
|
||||
* @abstract
|
||||
* @param {number} ts - Epoch milliseconds for which to compute the offset
|
||||
* @return {number}
|
||||
*/
|
||||
offset(ts) {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Zone is equal to another zone
|
||||
* @abstract
|
||||
* @param {Zone} otherZone - the zone to compare
|
||||
* @return {boolean}
|
||||
*/
|
||||
equals(otherZone) {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return whether this Zone is valid.
|
||||
* @abstract
|
||||
* @type {boolean}
|
||||
*/
|
||||
get isValid() {
|
||||
throw new ZoneIsAbstractError();
|
||||
}
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue