Compare commits
24 Commits
0.1.0
...
acces-libr
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
46fda975d4 | ||
|
|
670ab495d8 | ||
|
|
34bffe93af | ||
|
|
8ecb93656c | ||
|
|
eac737b889 | ||
|
|
dfeb1be0f0 | ||
|
|
0eefafdeb2 | ||
|
|
ce70d498ca | ||
|
|
991f6794c7 | ||
|
|
2b8a57b14a | ||
|
|
117f069351 | ||
|
|
981eba100d | ||
|
|
cf557c89cd | ||
|
|
2cc0b40b90 | ||
|
|
3dc6a41724 | ||
|
|
abaf506d56 | ||
|
|
866d89eb09 | ||
|
|
7df902df52 | ||
|
|
80b16f260e | ||
|
|
1735e468aa | ||
|
|
38b3acfb89 | ||
|
|
3d8c22598d | ||
|
|
b3ac0960ae | ||
| 4ced780d54 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -2,6 +2,7 @@
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
CACHEDIR.TAG
|
||||
|
||||
# C extensions
|
||||
*.so
|
||||
|
||||
@@ -20,7 +20,7 @@ To make this project work, you will need :
|
||||
- Some knowledge of the command line,
|
||||
- Some knowledge of Python.
|
||||
- 3h of your time, 5h if things need debugging.
|
||||
- `git`, `virtualenv`,`pip` and `python` >= 3.8.6.
|
||||
- `fswebcam`, `libjpeg-dev` ,`zlib1g-dev`,`libffi-dev`,`git`, `virtualenv`,`pip` and `python` >= 3.8.6.
|
||||
- A webcam for the webcam page to work. Will work on a smartphone. Not required.
|
||||
|
||||
## Context
|
||||
|
||||
@@ -6,6 +6,13 @@ vendor_id = 0x04b8
|
||||
device_id = 0x0e28
|
||||
upload_folder = "src/static/uploads"
|
||||
|
||||
# Raspberry Pi Configuration
|
||||
[rpi]
|
||||
button_gpio_port_number = 2
|
||||
indicator_gpio_port_number = 22
|
||||
flash_gpio_port_number = 16
|
||||
flash = true
|
||||
|
||||
# Users = Password
|
||||
[users]
|
||||
admin = "admin"
|
||||
|
||||
@@ -5,6 +5,12 @@
|
||||
vendor_id = "0x04b8"
|
||||
device_id = "0x0e28"
|
||||
|
||||
# Raspberry Pi Configuration
|
||||
[rpi]
|
||||
button_gpio_port_number = 17
|
||||
indicator_gpio_port_number = 18
|
||||
flash = true
|
||||
|
||||
# Users = Password
|
||||
[users]
|
||||
admin = "admin"
|
||||
|
||||
5
docs/Thermal Receipt/CMakeLists.txt
Normal file
5
docs/Thermal Receipt/CMakeLists.txt
Normal file
@@ -0,0 +1,5 @@
|
||||
#cmake_minimum_required(VERSION 2.8)
|
||||
add_executable(rastertotmtr
|
||||
filter/TmThermalReceipt.c
|
||||
)
|
||||
target_link_libraries(rastertotmtr cupsimage cups)
|
||||
171
docs/Thermal Receipt/EULA
Normal file
171
docs/Thermal Receipt/EULA
Normal file
@@ -0,0 +1,171 @@
|
||||
SEIKO EPSON CORPORATION
|
||||
SOFTWARE LICENSE AGREEMENT
|
||||
|
||||
IMPORTANT! READ THIS SOFTWARE LICENSE AGREEMENT CAREFULLY. The
|
||||
computer software product, fontware, typefaces and/or data, including
|
||||
any accompanying explanatory written materials (the "Software") should
|
||||
only be installed or used by the Licensee ("you") on the condition you
|
||||
agree with SEIKO EPSON CORPORATION ("EPSON") to the terms and
|
||||
conditions set forth in this Agreement. By installing or using the
|
||||
Software, you are representing to agree all the terms and conditions
|
||||
set forth in this Agreement. You should read this Agreement carefully
|
||||
before installing or using the Software. If you do not agree with the
|
||||
terms and conditions of this Agreement, you are not permitted to
|
||||
install or use the Software.
|
||||
|
||||
1. License. EPSON and its suppliers grant you a personal,
|
||||
nonexclusive, royalty-free, non-sublicensable limited license to
|
||||
install and use the Software solely for the purpose of using EPSON
|
||||
printer products ("Purpose") on any single computer or computers
|
||||
that you intend to use directly or via network. You may allow other
|
||||
users of the computers connected to the network to use the Software,
|
||||
provided that you (a) ensure that all such users agree and are bound
|
||||
by the terms and conditions of this Agreement, (b) ensure that all
|
||||
such users use the Software only in conjunction with the computers and
|
||||
in relation to the network of which they form part, and (c) indemnify
|
||||
and keep whole EPSON and its suppliers against all damages, losses,
|
||||
costs, expenses and liabilities which EPSON or its suppliers may incur
|
||||
as a consequence of such users failing to observe and perform the
|
||||
terms and conditions of this Agreement. You may also make copies of
|
||||
the Software as necessary for backup and archival purposes, provided
|
||||
that the copyright notice is reproduced in its entirety on the backup
|
||||
copy. The term "Software" shall include the software components,
|
||||
media, all copies made by you and any upgrades, modified versions,
|
||||
updates, additions and copies of the Software licensed to you by EPSON
|
||||
or its suppliers. EPSON and its suppliers reserve all rights not
|
||||
granted herein.
|
||||
|
||||
2. Other Rights and Limitations. You agree not to modify, adapt or
|
||||
translate the Software. You also agree not to attempt to reverse
|
||||
engineer, decompile, disassemble or otherwise attempt to discover the
|
||||
source code of the Software. You may not use the Software for any
|
||||
purposes other than the Purpose. You may not share, rent, lease,
|
||||
encumber, sublicense or lend the Software. You may, however, transfer
|
||||
all your rights to use the Software to another person or legal entity
|
||||
provided that you transfer this Agreement, the Software, including all
|
||||
copies, updates and prior versions, to such person or entity, and that
|
||||
you retain no copies, including copies stored on a computer. Some
|
||||
states or jurisdictions, however, do not allow the restriction or
|
||||
limitation on transfer of the Software, so the above limitations may
|
||||
not apply to you.
|
||||
|
||||
3. Ownership. Title, ownership rights, and intellectual property
|
||||
rights in and to the Software and any copies thereof shall remain with
|
||||
EPSON or its suppliers. There is no transfer to you of any title to
|
||||
or ownership of the Software and this License shall not be construed
|
||||
as a sale of any rights in the Software. The Software is protected by
|
||||
Japanese Copyright Law and international copyright treaties, as well
|
||||
as other intellectual property laws and treaties. Except as otherwise
|
||||
provided in this Agreement, you may not copy the Software. You also
|
||||
agree not to remove or alter any copyright and other proprietary
|
||||
notices on any copies of the Software.
|
||||
|
||||
4. LGPL. The Software uses the open source software programs which
|
||||
apply the GNU Lesser General Public License Version 2 or later version
|
||||
("LGPL"). Notwithstanding any provision of this Agreement, you may
|
||||
make modification of the Software for your own use and reverse
|
||||
engineering for debugging such modifications according to the terms
|
||||
and conditions of the LGPL.
|
||||
|
||||
5. Protection and Security. You agree to use your best efforts and
|
||||
take all reasonable steps to safeguard the Software to ensure that no
|
||||
unauthorized person has access to them and that no unauthorized copy,
|
||||
publication, disclosure or distribution of any of the Software is
|
||||
made. You acknowledge that the Software contains valuable,
|
||||
confidential information and trade secrets, that unauthorized use and
|
||||
copying are harmful to EPSON and its suppliers, and that you have a
|
||||
confidentiality obligation as to such valuable information and trade
|
||||
secrets.
|
||||
|
||||
6. Limited Warranty. In case of that you obtained the Software by
|
||||
media from EPSON or a dealer, EPSON warrants that the media on which
|
||||
the Software is recorded will be free from defects in workmanship and
|
||||
materials under normal use for a period of 90 days from the date of
|
||||
delivery to you. If the media is returned to EPSON or the dealer from
|
||||
which the media was obtained within 90 days of the date of delivery to
|
||||
you, and if EPSON determines the media to be defective and provided
|
||||
the media was not subject to misuse, abuse, misapplication or use in
|
||||
defective equipment, EPSON will replace the media, upon your return to
|
||||
EPSON of the Software, including all copies of any portions thereof.
|
||||
ALL IMPLIED WARRANTIES ON THE MEDIA, INCLUDING IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, ARE LIMITED TO
|
||||
THE DURATION OF THE EXPRESS WARRANTY SET FORTH ABOVE.
|
||||
|
||||
You acknowledge and agree that the use of the Software is at
|
||||
your sole risk. THE SOFTWARE IS PROVIDED "AS IS" AND WITHOUT
|
||||
ANY WARRANTY OF ANY KIND. EPSON AND ITS SUPPLIERS DO NOT AND
|
||||
CANNOT WARRANT THE PERFORMANCE OR RESULTS YOU MAY OBTAIN BY
|
||||
USING THE SOFTWARE. THE FOREGOING STATES THE SOLE AND
|
||||
EXCLUSIVE REMEDIES FOR EPSON'S AND ITS SUPPLIERS' BREACH OF
|
||||
WARRANTY. EXCEPT FOR THE FOREGOING LIMITED WARRANTY, EPSON
|
||||
AND ITS SUPPLIERS MAKE NO WARRANTIES, EXPRESS OR IMPLIED, AS
|
||||
TO NON-INFRINGEMENT, MERCHANTABILITY OR FITNESS FOR ANY
|
||||
PARTICULAR PURPOSE. Some states or jurisdictions do not allow
|
||||
the exclusion of implied warranties or limitations on how long
|
||||
an implied warranty may last, so the above limitations may not
|
||||
apply to you. This warranty gives you specific legal rights.
|
||||
You may have other rights which vary from state to state or
|
||||
jurisdiction to jurisdiction.
|
||||
|
||||
IN NO EVENT WILL EPSON OR ITS SUPPLIERS BE LIABLE TO YOU,
|
||||
WHETHER ARISING UNDER CONTRACT, TORT (INCLUDING NEGLIGENCE),
|
||||
STRICT LIABILITY, BREACH OR WARRANTY, MISREPRESENTATION OR
|
||||
OTHERWISE, FOR ANY DIRECT, CONSEQUENTIAL, INCIDENTAL OR
|
||||
SPECIAL DAMAGES, INCLUDING ANY LOST PROFITS OR LOST SAVINGS,
|
||||
EVEN IF EPSON, ITS SUPPLIERS OR ANY REPRESENTATIVE HAS BEEN
|
||||
ADVISED OF THE POSSIBILITY OF SUCH DAMAGES, OR FOR ANY CLAIM
|
||||
BY ANY THIRD PARTY. Some states or jurisdictions, however, do
|
||||
not allow the exclusion or limitation of incidental,
|
||||
consequential or special damages, so the above limitations may
|
||||
not apply to you.
|
||||
|
||||
7. Termination. Without prejudice to any other rights EPSON has, this
|
||||
Agreement shall automatically terminate upon failure by you to comply
|
||||
with its terms. You may also terminate this Agreement at any time by
|
||||
uninstalling and destroying the Software and all copies thereof.
|
||||
|
||||
8. Export Restriction. You agree not to transfer, export or re-export
|
||||
the Software and any data or information which you obtained from EPSON
|
||||
or use the Software without a proper license under Japanese law,
|
||||
restrictions and regulations, or the laws of the jurisdiction in which
|
||||
the Software is obtained.
|
||||
|
||||
9. Governing Law and General Provisions. This Agreement shall be
|
||||
governed and construed under by the laws of Japan without regard to
|
||||
its conflicts of law rules. This Agreement is the entire agreement
|
||||
between the parties with respect to the Software, and supersedes any
|
||||
purchase order, communication, advertisement, or representation
|
||||
concerning the Software. This Agreement shall be binding upon, and
|
||||
inure to the benefit of, the parties hereto and their respective
|
||||
successors, assigns and legal representatives. If any provision
|
||||
herein is found void or unenforceable, it will not affect the validity
|
||||
of the balance of the Agreement, which shall remain valid and
|
||||
enforceable according to its terms. This Agreement may only be
|
||||
modified in writing signed by an authorized officer of EPSON.
|
||||
|
||||
10. U.S. Government End Users. If you are acquiring the Software on
|
||||
behalf of any unit or agency of the United States Government, the
|
||||
following provisions apply. The Government agrees: (i) if the
|
||||
Software is supplied to the Department of Defense (DoD), the Software
|
||||
is classified as "Commercial Computer Software" and the Government is
|
||||
acquiring only "restricted rights" in the Software and its
|
||||
documentation as that term is defined in Clause 252.227-7013(c)(1) of
|
||||
the DFARS; and (ii) if the Software is supplied to any unit or agency
|
||||
of the United States Government other than DoD, the Government's
|
||||
rights in the Software and its documentation will be as defined in
|
||||
Clause 52.227-19(c)(2) of the FAR or, in the case of NASA, in Clause
|
||||
18-52.227-86(d) of the NASA supplement to the FAR.
|
||||
|
||||
11. Internet Connection. The Software may have the ability to connect
|
||||
over the Internet to transmit data and/or information to and from your
|
||||
computer regarding the EPSON hardware and/or software that you use
|
||||
("EPSON Products") including, but not limited to, EPSON Products model
|
||||
information, the country/region where you live, the condition of EPSON
|
||||
Products, etc. EPSON may alter the items of such data and/or
|
||||
information without your prior approval. EPSON does not collect any
|
||||
personally identifiable information without your permission. EPSON
|
||||
may, however, use non personally identifiable information for
|
||||
statistical purposes to improve the level of service we provide to our
|
||||
users If you agree to install the Software, any transmissions to or
|
||||
from the Internet will be in accordance with EPSON's then-current
|
||||
Privacy Policy provided in EPSON Internet site.
|
||||
340
docs/Thermal Receipt/LISENSE
Normal file
340
docs/Thermal Receipt/LISENSE
Normal file
@@ -0,0 +1,340 @@
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Lesser General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
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
|
||||
this service 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.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute 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 and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
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
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the 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 a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, 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.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE 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.
|
||||
|
||||
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
|
||||
convey 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 General Public License as published by
|
||||
the Free Software Foundation; either version 2 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 General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License along
|
||||
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License.
|
||||
96
docs/Thermal Receipt/README
Normal file
96
docs/Thermal Receipt/README
Normal file
@@ -0,0 +1,96 @@
|
||||
|
||||
EPSON TM Series Printer Driver for Linux Version 3.0.0
|
||||
|
||||
Copyright (C) Seiko Epson Corporation 2019.
|
||||
|
||||
|
||||
1. GENERAL
|
||||
----------
|
||||
This software is a printer driver for printing on an Epson
|
||||
TM series printer from Linux using CUPS.
|
||||
|
||||
1.1) Features
|
||||
+ A raster type printer driver for TM series printers.
|
||||
+ Can instantly print out images, texts and drawings displayed by
|
||||
an application.
|
||||
+ Various printer controls on paper cut timing, cash drawers,
|
||||
printing speed, blank line saving, inverted printing, etc.
|
||||
|
||||
2. ENVIRONMENT
|
||||
--------------
|
||||
2.1) Supported printers
|
||||
+ EPSON TM-m30
|
||||
+ EPSON TM-T88VI
|
||||
+ EPSON TM-H6000V
|
||||
|
||||
2.2) Confirmed distributions
|
||||
+ Ubuntu 18.04
|
||||
+ CentOS 7 1810
|
||||
+ openSUSE 13.1
|
||||
|
||||
3. FILES
|
||||
--------
|
||||
+ README .......... This file
|
||||
+ EULA ............ EPSON SOFTWARE LICENSE AGREEMENT
|
||||
+ LISENSE ......... GNU GENERAL PUBLIC LICENSE
|
||||
+ build.sh ........ Build script
|
||||
+ install.sh ...... Installation script
|
||||
+ CMakeList.txt ... input file of cmake
|
||||
+ /filter ......... source code of filter driver
|
||||
+ /ppd ............ ppd files
|
||||
|
||||
4. HOW TO BUILD & INSTALL
|
||||
-------------------------
|
||||
Ensure that you have following packages pre-installed
|
||||
+ Ubuntu ..... cmake, libcupsimage2-dev development
|
||||
+ CentOS ..... cmake, gcc, gcc-c++, cups-devel development
|
||||
+ openSUSE ... cmake, gcc, gcc-c++, cups-devel development
|
||||
|
||||
4.1) Execute build script
|
||||
#sudo ./build.sh
|
||||
|
||||
*Temporary folder build will be made when run script.
|
||||
|
||||
!!! You must the following install script with root privileges. !!!
|
||||
|
||||
4.2) Execute installation script
|
||||
#sudo ./install.sh
|
||||
|
||||
*Filter driver and ppd files will be copied to appropriate folders.
|
||||
|
||||
5. HOW TO PRINT
|
||||
---------------
|
||||
5.1 By command line
|
||||
1) Turn on printer
|
||||
Turn the printer on before registering a printer.
|
||||
2) Register a printer
|
||||
!!! You must run lpadmin command with root privilege. !!!
|
||||
#lpadmin -p <destination> -v <device-uri> -P <ppd-file> -E
|
||||
-p <destination>
|
||||
-v <device-uri>
|
||||
-P <ppd-file>
|
||||
-E
|
||||
|
||||
example)
|
||||
#lpadmin -p TM-m10 -v usb://EPSON/TM-m10 -P tm-ba-thermal-rastertotmtr.ppd -E
|
||||
|
||||
3) Print a file
|
||||
!!! You must always specify media options !!!
|
||||
$lpr -o <option> -P <printer> <file>
|
||||
example)
|
||||
$lpr -o media=RP80x2000 -P TM-m10 sample.pdf
|
||||
|
||||
5.2 by GUI
|
||||
Add a queue using OS tool & test print by GUI
|
||||
http://localhost:631 or http://127.0.0.1:631
|
||||
|
||||
6. LIMITATIONS
|
||||
--------------
|
||||
+ Support USB printer class only.
|
||||
Please configure printer class support manually.
|
||||
|
||||
7. HISTORY
|
||||
----------
|
||||
+ 2019/2/15 Version 3.0.0
|
||||
|
||||
[EOF]
|
||||
11
docs/Thermal Receipt/build.sh
Executable file
11
docs/Thermal Receipt/build.sh
Executable file
@@ -0,0 +1,11 @@
|
||||
#!/bin/sh
|
||||
|
||||
#build in directory
|
||||
if [ -d build ]
|
||||
then
|
||||
rm -R build
|
||||
fi
|
||||
mkdir build
|
||||
cd build
|
||||
cmake ..
|
||||
make
|
||||
1069
docs/Thermal Receipt/filter/TmThermalReceipt.c
Normal file
1069
docs/Thermal Receipt/filter/TmThermalReceipt.c
Normal file
File diff suppressed because it is too large
Load Diff
123
docs/Thermal Receipt/install.sh
Executable file
123
docs/Thermal Receipt/install.sh
Executable file
@@ -0,0 +1,123 @@
|
||||
#!/bin/sh
|
||||
|
||||
echo "EPSON TM series CUPS driver installer"
|
||||
echo "---------------------------------------"
|
||||
echo ""
|
||||
echo ""
|
||||
|
||||
ROOT_UID=0
|
||||
|
||||
if [ 0 -ne `id -u` ]
|
||||
then
|
||||
echo "This script requires root user access."
|
||||
echo "Re-run as root user."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
SERVERROOT=$(grep '^ServerRoot' /etc/cups/cupsd.conf | awk '{print $2}')
|
||||
|
||||
if [ -z $FILTERDIR ] || [ -z $PPDDIR ]
|
||||
then
|
||||
echo "Searching for ServerRoot, ServerBin, and DataDir tags in /etc/cups/cupsd.conf"
|
||||
echo ""
|
||||
|
||||
if [ -z $FILTERDIR ]
|
||||
then
|
||||
SERVERBIN=$(grep '^ServerBin' /etc/cups/cupsd.conf | awk '{print $2}')
|
||||
|
||||
if [ -z $SERVERBIN ]
|
||||
then
|
||||
echo "ServerBin tag not present in cupsd.conf - using default"
|
||||
FILTERDIR=/usr/lib/cups/filter
|
||||
elif [ ${SERVERBIN:0:1} = "/" ]
|
||||
then
|
||||
echo "ServerBin tag is present as an absolute path"
|
||||
FILTERDIR=$SERVERBIN/filter
|
||||
else
|
||||
echo "ServerBin tag is present as a relative path - appending to ServerRoot"
|
||||
FILTERDIR=$SERVERROOT/$SERVERBIN/filter
|
||||
fi
|
||||
fi
|
||||
|
||||
echo ""
|
||||
|
||||
if [ -z $PPDDIR ]
|
||||
then
|
||||
DATADIR=$(grep '^DataDir' /etc/cups/cupsd.conf | awk '{print $2}')
|
||||
|
||||
if [ -z $DATADIR ]
|
||||
then
|
||||
echo "DataDir tag not present in cupsd.conf - using default"
|
||||
PPDDIR=/usr/share/cups/model/EPSON
|
||||
elif [ ${DATADIR:0:1} = "/" ]
|
||||
then
|
||||
echo "DataDir tag is present as an absolute path"
|
||||
PPDDIR=$DATADIR/model/EPSON
|
||||
else
|
||||
echo "DataDir tag is present as a relative path - appending to ServerRoot"
|
||||
PPDDIR=$SERVERROOT/$DATADIR/model/EPSON
|
||||
fi
|
||||
fi
|
||||
|
||||
echo "SERVERBIN = $SERVERBIN"
|
||||
echo "FILTERDIR = $FILTERDIR"
|
||||
echo "PPDDIR = $PPDDIR"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
INSTALL=/usr/bin/install
|
||||
|
||||
echo "Installing filter driver ..."
|
||||
$INSTALL -s ./build/rastertotmtr $FILTERDIR
|
||||
echo ""
|
||||
|
||||
echo "Installing ppd files ..."
|
||||
$INSTALL -m 755 -d $PPDDIR
|
||||
$INSTALL -m 755 ./ppd/*.ppd $PPDDIR
|
||||
echo ""
|
||||
|
||||
if [ -z $RPMBUILD ]
|
||||
then
|
||||
echo "Restarting CUPS"
|
||||
if [ -x /etc/software/init.d/cups ]
|
||||
then
|
||||
/etc/software/init.d/cups stop
|
||||
/etc/software/init.d/cups start
|
||||
elif [ -x /etc/rc.d/init.d/cups ]
|
||||
then
|
||||
/etc/rc.d/init.d/cups stop
|
||||
/etc/rc.d/init.d/cups start
|
||||
elif [ -x /etc/init.d/cups ]
|
||||
then
|
||||
/etc/init.d/cups stop
|
||||
/etc/init.d/cups start
|
||||
elif [ -x /sbin/init.d/cups ]
|
||||
then
|
||||
/sbin/init.d/cups stop
|
||||
/sbin/init.d/cups start
|
||||
elif [ -x /etc/software/init.d/cupsys ]
|
||||
then
|
||||
/etc/software/init.d/cupsys stop
|
||||
/etc/software/init.d/cupsys start
|
||||
elif [ -x /etc/rc.d/init.d/cupsys ]
|
||||
then
|
||||
/etc/rc.d/init.d/cupsys stop
|
||||
/etc/rc.d/init.d/cupsys start
|
||||
elif [ -x /etc/init.d/cupsys ]
|
||||
then
|
||||
/etc/init.d/cupsys stop
|
||||
/etc/init.d/cupsys start
|
||||
elif [ -x /sbin/init.d/cupsys ]
|
||||
then
|
||||
/sbin/init.d/cupsys stop
|
||||
/sbin/init.d/cupsys start
|
||||
else
|
||||
echo "Could not restart CUPS"
|
||||
fi
|
||||
echo ""
|
||||
fi
|
||||
|
||||
echo "Installation Completed"
|
||||
echo "Add a printer queue using OS tool, http://localhost:631, or http://127.0.0.1:631"
|
||||
echo ""
|
||||
|
||||
120
docs/Thermal Receipt/ppd/tm-ba-thermal-rastertotmtr-180.ppd
Normal file
120
docs/Thermal Receipt/ppd/tm-ba-thermal-rastertotmtr-180.ppd
Normal file
@@ -0,0 +1,120 @@
|
||||
*PPD-Adobe: "4.3"
|
||||
*% Copyright (C) Seiko Epson Corporation 2018. All rights reserved.
|
||||
*%
|
||||
*% PPD file for TM Series Linux CUPS Printer Driver.
|
||||
|
||||
*FormatVersion: "4.3"
|
||||
*FileVersion: "2.0"
|
||||
*LanguageVersion: English
|
||||
*LanguageEncoding: ISOLatin1
|
||||
*PCFileName: "EPTMBATH.PPD"
|
||||
*Manufacturer: "EPSON"
|
||||
*Product: "(ThermalPrinter)"
|
||||
*ModelName:"EPSON TM Thermal"
|
||||
*ShortNickName:"TM Thermal"
|
||||
*NickName: "EPSON TM Thermal (180dpi)"
|
||||
*PSVersion: "(3010.000) 0"
|
||||
*LanguageLevel: "3"
|
||||
*ColorDevice: False
|
||||
*DefaultColorSpace: Gray
|
||||
*FileSystem: False
|
||||
*Throughput: "1"
|
||||
*LandscapeOrientation: Plus90
|
||||
*VariablePaperSize: True
|
||||
*TTRasterizer: Type42
|
||||
*cupsVersion: 1.2
|
||||
*cupsManualCopies: True
|
||||
*cupsModelNumber: 100
|
||||
*cupsFilter: "application/vnd.cups-raster 0 rastertotmtr"
|
||||
*cupsLanguages: "en"
|
||||
|
||||
|
||||
*% Printer option settings
|
||||
*OpenGroup: General/General
|
||||
|
||||
*% Page size settings.
|
||||
*OpenUI *PageSize/Media Size: PickOne
|
||||
*OrderDependency: 10 AnySetup *PageSize
|
||||
*DefaultPageSize:RP80x200
|
||||
*PageSize RP80x200/Roll paper 80 x 200 mm: "<</PageSize[205.0 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP80x2000/Roll paper 80 x 2000 mm: "<</PageSize[205.0 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP58x200/Roll paper 58 x 200 mm: "<</PageSize[144.0 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP58x2000/Roll paper 58 x 2000 mm: "<</PageSize[144.0 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*CloseUI: *PageSize
|
||||
|
||||
*OpenUI *PageRegion: PickOne
|
||||
*OrderDependency: 10 AnySetup *PageRegion
|
||||
*DefaultPageRegion:RP80x200
|
||||
*PageRegion RP80x200/Roll paper 80 x 200 mm: "<</PageSize[205.0 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP80x2000/Roll paper 80 x 2000 mm: "<</PageSize[205.0 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP58x200/Roll paper 58 x 200 mm: "<</PageSize[144.0 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP58x2000/Roll paper 58 x 2000 mm: "<</PageSize[144.0 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*CloseUI: *PageRegion
|
||||
|
||||
*DefaultImageableArea: RP80x200
|
||||
*ImageableArea RP80x200/Roll paper 80 x 200 mm: "0.0 0.0 205.0 566.9"
|
||||
*ImageableArea RP80x2000/Roll paper 80 x 2000 mm: "0.0 0.0 205.0 5669.3"
|
||||
*ImageableArea RP58x200/Roll paper 58 x 200 mm: "0.0 0.0 144.0 566.9"
|
||||
*ImageableArea RP58x2000/Roll paper 58 x 2000 mm: "0.0 0.0 144.0 5669.3"
|
||||
|
||||
*DefaultPaperDimension: RP80x200
|
||||
*PaperDimension RP80x200/Roll paper 80 x 200 mm: "205.0 566.9"
|
||||
*PaperDimension RP80x2000/Roll paper 80 x 2000 mm: "205.0 5669.3"
|
||||
*PaperDimension RP58x200/Roll paper 58 x 200 mm: "144.0 566.9"
|
||||
*PaperDimension RP58x2000/Roll paper 58 x 2000 mm: "144.0 5669.3"
|
||||
|
||||
*% Custom page size settings.
|
||||
*MaxMediaWidth: "205.0"
|
||||
*MaxMediaHeight: "5669.3"
|
||||
*NonUIOrderDependency: 100 AnySetup *CustomPageSize
|
||||
*CustomPageSize True: "pop pop pop <</PageSize [ 5 -2 roll ]/ImagingBBox null>>setpagedevice"
|
||||
*ParamCustomPageSize Width: 1 points 72 205.0
|
||||
*ParamCustomPageSize Height: 2 points 72 5669.3
|
||||
*ParamCustomPageSize WidthOffset: 3 points 0 0
|
||||
*ParamCustomPageSize HeightOffset: 4 points 0 0
|
||||
*ParamCustomPageSize Orientation: 5 int 0 0
|
||||
|
||||
*% Resolution settings.
|
||||
*OpenUI *Resolution/Resolution: PickOne
|
||||
*OrderDependency: 20 AnySetup *Resolution
|
||||
*DefaultResolution: 180x180dpi
|
||||
*Resolution 180x180dpi/180 x 180 dpi: "<</HWResolution[180 180]/cupsRowCount 24/cupsBitsPerColor 1>>setpagedevice"
|
||||
*CloseUI: *Resolution
|
||||
|
||||
*% Horizontal and Vertical motion units.
|
||||
*TmxMotionUnitHori: "180"
|
||||
*TmxMotionUnitVert: "180"
|
||||
|
||||
*% Paper reduction settings.
|
||||
*OpenUI *TmxPaperReduction/Paper Reduction: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxPaperReduction
|
||||
*DefaultTmxPaperReduction: Bottom
|
||||
*TmxPaperReduction Off/None: ""
|
||||
*TmxPaperReduction Top/Top: ""
|
||||
*TmxPaperReduction Bottom/Bottom: ""
|
||||
*TmxPaperReduction Both/Top & Bottom: ""
|
||||
*CloseUI: *TmxPaperReduction
|
||||
|
||||
*% Buzzer / Cash Drawer settings.
|
||||
*OpenUI *TmxBuzzerAndDrawer/Buzzer/ Cash Drawer: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxBuzzerAndDrawer
|
||||
*DefaultTmxBuzzerAndDrawer: NotUsed
|
||||
*TmxBuzzerAndDrawer NotUsed/Not used: ""
|
||||
*TmxBuzzerAndDrawer InternalBuzzer/Internal buzzer: ""
|
||||
*TmxBuzzerAndDrawer ExternalBuzzer/External buzzer: ""
|
||||
*TmxBuzzerAndDrawer OpenDrawer1/Open drawer #1: ""
|
||||
*TmxBuzzerAndDrawer OpenDrawer2/Open drawer #2: ""
|
||||
*CloseUI: *TmxBuzzerAndDrawer
|
||||
|
||||
*% Paper source settings.
|
||||
*OpenUI *TmxPaperCut/Paper Cut: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxPaperCut
|
||||
*DefaultTmxPaperCut: NoCut
|
||||
*TmxPaperCut NoCut/No cut: ""
|
||||
*TmxPaperCut CutPerJob/Cut per job: ""
|
||||
*TmxPaperCut CutPerPage/Cut per page: ""
|
||||
*CloseUI: *TmxPaperCut
|
||||
|
||||
*CloseGroup: General
|
||||
|
||||
*% End
|
||||
120
docs/Thermal Receipt/ppd/tm-ba-thermal-rastertotmtr-203.ppd
Normal file
120
docs/Thermal Receipt/ppd/tm-ba-thermal-rastertotmtr-203.ppd
Normal file
@@ -0,0 +1,120 @@
|
||||
*PPD-Adobe: "4.3"
|
||||
*% Copyright (C) Seiko Epson Corporation 2018. All rights reserved.
|
||||
*%
|
||||
*% PPD file for TM Series Linux CUPS Printer Driver.
|
||||
|
||||
*FormatVersion: "4.3"
|
||||
*FileVersion: "2.0"
|
||||
*LanguageVersion: English
|
||||
*LanguageEncoding: ISOLatin1
|
||||
*PCFileName: "EPTMBATH.PPD"
|
||||
*Manufacturer: "EPSON"
|
||||
*Product: "(ThermalPrinter)"
|
||||
*ModelName:"EPSON TM Thermal"
|
||||
*ShortNickName:"TM Thermal"
|
||||
*NickName: "EPSON TM Thermal (203dpi)"
|
||||
*PSVersion: "(3010.000) 0"
|
||||
*LanguageLevel: "3"
|
||||
*ColorDevice: False
|
||||
*DefaultColorSpace: Gray
|
||||
*FileSystem: False
|
||||
*Throughput: "1"
|
||||
*LandscapeOrientation: Plus90
|
||||
*VariablePaperSize: True
|
||||
*TTRasterizer: Type42
|
||||
*cupsVersion: 1.2
|
||||
*cupsManualCopies: True
|
||||
*cupsModelNumber: 100
|
||||
*cupsFilter: "application/vnd.cups-raster 0 rastertotmtr"
|
||||
*cupsLanguages: "en"
|
||||
|
||||
|
||||
*% Printer option settings
|
||||
*OpenGroup: General/General
|
||||
|
||||
*% Page size settings.
|
||||
*OpenUI *PageSize/Media Size: PickOne
|
||||
*OrderDependency: 10 AnySetup *PageSize
|
||||
*DefaultPageSize:RP80x200
|
||||
*PageSize RP80x200/Roll paper 80 x 200 mm: "<</PageSize[204.3 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP80x2000/Roll paper 80 x 2000 mm: "<</PageSize[204.3 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP58x200/Roll paper 58 x 200 mm: "<</PageSize[149.1 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageSize RP58x2000/Roll paper 58 x 2000 mm: "<</PageSize[149.1 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*CloseUI: *PageSize
|
||||
|
||||
*OpenUI *PageRegion: PickOne
|
||||
*OrderDependency: 10 AnySetup *PageRegion
|
||||
*DefaultPageRegion:RP80x200
|
||||
*PageRegion RP80x200/Roll paper 80 x 200 mm: "<</PageSize[204.3 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP80x2000/Roll paper 80 x 2000 mm: "<</PageSize[204.3 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP58x200/Roll paper 58 x 200 mm: "<</PageSize[149.1 566.9]/ImagingBBox null>>setpagedevice"
|
||||
*PageRegion RP58x2000/Roll paper 58 x 2000 mm: "<</PageSize[149.1 5669.3]/ImagingBBox null>>setpagedevice"
|
||||
*CloseUI: *PageRegion
|
||||
|
||||
*DefaultImageableArea: RP80x200
|
||||
*ImageableArea RP80x200/Roll paper 80 x 200 mm: "0.0 0.0 204.3 566.9"
|
||||
*ImageableArea RP80x2000/Roll paper 80 x 2000 mm: "0.0 0.0 204.3 5669.3"
|
||||
*ImageableArea RP58x200/Roll paper 58 x 200 mm: "0.0 0.0 149.1 566.9"
|
||||
*ImageableArea RP58x2000/Roll paper 58 x 2000 mm: "0.0 0.0 149.1 5669.3"
|
||||
|
||||
*DefaultPaperDimension: RP80x200
|
||||
*PaperDimension RP80x200/Roll paper 80 x 200 mm: "204.3 566.9"
|
||||
*PaperDimension RP80x2000/Roll paper 80 x 2000 mm: "204.3 5669.3"
|
||||
*PaperDimension RP58x200/Roll paper 58 x 200 mm: "149.1 566.9"
|
||||
*PaperDimension RP58x2000/Roll paper 58 x 2000 mm: "149.1 5669.3"
|
||||
|
||||
*% Custom page size settings.
|
||||
*MaxMediaWidth: "204.3"
|
||||
*MaxMediaHeight: "5669.3"
|
||||
*NonUIOrderDependency: 100 AnySetup *CustomPageSize
|
||||
*CustomPageSize True: "pop pop pop <</PageSize [ 5 -2 roll ]/ImagingBBox null>>setpagedevice"
|
||||
*ParamCustomPageSize Width: 1 points 72 204.3
|
||||
*ParamCustomPageSize Height: 2 points 72 5669.3
|
||||
*ParamCustomPageSize WidthOffset: 3 points 0 0
|
||||
*ParamCustomPageSize HeightOffset: 4 points 0 0
|
||||
*ParamCustomPageSize Orientation: 5 int 0 0
|
||||
|
||||
*% Resolution settings.
|
||||
*OpenUI *Resolution/Resolution: PickOne
|
||||
*OrderDependency: 20 AnySetup *Resolution
|
||||
*DefaultResolution: 203x203dpi
|
||||
*Resolution 203x203dpi/203 x 203 dpi: "<</HWResolution[203 203]/cupsRowCount 24/cupsBitsPerColor 1>>setpagedevice"
|
||||
*CloseUI: *Resolution
|
||||
|
||||
*% Horizontal and Vertical motion units.
|
||||
*TmxMotionUnitHori: "203"
|
||||
*TmxMotionUnitVert: "203"
|
||||
|
||||
*% Paper reduction settings.
|
||||
*OpenUI *TmxPaperReduction/Paper Reduction: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxPaperReduction
|
||||
*DefaultTmxPaperReduction: Bottom
|
||||
*TmxPaperReduction Off/None: ""
|
||||
*TmxPaperReduction Top/Top: ""
|
||||
*TmxPaperReduction Bottom/Bottom: ""
|
||||
*TmxPaperReduction Both/Top & Bottom: ""
|
||||
*CloseUI: *TmxPaperReduction
|
||||
|
||||
*% Buzzer / Cash Drawer settings.
|
||||
*OpenUI *TmxBuzzerAndDrawer/Buzzer/ Cash Drawer: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxBuzzerAndDrawer
|
||||
*DefaultTmxBuzzerAndDrawer: NotUsed
|
||||
*TmxBuzzerAndDrawer NotUsed/Not used: ""
|
||||
*TmxBuzzerAndDrawer InternalBuzzer/Internal buzzer: ""
|
||||
*TmxBuzzerAndDrawer ExternalBuzzer/External buzzer: ""
|
||||
*TmxBuzzerAndDrawer OpenDrawer1/Open drawer #1: ""
|
||||
*TmxBuzzerAndDrawer OpenDrawer2/Open drawer #2: ""
|
||||
*CloseUI: *TmxBuzzerAndDrawer
|
||||
|
||||
*% Paper source settings.
|
||||
*OpenUI *TmxPaperCut/Paper Cut: PickOne
|
||||
*OrderDependency: 30 AnySetup *TmxPaperCut
|
||||
*DefaultTmxPaperCut: NoCut
|
||||
*TmxPaperCut NoCut/No cut: ""
|
||||
*TmxPaperCut CutPerJob/Cut per job: ""
|
||||
*TmxPaperCut CutPerPage/Cut per page: ""
|
||||
*CloseUI: *TmxPaperCut
|
||||
|
||||
*CloseGroup: General
|
||||
|
||||
*% End
|
||||
12
littleprynter.service
Normal file
12
littleprynter.service
Normal file
@@ -0,0 +1,12 @@
|
||||
[Unit]
|
||||
Description=LittlePrynter
|
||||
After=network.target
|
||||
|
||||
[Service]
|
||||
Type=simple
|
||||
WorkingDirectory=/home/pi/littleprynter/
|
||||
Environment=FLASK_APP=src/main.py
|
||||
ExecStart=/home/pi/littleprynter/bin/flask run --host 0.0.0.0 --debug --no-reload
|
||||
|
||||
[Install]
|
||||
WantedBy=multi-user.target
|
||||
292
requirements.txt
292
requirements.txt
@@ -1,31 +1,261 @@
|
||||
Adafruit-Thermal~=1.1.0
|
||||
appdirs~=1.4.4
|
||||
argcomplete~=2.0.0
|
||||
click~=8.1.3
|
||||
commonmark~=0.9.1
|
||||
Deprecated~=1.2.13
|
||||
escpos~=1.9
|
||||
Flask~=2.1.2
|
||||
Flask-Limiter~=2.4.5.1
|
||||
future~=0.18.2
|
||||
itsdangerous~=2.1.2
|
||||
Jinja2~=3.1.2
|
||||
limits~=2.6.1
|
||||
MarkupSafe~=2.1.1
|
||||
packaging~=21.3
|
||||
Pillow~=9.1.0
|
||||
Pygments~=2.12.0
|
||||
pyparsing~=3.0.8
|
||||
pyserial~=3.5
|
||||
python-barcode~=0.13.1
|
||||
pyusb~=1.2.1
|
||||
PyYAML~=6.0
|
||||
qrcode~=7.3.1
|
||||
rich~=12.4.1
|
||||
six~=1.16.0
|
||||
toml~=0.10.2
|
||||
typing_extensions~=4.2.0
|
||||
Unidecode~=1.3.4
|
||||
viivakoodi~=0.8.0
|
||||
Werkzeug~=2.1.2
|
||||
wrapt~=1.14.1
|
||||
appdirs==1.4.4
|
||||
apt-listchanges==4.8
|
||||
argcomplete==3.6.2
|
||||
arrow==1.3.0
|
||||
attrs==25.3.0
|
||||
babel==2.17.0
|
||||
bcrypt==4.2.0
|
||||
bidict==0.23.1
|
||||
blinker==1.9.0
|
||||
certifi==2025.1.31
|
||||
cffi==1.17.1
|
||||
chardet==5.2.0
|
||||
charset-normalizer==3.4.2
|
||||
click==8.3.0
|
||||
cloud-init==25.2
|
||||
colorzero==2.0
|
||||
configobj==5.0.9
|
||||
cryptography==43.0.0
|
||||
Deprecated==1.2.18
|
||||
distlib==0.3.9
|
||||
distro==1.9.0
|
||||
escpos==2.0.0
|
||||
filelock==3.18.0
|
||||
Flask==3.1.2
|
||||
Flask-Limiter==4.0.0
|
||||
Flask-SocketIO==5.5.1
|
||||
fqdn==1.5.1
|
||||
future==1.0.0
|
||||
gpiod==2.2.0
|
||||
gpiozero==2.0.1
|
||||
h11==0.16.0
|
||||
idna==3.10
|
||||
isoduration==20.11.0
|
||||
itsdangerous==2.2.0
|
||||
Jinja2==3.1.6
|
||||
jsonpatch==1.32
|
||||
jsonpointer==2.4
|
||||
jsonschema==4.19.2
|
||||
jsonschema-specifications==2023.12.1
|
||||
lgpio==0.2.2.0
|
||||
limits==5.6.0
|
||||
linkify-it-py==2.0.3
|
||||
markdown-it-py==3.0.0
|
||||
MarkupSafe==2.1.5
|
||||
mdurl==0.1.2
|
||||
numpy==2.3.4
|
||||
oauthlib==3.2.2
|
||||
olefile==0.47
|
||||
ordered-set==4.1.0
|
||||
packaging==25.0
|
||||
pillow==11.1.0
|
||||
platformdirs==4.3.7
|
||||
ply==3.11
|
||||
pycparser==2.22
|
||||
pycryptodomex==3.20.0
|
||||
Pygments==2.18.0
|
||||
PyJWT==2.10.1
|
||||
pyserial==3.5
|
||||
python-apt==3.0.0
|
||||
python-barcode==0.16.1
|
||||
python-dateutil==2.9.0
|
||||
python-engineio==4.12.3
|
||||
python-socketio==5.14.1
|
||||
pyusb==1.3.1
|
||||
PyYAML==6.0.2
|
||||
qrcode==8.2
|
||||
referencing==0.36.2
|
||||
requests==2.32.3
|
||||
rfc3339-validator==0.1.4
|
||||
rfc3986-validator==0.1.1
|
||||
rfc3987==1.3.8
|
||||
rich==13.9.4
|
||||
rpds-py==0.21.0
|
||||
rpi-keyboard-config==1.0
|
||||
rpi-lgpio==0.6
|
||||
setuptools==80.9.0
|
||||
simple-websocket==1.1.0
|
||||
six==1.17.0
|
||||
smbus2==0.4.3
|
||||
spidev==3.6
|
||||
ssh-import-id==5.10
|
||||
toml==0.10.2
|
||||
types-aiofiles==24.1
|
||||
types-antlr4-python3-runtime==4.13
|
||||
types-assertpy==1.1
|
||||
types-atheris==2.3
|
||||
types-aws-xray-sdk==2.14
|
||||
types-beautifulsoup4==4.12
|
||||
types-bleach==6.2
|
||||
types-boltons==24.1
|
||||
types-braintree==4.31
|
||||
types-cachetools==5.5
|
||||
types-caldav==1.3
|
||||
types-capturer==3.0
|
||||
types-cffi==1.16
|
||||
types-chevron==0.14
|
||||
types-click-default-group==1.2
|
||||
types-click-spinner==0.1
|
||||
types-colorama==0.4
|
||||
types-commonmark==0.9
|
||||
types-console-menu==0.8
|
||||
types-corus==0.10
|
||||
types-croniter==5.0.1
|
||||
types-dateparser==1.2
|
||||
types-decorator==5.1
|
||||
types-defusedxml==0.7
|
||||
types-Deprecated==1.2.15
|
||||
types-docker==7.1
|
||||
types-dockerfile-parse==2.0
|
||||
types-docutils==0.21
|
||||
types-editdistance==0.8
|
||||
types-entrypoints==0.4
|
||||
types-ExifRead==3.0
|
||||
types-fanstatic==1.4
|
||||
types-first==2.0
|
||||
types-flake8==7.1
|
||||
types-flake8-bugbear==24.12.12
|
||||
types-flake8-builtins==2.5
|
||||
types-flake8-docstrings==1.7
|
||||
types-flake8-rst-docstrings==0.3
|
||||
types-flake8-simplify==0.21
|
||||
types-flake8-typing-imports==1.16
|
||||
types-Flask-Cors==5.0
|
||||
types-Flask-Migrate==4.0
|
||||
types-Flask-SocketIO==5.4
|
||||
types-fpdf2==2.8.2
|
||||
types-gdb==15.0
|
||||
types-gevent==24.11
|
||||
types-google-cloud-ndb==2.3
|
||||
types-greenlet==3.1
|
||||
types-hdbcli==2.23
|
||||
types-html5lib==1.1
|
||||
types-httplib2==0.22
|
||||
types-humanfriendly==10.0
|
||||
types-hvac==2.3
|
||||
types-ibm-db==3.2.4
|
||||
types-icalendar==6.1
|
||||
types-influxdb-client==1.45
|
||||
types-inifile==0.4
|
||||
types-JACK-Client==0.5
|
||||
types-Jetson.GPIO==2.1
|
||||
types-jmespath==1.0
|
||||
types-jsonschema==4.23
|
||||
types-jwcrypto==1.5
|
||||
types-keyboard==0.13
|
||||
types-ldap3==2.9
|
||||
types-libsass==0.23
|
||||
types-lupa==2.2
|
||||
types-lzstring==1.0
|
||||
types-m3u8==6.0
|
||||
types-Markdown==3.7
|
||||
types-mock==5.1
|
||||
types-mypy-extensions==1.0
|
||||
types-mysqlclient==2.2
|
||||
types-nanoid==2.0.0
|
||||
types-netaddr==1.3
|
||||
types-netifaces==0.11
|
||||
types-networkx==3.4.2
|
||||
types-oauthlib==3.2
|
||||
types-objgraph==3.6
|
||||
types-olefile==0.47
|
||||
types-openpyxl==3.1.5
|
||||
types-opentracing==2.4
|
||||
types-paramiko==3.5
|
||||
types-parsimonious==0.10
|
||||
types-passlib==1.7
|
||||
types-passpy==1.0
|
||||
types-peewee==3.17.8
|
||||
types-pep8-naming==0.14
|
||||
types-pexpect==4.9
|
||||
types-pika-ts==1.3
|
||||
types-polib==1.2
|
||||
types-portpicker==1.6
|
||||
types-protobuf==5.29.1
|
||||
types-psutil==6.1
|
||||
types-psycopg2==2.9.10
|
||||
types-pyasn1==0.6
|
||||
types-pyaudio==0.2
|
||||
types-PyAutoGUI==0.9
|
||||
types-pycocotools==2.0
|
||||
types-pycurl==7.45.4
|
||||
types-pyfarmhash==0.4
|
||||
types-pyflakes==3.2
|
||||
types-pygit2==1.15
|
||||
types-Pygments==2.18
|
||||
types-pyinstaller==6.11
|
||||
types-pyjks==20.0
|
||||
types-PyMySQL==1.1
|
||||
types-pynput==1.7.7
|
||||
types-pyOpenSSL==24.1
|
||||
types-pyRFC3339==2.0.1
|
||||
types-PyScreeze==1.0.1
|
||||
types-pyserial==3.5
|
||||
types-pysftp==0.2
|
||||
types-pytest-lazy-fixture==0.6
|
||||
types-python-crontab==3.2
|
||||
types-python-datemath==3.0.1
|
||||
types-python-dateutil==2.9
|
||||
types-python-http-client==3.3.7
|
||||
types-python-jenkins==1.8
|
||||
types-python-jose==3.3
|
||||
types-python-nmap==0.7
|
||||
types-python-xlib==0.33
|
||||
types-pytz==2024.2
|
||||
types-pywin32==308
|
||||
types-pyxdg==0.28
|
||||
types-PyYAML==6.0
|
||||
types-qrbill==1.1
|
||||
types-qrcode==8.0
|
||||
types-regex==2024.11.6
|
||||
types-reportlab==4.2.5
|
||||
types-requests==2.32
|
||||
types-requests-oauthlib==2.0
|
||||
types-retry==0.9
|
||||
types-RPi.GPIO==0.7
|
||||
types-s2clientprotocol==5
|
||||
types-seaborn==0.13.2
|
||||
types-Send2Trash==1.8
|
||||
types-setuptools==75.6
|
||||
types-shapely==2.0
|
||||
types-simplejson==3.19
|
||||
types-singledispatch==4.1
|
||||
types-six==1.17
|
||||
types-slumber==0.7
|
||||
types-str2bool==1.1
|
||||
types-tabulate==0.9
|
||||
types-tensorflow==2.18.0
|
||||
types-TgCrypto==1.2
|
||||
types-toml==0.10
|
||||
types-toposort==1.10
|
||||
types-tqdm==4.67
|
||||
types-translationstring==1.4
|
||||
types-tree-sitter-languages==1.10
|
||||
types-ttkthemes==3.2
|
||||
types-ujson==5.10
|
||||
types-unidiff==0.7
|
||||
types-untangle==1.2
|
||||
types-usersettings==1.1
|
||||
types-uWSGI==2.0
|
||||
types-vobject==0.9.9
|
||||
types-waitress==3.0.1
|
||||
types-WebOb==1.8
|
||||
types-whatthepatch==1.0
|
||||
types-workalendar==17.0
|
||||
types-WTForms==3.2.1
|
||||
types-wurlitzer==3.1
|
||||
types-xdgenvpy==3.0
|
||||
types-xmltodict==0.14
|
||||
types-zstd==1.5
|
||||
types-zxcvbn==4.4
|
||||
typing_extensions==4.15.0
|
||||
uc-micro-py==1.0.3
|
||||
uritemplate==4.1.1
|
||||
urllib3==2.3.0
|
||||
viivakoodi==0.8.0
|
||||
virtualenv==20.31.2
|
||||
webcolors==1.13
|
||||
Werkzeug==3.1.3
|
||||
wheel==0.46.1
|
||||
wrapt==1.17.3
|
||||
wsproto==1.2.0
|
||||
|
||||
126
src/main.py
126
src/main.py
@@ -6,11 +6,17 @@
|
||||
# Then we build the API around Flask,
|
||||
# Then we build the web interface, using the simple Jinja2 templating.
|
||||
|
||||
# We support two modes :
|
||||
# The first is a simple mode, where a computer, connected to a thermal printer, runs this program and exposes a web interface that makes use of the client's camera
|
||||
# The seconde is booth mode, where a Raspberry Pi is connected to a thermal printer, a button and a flash. The web interface exists but may not be used, as the press of the button with take a picture and activate the flash while simply informing the web page.
|
||||
|
||||
# Following are the librairies we import,
|
||||
from flask import Flask, request, render_template, flash, abort, redirect, url_for, make_response, jsonify # Used for the web framework
|
||||
from flask_socketio import SocketIO
|
||||
from flask_limiter import Limiter
|
||||
from flask_limiter.util import get_remote_address
|
||||
from printer import Printer # The wrapper for the printer class
|
||||
from raspberry import Raspberry # The Raspberry pi control Class
|
||||
from web import Web # Wrapper for the web routes and API
|
||||
import toml # Used for the config file parsing
|
||||
import pprint # To pretty print JSON
|
||||
@@ -19,6 +25,8 @@ import os # For VARS from the shell.
|
||||
# Variables
|
||||
|
||||
app = Flask(__name__)
|
||||
socketio = SocketIO(app)
|
||||
ALLOWED_EXTENSIONS = {'png', 'jpg', 'jpeg', 'gif'}
|
||||
|
||||
# Load the configuration file
|
||||
try:
|
||||
@@ -35,12 +43,22 @@ except Exception as e:
|
||||
exit(-1)
|
||||
|
||||
app.logger.debug("Config file loaded !")
|
||||
|
||||
# Define the USB connections here.
|
||||
vendor_id = configuration_file["printer"]["vendor_id"]
|
||||
device_id = configuration_file["printer"]["device_id"]
|
||||
UPLOAD_FOLDER = str(configuration_file["printer"]["upload_folder"])
|
||||
|
||||
ALLOWED_EXTENSIONS = {'png', 'jpg', 'jpeg', 'gif'}
|
||||
try:
|
||||
os.mkdir(UPLOAD_FOLDER)
|
||||
app.logger.debug(f"Directory '{UPLOAD_FOLDER}' created successfully.")
|
||||
except FileExistsError:
|
||||
app.logger.debug(f"Directory '{UPLOAD_FOLDER}' already exists.")
|
||||
except PermissionError:
|
||||
app.logger.error(f"Permission denied: Unable to create '{UPLOAD_FOLDER}'.")
|
||||
except Exception as e:
|
||||
app.logger.error(f"An error occurred: {e}")
|
||||
|
||||
|
||||
# Output the config file
|
||||
if os.getenv('LIPY_DEBUG') == True:
|
||||
@@ -50,7 +68,7 @@ if os.getenv('LIPY_DEBUG') == True:
|
||||
app.secret_key = configuration_file["secrets"]["flask_secret_key"]
|
||||
app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER
|
||||
app.config['ALLOWED_EXTENSIONS'] = ALLOWED_EXTENSIONS
|
||||
app.config['MAX_CONTENT_LENGTH'] = 3 * 1000 * 1000 # Maximum 3Mb for a file upload
|
||||
app.config['MAX_CONTENT_LENGTH'] = 10 * 1000 * 1000 # Maximum 3Mb for a file upload
|
||||
app.config['TEMPLATES_AUTO_RELOAD'] = True
|
||||
|
||||
# Printer connection
|
||||
@@ -58,15 +76,32 @@ app.config['TEMPLATES_AUTO_RELOAD'] = True
|
||||
printer = Printer(app,0x04b8, 0x0e28)
|
||||
printer.init_printer()
|
||||
|
||||
# Web routes
|
||||
# Find out if we are running on a Raspberry Pi
|
||||
rpi = Raspberry(printer,
|
||||
app,
|
||||
socketio,
|
||||
configuration_file['rpi']['button_gpio_port_number'], configuration_file['rpi']['indicator_gpio_port_number'],
|
||||
configuration_file['rpi']['flash_gpio_port_number'],
|
||||
configuration_file['rpi']['flash'] )
|
||||
|
||||
RASPBERRY_PI_CONNECTED = rpi.is_raspberry_pi()
|
||||
|
||||
|
||||
#############################################################
|
||||
# Web & API routes
|
||||
#############################################################
|
||||
|
||||
|
||||
web = Web(app, printer)
|
||||
|
||||
if __name__ == "__main__":
|
||||
app.run(debug=True, use_reloader=False, host='0.0.0.0', ssl_context='adhoc')
|
||||
|
||||
limiter = Limiter(
|
||||
app,
|
||||
key_func=get_remote_address,
|
||||
get_remote_address,
|
||||
app=app,
|
||||
default_limits=["1500 per day", "500 per hour"]
|
||||
)
|
||||
)
|
||||
|
||||
@app.route('/')
|
||||
@limiter.limit("1/second", override_defaults=False)
|
||||
@@ -103,15 +138,12 @@ def api_print_sms():
|
||||
txt = request.form["txt"]
|
||||
sign = request.form["signature"]
|
||||
except Exception as e:
|
||||
flash(e,'error')
|
||||
redirect(url_for('index'))
|
||||
app.logger.error("Whoops, no forms submitted or missing signature :" + str(e))
|
||||
flash("Whoops, no forms submitted or missing signature : " + str(e))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
try:
|
||||
web.print_sms(txt,sign)
|
||||
except Exception as e:
|
||||
pass
|
||||
|
||||
return redirect(url_for('index'))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
@app.route('/api/print/img', methods=['POST'])
|
||||
@limiter.limit("6/minute", override_defaults=False)
|
||||
@@ -121,44 +153,63 @@ def api_print_image():
|
||||
try:
|
||||
sign = request.form["signature"]
|
||||
except Exception as e:
|
||||
flash(str(e),'error')
|
||||
app.logger.error(str(e) + " - Whoops, no forms submitted or missing signature.")
|
||||
return redirect(url_for('index'))
|
||||
app.logger.error("Whoops, no forms submitted or missing signature :" + str(e))
|
||||
flash("Whoops, no forms submitted or missing signature : " + str(e))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
|
||||
|
||||
if request.method == 'POST':
|
||||
# check if the post request has the file part
|
||||
try:
|
||||
if 'img' not in request.files:
|
||||
flash('No file found. Did you use the good form ?', 'error')
|
||||
app.logger.error("No file found. Did you use the good form ?")
|
||||
return redirect(url_for("index"))
|
||||
app.logger.error("Whoops, no images submitted :" + str(e))
|
||||
flash("Whoops, no images submitted : " + str(e))
|
||||
else:
|
||||
file = request.files['img']
|
||||
except Exception as e:
|
||||
if sign is not None and photo is not None:
|
||||
pass
|
||||
else:
|
||||
flash(str(e), 'error')
|
||||
app.logger.error("Couldn't get an image nor signature : " + str(e))
|
||||
app.logger.error('Error getting the files :' + str(e))
|
||||
flash('Error getting the files :' + str(e))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
# If the user does not select a file, the browser submits an
|
||||
# empty file without a filename.
|
||||
if file.filename == '':
|
||||
app.logger.error("Submitted file has no filename !")
|
||||
flash('No file submitted, please select a file','error')
|
||||
flash("Submitted file has no filename !")
|
||||
return redirect(url_for("index"))
|
||||
|
||||
|
||||
try:
|
||||
app.logger.debug("Sending the image to the printer.")
|
||||
web.print_image(file, sign)
|
||||
except Exception as e:
|
||||
pass
|
||||
else:
|
||||
flash('Cannot access to page with this method.','error')
|
||||
app.logger.debug('Bad access type to this API.')
|
||||
|
||||
app.logger.error("The image could not be printed because : " + str(e) )
|
||||
flash("The image could not be printed because : " + str(e))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
else:
|
||||
app.logger.error("Method not allowed")
|
||||
flash("Method not allowed")
|
||||
return redirect(url_for("index"))
|
||||
|
||||
flash('Picture printed ! '),
|
||||
return redirect(url_for("index"))
|
||||
|
||||
|
||||
@app.route('/api/camera/picture')
|
||||
def camera_picture():
|
||||
# Returns a picture taken by the camera
|
||||
if RASPBERRY_PI_CONNECTED:
|
||||
try:
|
||||
return rpi.camera_picture()
|
||||
except Exception as e:
|
||||
return jsonify({'message': 'Error getting the stream : ' + e}), 500
|
||||
else:
|
||||
return jsonify({'message': 'No camera present'}), 500
|
||||
|
||||
|
||||
|
||||
@app.route('/login')
|
||||
@limiter.limit("1/second", override_defaults=False)
|
||||
def login_page():
|
||||
@@ -174,10 +225,25 @@ def logout_page():
|
||||
@app.errorhandler(429)
|
||||
def ratelimit_handler(e):
|
||||
flash("Rate limit reached, please slow down :) ( Currently at "+ e.description + ")", 'error')
|
||||
app.logger.debug('Rate limit reached ' + str(e.description))
|
||||
return redirect(url_for("index"))
|
||||
|
||||
@app.route("/ping")
|
||||
@limiter.exempt
|
||||
def ping():
|
||||
flash("🏓 Pong !",'info')
|
||||
app.logger.debug('🏓 Pong !')
|
||||
return redirect(url_for("index"))
|
||||
|
||||
@socketio.on('ping')
|
||||
def handle_message(data):
|
||||
app.logger.debug('Received : ' + str(data))
|
||||
socketio.emit('pong',"Pong !")
|
||||
|
||||
@socketio.on('get_camera_status')
|
||||
def camera_status():
|
||||
app.logger.debug('Client asked if we had a camera')
|
||||
if RASPBERRY_PI_CONNECTED:
|
||||
socketio.emit("camera_status", True)
|
||||
else:
|
||||
socketio.emit("camera_status", False)
|
||||
|
||||
170
src/printer.py
170
src/printer.py
@@ -1,9 +1,9 @@
|
||||
# Importing the module to mage the connection to the printer.
|
||||
from flask import flash
|
||||
from escpos.printer import Usb, USBNotFoundError
|
||||
from time import sleep, gmtime, strftime
|
||||
import os.path
|
||||
from PIL import Image
|
||||
from PIL import Image, ImageEnhance, ImageOps
|
||||
import numpy as np
|
||||
|
||||
|
||||
class Printer(object):
|
||||
@@ -45,18 +45,14 @@ class Printer(object):
|
||||
match status:
|
||||
case 0:
|
||||
self.app.logger.error('Printer has no more paper, aborting...')
|
||||
flash("No more paper on the printer. Sorry.",category='error')
|
||||
self.printer.close()
|
||||
return False
|
||||
raise Exception("No more paper in the printer")
|
||||
case 1:
|
||||
self.app.logger.warning('Printer needs paper to be changed very soon ! ')
|
||||
flash('Printer needs paper to be changed very soon ! ', category='info')
|
||||
self.printer.close()
|
||||
return True
|
||||
case 2:
|
||||
self.app.logger.debug('Printer has paper, good to go')
|
||||
self.printer.close()
|
||||
return True
|
||||
|
||||
def init_printer(self):
|
||||
|
||||
@@ -98,81 +94,149 @@ class Printer(object):
|
||||
self.ready = True;
|
||||
self.printer.close();
|
||||
|
||||
if not self.check_paper():
|
||||
return False
|
||||
self.check_paper()
|
||||
|
||||
return True
|
||||
|
||||
def print_sms(self, msg, signature) -> bool:
|
||||
clean_msg = str(msg)
|
||||
def print_sms(self, msg, signature="",bold=False):
|
||||
clean_msg = str(msg) + "\n"
|
||||
clean_signature = str(signature)
|
||||
|
||||
if not self.check_paper():
|
||||
return False
|
||||
|
||||
if len(clean_msg) > 256 or len(clean_msg) < 3 :
|
||||
if len(clean_msg) > 4096:
|
||||
self.app.logger.warning("Could not print message of this length: " + str(len(clean_msg)))
|
||||
flash("Could not print message of this length :" + str(len(clean_msg)) + ", needs to between 3 and 256 caracters long.",category='error')
|
||||
return False
|
||||
raise Exception("Could not print message of this length :" + str(len(clean_msg)) + ", needs to be below 4096 caracters long.")
|
||||
|
||||
if len(signature) > 256 or len(signature) < 3:
|
||||
self.app.logger.warning("Could not print message without a signature.")
|
||||
flash("Could not print message without a signature.",category='error')
|
||||
return False
|
||||
if len(signature) > 256:
|
||||
self.app.logger.warning("Could not print signature of this length: " + str(len(clean_signature)))
|
||||
raise Exception("Could not print signature of this length :" + str(len(clean_signature)) + ", needs to be below 256 caracters long.")
|
||||
|
||||
if not os.getenv('LIPY_DEBUG') == True:
|
||||
try:
|
||||
self.printer.open(self.usb_args);
|
||||
self.printer.set(align='left', font='a', bold=False, underline=0, width=1, height=1, density=8, invert=False, smooth=True, flip=False, double_width=False, double_height=False, custom_size=False)
|
||||
self.printer.textln(clean_msg)
|
||||
self.printer.set(align='left', font='b', bold=False, underline=1, width=1, height=1, density=9, invert=False, smooth=True, flip=False, double_width=False, double_height=False, custom_size=False)
|
||||
self.printer.textln("> " + clean_signature + " @ " + strftime("%Y-%m-%d %H:%M:%S", gmtime()))
|
||||
self.printer.cut()
|
||||
self.printer.close
|
||||
self.printer.set(align='center', font='a', bold=bold)
|
||||
self.printer.textln(clean_msg )
|
||||
if clean_signature:
|
||||
self.printer.textln(clean_signature )
|
||||
self.printer.close()
|
||||
except Exception as e:
|
||||
flash("Unable to print because : " + e)
|
||||
self.app.logger.error("Unable to print because : " + str(e))
|
||||
|
||||
flash("Message printed : " + clean_msg ,category='info')
|
||||
self.app.logger.info("Printed text")
|
||||
return True
|
||||
|
||||
def print_img(self, path, sign):
|
||||
def print_img(self, path, sign="",center=True,process=False):
|
||||
clean_signature = str(sign)
|
||||
|
||||
if len(sign) > 256:
|
||||
self.app.logger.warning("Could not print signature of this length: " + str(len(clean_signature)))
|
||||
raise Exception("Could not print signature of this length :" + str(len(clean_signature)) + ", needs to be below 256 caracters long.")
|
||||
|
||||
if not os.path.isfile(str(path)):
|
||||
self.app.logger.warning("File does not exist : " + str(path))
|
||||
flash('The file path for this image :' + str(path) + " wasn't found. Please try again.", 'error')
|
||||
return False
|
||||
raise Exception('The file path for this image :' + str(path) + " wasn't found. Please try again.")
|
||||
else:
|
||||
self.app.logger.debug("Printing file from " + str(path))
|
||||
|
||||
|
||||
|
||||
if process:
|
||||
try:
|
||||
self.app.logger.debug("Resizing the image")
|
||||
with Image.open(path) as im:
|
||||
|
||||
basewidth = 575
|
||||
img = Image.open(path)
|
||||
wpercent = (basewidth/float(img.size[0]))
|
||||
hsize = int((float(img.size[1])*float(wpercent)))
|
||||
img = img.resize((basewidth,hsize), Image.ANTIALIAS)
|
||||
if img.height > 1000:
|
||||
flash("Image is too long, sorry ! Keep it below 500×1000 pixels.",'error')
|
||||
return False
|
||||
img.save(path)
|
||||
self.app.logger.debug("Proccessing the image")
|
||||
path = process_image(self, path)
|
||||
except Exception as e:
|
||||
flash(str(e))
|
||||
self.app.logger.error(str(e))
|
||||
return False
|
||||
else:
|
||||
self.app.logger.warning("Not proccessing the image")
|
||||
|
||||
|
||||
try:
|
||||
self.printer.open(self.usb_args)
|
||||
self.printer.textln("> " + clean_signature + " @ " + strftime("%Y-%m-%d %H:%M:%S", gmtime()))
|
||||
self.printer.image(path)
|
||||
self.printer.cut()
|
||||
self.printer.image(path,center=center)
|
||||
self.printer.close()
|
||||
self.app.logger.debug("Printed an image : " + str(path))
|
||||
return True
|
||||
os.remove(path)
|
||||
self.app.logger.debug("Removed image : " + str(path))
|
||||
except Exception as e:
|
||||
self.printer.close()
|
||||
flash(str(e),'error')
|
||||
self.app.logger.error(str(e))
|
||||
return False
|
||||
|
||||
self.app.logger.info("Printed a picture")
|
||||
return True
|
||||
|
||||
def qr(self, content):
|
||||
try:
|
||||
self.printer.open(self.usb_args)
|
||||
self.printer.qr(content, center=True)
|
||||
self.printer.close()
|
||||
except Exception as e:
|
||||
self.printer.close()
|
||||
self.app.logger.error(str(e))
|
||||
return False
|
||||
|
||||
self.app.logger.info("Printed a QR")
|
||||
return True
|
||||
|
||||
|
||||
|
||||
def cut(self):
|
||||
try:
|
||||
self.printer.open(self.usb_args)
|
||||
self.printer.cut()
|
||||
self.printer.close()
|
||||
except Exception as e:
|
||||
self.printer.close()
|
||||
self.app.logger.error(str(e))
|
||||
raise e
|
||||
|
||||
self.app.logger.info("Did a cut")
|
||||
return True
|
||||
|
||||
|
||||
def process_image(self, path):
|
||||
brightness_factor = 1.5 # Used only if image is too dark
|
||||
brightness_threshold = 100 # Brightness threshold (0–255)
|
||||
contrast_factor = 0.6 # Less than 1.0 = lower contrast
|
||||
max_width = 575
|
||||
max_height = 1000
|
||||
|
||||
with Image.open(path) as original_img:
|
||||
# Convert to RGB if needed (JPEG doesn't support alpha)
|
||||
if original_img.mode in ("RGBA", "P"):
|
||||
self.app.logger.debug("Converting the image to RGB from RGBA")
|
||||
original_img = original_img.convert("RGB")
|
||||
|
||||
# Resize while maintaining aspect ratio
|
||||
original_img.thumbnail((max_width, max_height), Image.LANCZOS)
|
||||
self.app.logger.debug("Resized the image")
|
||||
|
||||
# # Convert to grayscale for dithering
|
||||
# dithered_img = original_img.convert("L").convert("1") # Dithering using default method (Floyd–Steinberg)
|
||||
# self.app.logger.debug("Dithered the image")
|
||||
|
||||
# Compute brightness of original image (grayscale average)
|
||||
grayscale = original_img.convert("L")
|
||||
avg_brightness = np.array(grayscale).mean()
|
||||
self.app.logger.debug("Average brightness of the image : " + str(avg_brightness) )
|
||||
|
||||
# Dynamically compute brightness factor if too dark
|
||||
if avg_brightness < brightness_threshold:
|
||||
brightness_factor = 1 + (brightness_threshold - avg_brightness) / brightness_threshold
|
||||
brightness_factor = min(max(brightness_factor, 1.1), 2.5) # Clamp between 1.1 and 2.5
|
||||
self.app.logger.debug(f"Image too dark, increasing brightness by a factor of {brightness_factor:.2f}")
|
||||
enhancer = ImageEnhance.Brightness(original_img)
|
||||
original_img = enhancer.enhance(brightness_factor)
|
||||
|
||||
# # Reduce contrast
|
||||
# contrast_enhancer = ImageEnhance.Contrast(original_img)
|
||||
# original_img = contrast_enhancer.enhance(contrast_factor)
|
||||
|
||||
# Final resize check
|
||||
if original_img.height > max_height:
|
||||
raise ValueError("Image is too long, sorry! Keep it below 575×1000 pixels.")
|
||||
self.app.logger.error("Image is too long, sorry! Keep it below 575×1000 pixels.")
|
||||
return False
|
||||
|
||||
# Convert to JPEG and save
|
||||
jpeg_path = os.path.splitext(path)[0] + "_processed.jpg"
|
||||
original_img.save(jpeg_path, format='JPEG', quality=95, optimize=True)
|
||||
|
||||
return jpeg_path
|
||||
|
||||
230
src/raspberry.py
Normal file
230
src/raspberry.py
Normal file
@@ -0,0 +1,230 @@
|
||||
from flask_socketio import SocketIO
|
||||
from gpiozero import Button, LED, DigitalOutputDevice
|
||||
from time import sleep, gmtime, strftime
|
||||
from PIL import Image
|
||||
import io # To check if we are on a Raspberry Pi
|
||||
import subprocess
|
||||
import os
|
||||
|
||||
class Raspberry(object):
|
||||
"""
|
||||
This class will manage three things :
|
||||
- Connecting to a USB webcam
|
||||
- Managing a push button
|
||||
- Activating a flash ( or light )
|
||||
- Flash an indicator light
|
||||
|
||||
"""
|
||||
def __init__(self, printer, app, socketio, button_gpio_port_number, indicator_gpio_port_number, flash_gpio_port_number, is_flash_present,):
|
||||
self.printer = printer
|
||||
self.socketio = socketio
|
||||
self.app = app
|
||||
|
||||
self.flash_gpio = flash_gpio_port_number
|
||||
self.is_flash_present = is_flash_present
|
||||
self.button_gpio = button_gpio_port_number
|
||||
self.led_gpio = indicator_gpio_port_number
|
||||
self.image_path = self.app.config['UPLOAD_FOLDER'] + '/image.jpg'
|
||||
|
||||
def is_raspberry_pi(self, raise_on_errors=False):
|
||||
# Check if we are running on a raspberry pi
|
||||
try:
|
||||
with io.open('/proc/cpuinfo', 'r') as cpuinfo:
|
||||
found = False
|
||||
for line in cpuinfo:
|
||||
if line.startswith('Hardware'):
|
||||
found = True
|
||||
label, value = line.strip().split(':', 1)
|
||||
value = value.strip()
|
||||
if value not in (
|
||||
'BCM2708',
|
||||
'BCM2709',
|
||||
'BCM2711',
|
||||
'BCM2835',
|
||||
'BCM2836'
|
||||
):
|
||||
self.app.logger.debug('This system does not appear to be a Raspberry Pi.')
|
||||
return False
|
||||
|
||||
if not found:
|
||||
self.app.logger.error('Couldn\'t get sufficient hardware information from /proc/cpuinfo, Unable to determine if we are on a Raspberry Pi.')
|
||||
return False
|
||||
except IOError:
|
||||
self.app.logger.error('Unable to open `/proc/cpuinfo`.')
|
||||
return False
|
||||
|
||||
self.app.logger.debug('It seems we are on a Raspberry Pi')
|
||||
|
||||
try:
|
||||
self.initialise_gpio()
|
||||
except Exception as e:
|
||||
self.app.logger.debug('Could not init GPIO : ' + str(e))
|
||||
raise e
|
||||
return True
|
||||
|
||||
def initialise_gpio(self):
|
||||
self.app.logger.debug('Initializing GPIO')
|
||||
|
||||
self.led = LED(self.led_gpio)
|
||||
self.app.logger.debug('Activated indicator LED')
|
||||
self.indicator_countdown(iters=3)
|
||||
self.button = Button(self.button_gpio, pull_up=True, bounce_time=0.1)
|
||||
self.button.when_pressed = self.on_button_pressed
|
||||
self.app.logger.debug('Activated button')
|
||||
|
||||
# The "flash" is a relay-controlled device ( light bulb for example )
|
||||
self.flash = DigitalOutputDevice(self.flash_gpio)
|
||||
self.flash_toggle()
|
||||
self.app.logger.debug('Activated flash')
|
||||
|
||||
def indicator_countdown(self,iters=10,multi=10):
|
||||
for i in range(iters,0,-1):
|
||||
self.led.on()
|
||||
sleep(i/multi)
|
||||
self.led.off()
|
||||
sleep(i/multi)
|
||||
|
||||
def indicator_led(self,timing=0.2,l=5):
|
||||
for i in range(l):
|
||||
self.app.logger.debug("LED turned on")
|
||||
self.led.on()
|
||||
sleep(timing)
|
||||
self.led.off()
|
||||
self.app.logger.debug("LED turned off")
|
||||
sleep(timing)
|
||||
|
||||
def flash_toggle(self):
|
||||
self.app.logger.debug("Flash turned on")
|
||||
self.flash.on()
|
||||
sleep(0.3)
|
||||
self.flash.off()
|
||||
self.app.logger.debug("Flash turned off")
|
||||
|
||||
def take_picture(self):
|
||||
# Validate if the image path is valid
|
||||
if not os.path.isdir(os.path.dirname(self.image_path)):
|
||||
self.app.logger.error(f"Invalid directory for image path: {self.image_path}")
|
||||
return False
|
||||
|
||||
try:
|
||||
result = subprocess.run(
|
||||
['fswebcam', '--no-banner', '-r', '1920x1080', self.image_path],
|
||||
check=True, # Will raise CalledProcessError if the command fails
|
||||
stdout=subprocess.PIPE, # Capture standard output
|
||||
stderr=subprocess.PIPE # Capture error output
|
||||
)
|
||||
|
||||
# Optionally log the command output
|
||||
self.app.logger.debug(f"Image captured successfully: {result.stdout.decode()}")
|
||||
except subprocess.CalledProcessError as e:
|
||||
# Log error output if available
|
||||
self.app.logger.error(f"Unable to take a picture. Error: {e.stderr.decode()}")
|
||||
return False
|
||||
except Exception as e:
|
||||
# Catch any unexpected errors
|
||||
self.app.logger.error(f"Unexpected error while taking picture: {str(e)}")
|
||||
return False
|
||||
|
||||
# # Overlay logo
|
||||
# logo_path = 'src/static/images/requin.png' # Update path as needed
|
||||
# if not self.overlay_logo(self.image_path, logo_path):
|
||||
# self.app.logger.warning("Picture taken but failed to overlay logo.")
|
||||
|
||||
return True
|
||||
|
||||
|
||||
def overlay_logo(self, image_path, logo_path, output_path=None, position='bottom_right', margin=10):
|
||||
try:
|
||||
image = Image.open(image_path).convert("RGBA")
|
||||
logo = Image.open(logo_path).convert("RGBA")
|
||||
|
||||
# Resize logo if it's too big (logo will be 30% the width of the image)
|
||||
logo_ratio = 0.30 # You can change the ratio if you want the logo bigger or smaller
|
||||
logo_width = int(image.width * logo_ratio)
|
||||
logo_height = int(logo.height * (logo_width / logo.width))
|
||||
logo = logo.resize((logo_width, logo_height), Image.Resampling.LANCZOS)
|
||||
|
||||
# Calculate position based on the chosen location
|
||||
if position == 'bottom_right':
|
||||
x = image.width - logo.width - margin
|
||||
y = image.height - logo.height - margin
|
||||
elif position == 'top_left':
|
||||
x, y = margin, margin
|
||||
elif position == 'top_right':
|
||||
x = image.width - logo.width - margin
|
||||
y = margin
|
||||
elif position == 'bottom_left':
|
||||
x = margin
|
||||
y = image.height - logo.height - margin
|
||||
else:
|
||||
raise ValueError("Invalid position. Choose from 'bottom_right', 'top_left', 'top_right', or 'bottom_left'.")
|
||||
|
||||
# Composite the logo onto the image
|
||||
image.paste(logo, (x, y), logo) # Use logo as its own alpha mask
|
||||
|
||||
# Save the result
|
||||
if not output_path:
|
||||
output_path = image_path # Overwrite the original image if no output path is given
|
||||
image.save(output_path)
|
||||
|
||||
except Exception as e:
|
||||
self.app.logger.error(f"Error overlaying logo: {e}")
|
||||
return False
|
||||
|
||||
return True
|
||||
|
||||
def crop_to_square(self, image_path, output_path=None):
|
||||
try:
|
||||
image = Image.open(image_path)
|
||||
width, height = image.size
|
||||
|
||||
# Determine shorter side
|
||||
new_edge = min(width, height)
|
||||
|
||||
# Calculate cropping box (centered)
|
||||
left = (width - new_edge) // 2
|
||||
top = (height - new_edge) // 2
|
||||
right = left + new_edge
|
||||
bottom = top + new_edge
|
||||
|
||||
image = image.crop((left, top, right, bottom))
|
||||
|
||||
if not output_path:
|
||||
output_path = image_path # Overwrite original
|
||||
|
||||
image.save(output_path)
|
||||
|
||||
except Exception as e:
|
||||
self.app.logger.error(f"Error cropping image to square: {e}")
|
||||
return False
|
||||
|
||||
return True
|
||||
|
||||
def on_button_pressed(self):
|
||||
self.app.logger.debug("Button has been pressed")
|
||||
self.led.on()
|
||||
self.app.logger.debug("Counting down")
|
||||
self.indicator_countdown(iters=4,multi=20) # The indicator will flash a countdown LED
|
||||
self.app.logger.debug("Taking picture")
|
||||
try:
|
||||
self.flash.on()
|
||||
self.take_picture()
|
||||
except Exception as e:
|
||||
self.app.logger.error("Could not take a picture after the button press : " + str(e))
|
||||
finally:
|
||||
self.flash.off()
|
||||
self.app.logger.debug("Printing picture")
|
||||
self.led.on()
|
||||
self.crop_to_square(self.image_path)
|
||||
self.printer.print_img('src/static/images/extase-club.png',process=True)
|
||||
self.printer.print_img(self.image_path,process=True)
|
||||
self.printer.print_sms("")
|
||||
self.printer.print_sms("With Love From Société.Vide",signature="",bold=True)
|
||||
self.printer.print_sms("Printed by LittlePrynter",signature="")
|
||||
self.printer.print_sms("n07070.xyz",signature="")
|
||||
self.printer.print_sms(strftime("%Y-%m-%d %H:%M", gmtime()),signature="")
|
||||
self.printer.qr("https://n07070.xyz/articles/littleprynter")
|
||||
self.printer.cut()
|
||||
self.led.off()
|
||||
self.app.logger.debug("Done printing picture")
|
||||
return True
|
||||
BIN
src/static/images/extase-club.png
Normal file
BIN
src/static/images/extase-club.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 61 KiB |
BIN
src/static/images/requin.png
Normal file
BIN
src/static/images/requin.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 32 KiB |
@@ -1,39 +1,159 @@
|
||||
let streaming;
|
||||
var current_stream;
|
||||
var current_camera_is;
|
||||
var supports_facing_mode;
|
||||
var camera_options;
|
||||
|
||||
var width = document.getElementById("video").parentNode.parentElement.clientWidth;
|
||||
var height = width / (4 / 3);
|
||||
var socket = io();
|
||||
|
||||
function startup(){
|
||||
socket.on('connect', function() {
|
||||
socket.emit('ping', {data: 'I\'m connected!'});
|
||||
console.log("Sent a ping to the server :)");
|
||||
});
|
||||
|
||||
|
||||
socket.on('pong', () => {
|
||||
console.log('Received pong back ! ');
|
||||
});
|
||||
|
||||
socket.on('new_image', () => {
|
||||
console.log("Received new image event");
|
||||
const img = document.getElementById('snapshot');
|
||||
img.src = '/image?' + new Date().getTime(); // Bust cache
|
||||
});
|
||||
|
||||
|
||||
async function startup(){
|
||||
video = document.getElementById('video');
|
||||
canvas = document.getElementById('canvas');
|
||||
photo = document.getElementById('photo');
|
||||
switch_cameras = document.getElementById('flip')
|
||||
printButton = document.getElementById('print_button');
|
||||
|
||||
if (check_webcam() === true ){
|
||||
console.log("Checking for client webcam capabilities");
|
||||
// We have a camera_options dictionnary in return, or false if the device does not support webcams.
|
||||
let client_webcam_capabilities = await check_webcam_capabilies();
|
||||
|
||||
|
||||
if ( client_webcam_capabilities != false ){
|
||||
get_webcam(client_webcam_capabilities);
|
||||
setup_events();
|
||||
clear_canvas();
|
||||
} else {
|
||||
console.log("Checking for server webcam capabilities");
|
||||
let server_webcam_capabilities = await check_server_camera();
|
||||
if (server_webcam_capabilities === true ){
|
||||
console.log("The server has a camera, using it.");
|
||||
console.log("TODO");
|
||||
}
|
||||
else {
|
||||
no_webcam_error();
|
||||
console.log("Seems like it's impossible to get a webcam.");
|
||||
console.log("Seems like it's impossible to get a webcam from the client nor the server.");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function check_webcam(){
|
||||
console.log("Cheking for a camera...");
|
||||
if (get_front_webcam()) {
|
||||
console.log("Got front camera !");
|
||||
return true;
|
||||
async function check_webcam_capabilies(){
|
||||
console.log("Checking for a the capabilities of the client's media devices");
|
||||
|
||||
// We try to start with the front facing camera,
|
||||
// if we have no support, we switch back to a normal camera.
|
||||
try {
|
||||
// We first check if the navigator has a media device
|
||||
if (!navigator.mediaDevices?.enumerateDevices) {
|
||||
console.log("enumerateDevices() not supported.");
|
||||
return false;
|
||||
} else {
|
||||
// List cameras and microphones.
|
||||
console.log("The device has the following media devices : ")
|
||||
await navigator.mediaDevices
|
||||
.enumerateDevices()
|
||||
.then((devices) => {
|
||||
devices.forEach((device) => {
|
||||
console.log(`${device.kind}: ${device.label} id = ${device.deviceId}`);
|
||||
});
|
||||
})
|
||||
.catch((err) => {
|
||||
console.error(`${err.name}: ${err.message}`);
|
||||
return false;
|
||||
});
|
||||
}
|
||||
} catch (e) {
|
||||
console.log("The device does not seem to support webcams " + e);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (get_any_webcam()) {
|
||||
console.log("Got a webcam !");
|
||||
return true;
|
||||
console.log("Checking for the supported constraints of the media devices ");
|
||||
try {
|
||||
const supports = navigator.mediaDevices.getSupportedConstraints();
|
||||
if (!supports['facingMode']) {
|
||||
throw new Error("This browser does not support facingMode!");
|
||||
} else {
|
||||
console.log("The device supports facing mode, selecting the front camera by default");
|
||||
supports_facing_mode = true;
|
||||
camera_options = {
|
||||
video: {
|
||||
facingMode: 'user', // Or 'environment' if we want a camera facing away
|
||||
},
|
||||
audio: false
|
||||
};
|
||||
return camera_options;
|
||||
}
|
||||
console.log("Nope");
|
||||
} catch (e) {
|
||||
console.log("Resetting to default camera : " + e);
|
||||
supports_facing_mode = false;
|
||||
camera_options = {
|
||||
video: true,
|
||||
audio: false
|
||||
};
|
||||
return camera_options
|
||||
}
|
||||
}
|
||||
|
||||
async function get_webcam(options){
|
||||
stop_video_streams();
|
||||
|
||||
try {
|
||||
navigator.mediaDevices.getUserMedia(options)
|
||||
.then(function(stream) {
|
||||
// on success, stream it in video tag
|
||||
// the video tag is hidden, as is the canvas.
|
||||
console.log("Got the webcam stream");
|
||||
|
||||
printButton.removeAttribute("disabled","");
|
||||
current_stream = stream;
|
||||
video.srcObject = stream;
|
||||
video.setAttribute('autoplay', '');
|
||||
video.setAttribute('muted', '');
|
||||
video.setAttribute('playsinline', '')
|
||||
video.play();
|
||||
return true;
|
||||
})
|
||||
.catch(function(err) {
|
||||
console.log("Didn't manage to get a camera :" + err);
|
||||
return false;
|
||||
});
|
||||
} catch (err) {
|
||||
console.log("Didn't manage to get a camera :" + err);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
async function check_server_camera(){
|
||||
console.log("Checking for a camera on the server");
|
||||
|
||||
await socket.emit('get_camera_status', (response) => {
|
||||
if (response){
|
||||
console.log("Server has a camera !");
|
||||
return true;
|
||||
} else {
|
||||
console.log("The server doesn't seem to have a camera");
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
function setup_events(){
|
||||
@@ -66,7 +186,7 @@ function setup_events(){
|
||||
}, false);
|
||||
|
||||
switch_cameras.addEventListener('click', function(ev) {
|
||||
flip_cameras();
|
||||
flip_cameras(camera_options);
|
||||
}, false );
|
||||
|
||||
printButton.addEventListener('click', function(ev){
|
||||
@@ -124,7 +244,7 @@ function print_picture(data){
|
||||
let time = currentDate.getHours() + ":" + currentDate.getMinutes() + ":" + currentDate.getSeconds();
|
||||
|
||||
formData.set("img", picture, "picture.png");
|
||||
formData.set("signature", "Printed via the webcam @ " + time)
|
||||
formData.set("signature", "Webcam")
|
||||
|
||||
fetch(url, {
|
||||
method: 'POST', // or 'PUT'
|
||||
@@ -132,31 +252,14 @@ function print_picture(data){
|
||||
// headers:{
|
||||
// 'Content-Type': 'multipart/form-data'
|
||||
// }
|
||||
}).then(function(response) { console.log('Success:', response); alert("Picture printed."); } , true)
|
||||
}).then(function(response) { console.log('Success:', response); } , true)
|
||||
.catch(error => console.error('Error:', error), false);
|
||||
|
||||
}
|
||||
|
||||
function flip_cameras(){
|
||||
switch (current_camera_is) {
|
||||
case "front":
|
||||
try {
|
||||
get_any_webcam();
|
||||
} catch (e) {
|
||||
console.log("Could not get another camera");
|
||||
get_front_webcam();
|
||||
}
|
||||
break;
|
||||
case "any":
|
||||
try {
|
||||
get_front_webcam();
|
||||
} catch (e) {
|
||||
console.log("Could not get another camera");
|
||||
get_any_webcam();
|
||||
}
|
||||
break;
|
||||
default:
|
||||
console.log("Impossible to switch cameras : none is selected.");
|
||||
function flip_cameras(camera_options){
|
||||
if ( supports_facing_mode ) {
|
||||
get_webcam((camera_options.video.facingMode === 'user' ? 'environment' : 'user'));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -177,81 +280,6 @@ function stop_video_streams(){
|
||||
}
|
||||
}
|
||||
|
||||
async function get_webcam(options){
|
||||
stop_video_streams();
|
||||
|
||||
try {
|
||||
await navigator.mediaDevices.getUserMedia(options)
|
||||
.then(function(stream) {
|
||||
// on success, stream it in video tag
|
||||
// the video tag is hidden, as is the canvas.
|
||||
console.log("Got a camera ( generic )");
|
||||
|
||||
printButton.removeAttribute("disabled","");
|
||||
current_stream = stream;
|
||||
video.srcObject = stream;
|
||||
video.play();
|
||||
return true;
|
||||
})
|
||||
.catch(function(err) {
|
||||
console.log("Didn't manage to get a camera :" + err);
|
||||
return false;
|
||||
});
|
||||
} catch (err) {
|
||||
console.log("Didn't manage to get a camera :" + err);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
function get_any_webcam(){
|
||||
var camera_options = {
|
||||
video: {
|
||||
facingMode: 'environment', // Or 'environment' if we want a camera facing away
|
||||
},
|
||||
audio: false
|
||||
};
|
||||
|
||||
if(get_webcam(camera_options)){
|
||||
console.log("Got any camera, or environment camera.");
|
||||
current_camera_is = "any";
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
function get_front_webcam(){
|
||||
// We try to start with the front facing camera,
|
||||
// if we have no support, we switch back to a normal camera.
|
||||
try {
|
||||
const supports = navigator.mediaDevices.getSupportedConstraints();
|
||||
if (!supports['facingMode']) {
|
||||
throw new Error("This browser does not support facingMode!");
|
||||
} else {
|
||||
var camera_options = {
|
||||
video: {
|
||||
facingMode: 'user', // Or 'environment' if we want a camera facing away
|
||||
},
|
||||
audio: false
|
||||
};
|
||||
}
|
||||
} catch (e) {
|
||||
console.log("Resetting to default camera : " + e);
|
||||
var camera_options = {
|
||||
video: true,
|
||||
audio: false
|
||||
};
|
||||
}
|
||||
|
||||
if(get_webcam(camera_options)){
|
||||
console.log("Got the front camera");
|
||||
current_camera_is = "front";
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
function no_webcam_error(){
|
||||
console.log("Seems like they is no webcam available.")
|
||||
|
||||
|
||||
@@ -56,4 +56,6 @@
|
||||
<p class=" text-center">Little Prynter is built by <a href="https://n07070.xyz/about-me/">n07070</a> because it's fun :) | <a href="https://git.n07070.xyz/n07070/littleprynter">Source code - AGPLv3</a></p>
|
||||
</footer>
|
||||
</body>
|
||||
<script src="https://cdn.socket.io/4.0.0/socket.io.min.js"></script>
|
||||
|
||||
</html>
|
||||
|
||||
@@ -27,11 +27,6 @@
|
||||
<path d="M15 12a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1V6a1 1 0 0 1 1-1h1.172a3 3 0 0 0 2.12-.879l.83-.828A1 1 0 0 1 6.827 3h2.344a1 1 0 0 1 .707.293l.828.828A3 3 0 0 0 12.828 5H14a1 1 0 0 1 1 1v6zM2 4a2 2 0 0 0-2 2v6a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V6a2 2 0 0 0-2-2h-1.172a2 2 0 0 1-1.414-.586l-.828-.828A2 2 0 0 0 9.172 2H6.828a2 2 0 0 0-1.414.586l-.828.828A2 2 0 0 1 3.172 4H2z"/>
|
||||
<path d="M8 11a2.5 2.5 0 1 1 0-5 2.5 2.5 0 0 1 0 5zm0 1a3.5 3.5 0 1 0 0-7 3.5 3.5 0 0 0 0 7zM3 6.5a.5.5 0 1 1-1 0 .5.5 0 0 1 1 0z"/>
|
||||
</svg>
|
||||
<b>&</b>
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="54" height="54" fill="currentColor" class="bi bi-printer" viewBox="0 0 16 16">
|
||||
<path d="M2.5 8a.5.5 0 1 0 0-1 .5.5 0 0 0 0 1z"/>
|
||||
<path d="M5 1a2 2 0 0 0-2 2v2H2a2 2 0 0 0-2 2v3a2 2 0 0 0 2 2h1v1a2 2 0 0 0 2 2h6a2 2 0 0 0 2-2v-1h1a2 2 0 0 0 2-2V7a2 2 0 0 0-2-2h-1V3a2 2 0 0 0-2-2H5zM4 3a1 1 0 0 1 1-1h6a1 1 0 0 1 1 1v2H4V3zm1 5a2 2 0 0 0-2 2v1H2a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h12a1 1 0 0 1 1 1v3a1 1 0 0 1-1 1h-1v-1a2 2 0 0 0-2-2H5zm7 2v3a1 1 0 0 1-1 1H5a1 1 0 0 1-1-1v-3a1 1 0 0 1 1-1h6a1 1 0 0 1 1 1z"/>
|
||||
</svg>
|
||||
</button>
|
||||
<button class="col-sm col-lg-3 offset-lg-2 btn btn-secondary justify-content-center" name="flip" id="flip" data-bs-toggle="tooltip" data-bs-placement="top" title="Change camera" disabled="">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="54" height="54" fill="currentColor" class="bi bi-arrow-repeat" viewBox="0 0 16 16">
|
||||
@@ -104,6 +99,7 @@
|
||||
|
||||
</style>
|
||||
|
||||
<script src="https://cdn.socket.io/4.0.0/socket.io.min.js"></script>
|
||||
<script type="text/javascript" src="{{ url_for('static',filename='js/webcam.js') }}"></script>
|
||||
|
||||
<br>
|
||||
|
||||
29
src/web.py
29
src/web.py
@@ -1,4 +1,4 @@
|
||||
from flask import Flask, request
|
||||
from flask import Flask, request, flash
|
||||
from werkzeug.utils import secure_filename
|
||||
from printer import Printer
|
||||
import time
|
||||
@@ -15,28 +15,29 @@ class Web(object):
|
||||
def print_sms(self, texte, sign: str):
|
||||
# TODO: verify the texte before printing it here ?
|
||||
self.app.logger.debug("Printing : " + str(texte) + " from " + str(sign))
|
||||
if not os.getenv('LIPY_DEBUG'):
|
||||
time.sleep(1)
|
||||
try:
|
||||
self.printer.print_sms(texte, sign)
|
||||
self.printer.cut()
|
||||
except Exception as e:
|
||||
self.app.logger.error(e)
|
||||
flash("Error while printing the SMS : "+ str(e))
|
||||
|
||||
return self.printer.print_sms(texte, sign)
|
||||
flash("You message " + str( texte ) + " has been printed :)")
|
||||
|
||||
def print_image(self, image, sign: str) -> bool:
|
||||
|
||||
def print_image(self, image, sign):
|
||||
self.app.logger.debug("Uploading file")
|
||||
try:
|
||||
self.app.logger.debug("Uploading file from " + str(sign))
|
||||
if self.upload_file(image):
|
||||
self.app.logger.debug("File has been uploaded, printing...")
|
||||
self.printer.print_img(os.path.join(self.app.config['UPLOAD_FOLDER'], secure_filename(image.filename)), sign)
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
self.printer.print_img(os.path.join(self.app.config['UPLOAD_FOLDER'], secure_filename(image.filename)), sign=sign,process=True)
|
||||
self.printer.cut()
|
||||
except Exception as e:
|
||||
self.app.logger.error(e)
|
||||
raise Exception
|
||||
flash("Could not upload file." + str(e))
|
||||
|
||||
else:
|
||||
flash("Could not upload file.",'error')
|
||||
return False
|
||||
flash("Your image has been printed :)")
|
||||
|
||||
def login(username: str,password: str) -> bool:
|
||||
pass
|
||||
@@ -58,7 +59,7 @@ class Web(object):
|
||||
self.app.logger.debug("File saved")
|
||||
except Exception as e:
|
||||
self.app.logger.error("Could not save file")
|
||||
flash(e,'error')
|
||||
flash(str(e),'error')
|
||||
return False
|
||||
|
||||
self.app.logger.debug("File saved to " + str(os.path.join(self.app.config['UPLOAD_FOLDER'], filename)))
|
||||
|
||||
Reference in New Issue
Block a user