mirror of
https://github.com/jayofelony/pwnagotchi.git
synced 2025-07-01 18:37:27 -04:00
Compare commits
35 Commits
Author | SHA1 | Date | |
---|---|---|---|
8edb0fdaa4 | |||
b557768159 | |||
5a6967eb4d | |||
c0c35231cf | |||
88362b3354 | |||
e50cf04967 | |||
85a64a3914 | |||
63d0c20a3e | |||
044639a6d4 | |||
7fc2c6b25f | |||
c5d5d9bb14 | |||
129ae92447 | |||
2560692ee2 | |||
1424e0d510 | |||
2c9eaa436a | |||
0c5c6058a5 | |||
a3fb39d78c | |||
bac62bc4aa | |||
f7a5f2a554 | |||
6921fd14a3 | |||
0dbd611ed5 | |||
18b9093f70 | |||
26913016b9 | |||
0786b6757e | |||
0f4eda8039 | |||
1672a35f09 | |||
80e7e6c550 | |||
ca274036ab | |||
649e8a4222 | |||
63f0a761c1 | |||
8a64918b55 | |||
c27cc0ac24 | |||
c9bdce0ea3 | |||
6422371346 | |||
3c5a95e0b8 |
3
.idea/misc.xml
generated
3
.idea/misc.xml
generated
@ -4,4 +4,7 @@
|
|||||||
<option name="sdkName" value="Python 3.11 (pwnagotchi)" />
|
<option name="sdkName" value="Python 3.11 (pwnagotchi)" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.10 (pwnagotchi-torch-bookworm)" project-jdk-type="Python SDK" />
|
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.10 (pwnagotchi-torch-bookworm)" project-jdk-type="Python SDK" />
|
||||||
|
<component name="PythonCompatibilityInspectionAdvertiser">
|
||||||
|
<option name="version" value="3" />
|
||||||
|
</component>
|
||||||
</project>
|
</project>
|
@ -11,7 +11,7 @@ Select ‘Credentials’ from the left menu, click ‘Create Credentials’, sel
|
|||||||
|
|
||||||
Now, the product name and consent screen need to be set -> click ‘Configure consent screen’ and follow the instructions. Once finished:
|
Now, the product name and consent screen need to be set -> click ‘Configure consent screen’ and follow the instructions. Once finished:
|
||||||
|
|
||||||
Select ‘Application type’ to be Desktop application.
|
Select ‘Application type’ to be Web application.
|
||||||
|
|
||||||
Enter an appropriate name.
|
Enter an appropriate name.
|
||||||
|
|
||||||
@ -19,7 +19,7 @@ Input http://localhost/ for ‘Authorized redirect URIs’.
|
|||||||
|
|
||||||
Select the correct oauth scope:
|
Select the correct oauth scope:
|
||||||
|
|
||||||
- drive
|
- drive.file
|
||||||
- drive.install
|
- drive.install
|
||||||
|
|
||||||
Click ‘Create’.
|
Click ‘Create’.
|
||||||
|
@ -7,7 +7,7 @@ It seems the Pi 5 is unable to run in monitor mode, will keep you updated on thi
|
|||||||
If you are using an older 32-bit version Raspberry Pi, ZeroWH, use this [fork](https://github.com/jayofelony/pwnagotchi-torch/releases/tag/v2.6.4) and make sure you download the `armhf` version.
|
If you are using an older 32-bit version Raspberry Pi, ZeroWH, use this [fork](https://github.com/jayofelony/pwnagotchi-torch/releases/tag/v2.6.4) and make sure you download the `armhf` version.
|
||||||
|
|
||||||
---
|
---
|
||||||
Download latest image file [here](https://github.com/jayofelony/pwnagotchi-bookworm/releases/tag/v2.6.7), and let it auto-update from here on out.
|
Download latest image file [here](https://github.com/jayofelony/pwnagotchi-bookworm/releases/tag/v2.7.3), and let it auto-update from here on out.
|
||||||
|
|
||||||
**Use RPi imager to flash, please don't flash a new user as this will mess with logs created.**
|
**Use RPi imager to flash, please don't flash a new user as this will mess with logs created.**
|
||||||
|
|
||||||
|
@ -176,7 +176,11 @@ def pwnagotchi_cli():
|
|||||||
os.system("rm /root/.auto-update && systemctl restart pwnagotchi")
|
os.system("rm /root/.auto-update && systemctl restart pwnagotchi")
|
||||||
os.system("systemctl restart pwnagotchi")
|
os.system("systemctl restart pwnagotchi")
|
||||||
print("Okay, give me a couple minutes. Just watch pwnlog while you wait.")
|
print("Okay, give me a couple minutes. Just watch pwnlog while you wait.")
|
||||||
|
elif user_input.lower() in ('n', 'no'):
|
||||||
print("Okay, guess not!")
|
print("Okay, guess not!")
|
||||||
|
else:
|
||||||
|
print("Invalid input.")
|
||||||
|
else:
|
||||||
print("You are currently on the latest release, v%s." % pwnagotchi.__version__)
|
print("You are currently on the latest release, v%s." % pwnagotchi.__version__)
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
|
15
builder/data/etc/default/zramswap
Normal file
15
builder/data/etc/default/zramswap
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# Specifies amount of zram devices to create.
|
||||||
|
# By default, zramswap-start will use all available cores.
|
||||||
|
#CORES=1
|
||||||
|
|
||||||
|
# Specifies the amount of RAM that should be used for zram
|
||||||
|
# based on a percentage the total amount of available memory
|
||||||
|
PERCENTAGE=60
|
||||||
|
|
||||||
|
# Specifies a static amount of RAM that should be used for
|
||||||
|
# the ZRAM devices, this is in MiB
|
||||||
|
#ALLOCATION=256
|
||||||
|
|
||||||
|
# Specifies the priority for the swap devices, see swapon(2)
|
||||||
|
# for more details.
|
||||||
|
#PRIORITY=100
|
@ -1,26 +0,0 @@
|
|||||||
# /etc/dphys-swapfile - user settings for dphys-swapfile package
|
|
||||||
# author Neil Franklin, last modification 2010.05.05
|
|
||||||
# copyright ETH Zuerich Physics Departement
|
|
||||||
# use under either modified/non-advertising BSD or GPL license
|
|
||||||
|
|
||||||
# this file is sourced with . so full normal sh syntax applies
|
|
||||||
|
|
||||||
# the default settings are added as commented out CONF_*=* lines
|
|
||||||
|
|
||||||
|
|
||||||
# where we want the swapfile to be, this is the default
|
|
||||||
#CONF_SWAPFILE=/var/swap
|
|
||||||
|
|
||||||
# set size to absolute value, leaving empty (default) then uses computed value
|
|
||||||
# you most likely don't want this, unless you have an special disk situation
|
|
||||||
CONF_SWAPSIZE=2048
|
|
||||||
|
|
||||||
# set size to computed value, this times RAM size, dynamically adapts,
|
|
||||||
# guarantees that there is enough swap without wasting disk space on excess
|
|
||||||
#CONF_SWAPFACTOR=2
|
|
||||||
|
|
||||||
# restrict size (computed and absolute!) to maximally this limit
|
|
||||||
# can be set to empty for no limit, but beware of filled partitions!
|
|
||||||
# this is/was a (outdated?) 32bit kernel limit (in MBytes), do not overrun it
|
|
||||||
# but is also sensible on 64bit to prevent filling /var or even / partition
|
|
||||||
#CONF_MAXSWAP=2048
|
|
@ -56,7 +56,7 @@ stop_monitor_interface() {
|
|||||||
ifconfig wlan0 up
|
ifconfig wlan0 up
|
||||||
}
|
}
|
||||||
|
|
||||||
# returns 0 if the specificed network interface is up
|
# returns 0 if the specified network interface is up
|
||||||
is_interface_up() {
|
is_interface_up() {
|
||||||
if grep -qi 'up' /sys/class/net/"$1"/operstate; then
|
if grep -qi 'up' /sys/class/net/"$1"/operstate; then
|
||||||
return 0
|
return 0
|
||||||
|
@ -23,21 +23,17 @@
|
|||||||
services:
|
services:
|
||||||
enable:
|
enable:
|
||||||
- bettercap.service
|
- bettercap.service
|
||||||
- dphys-swapfile.service
|
|
||||||
- fstrim.timer
|
- fstrim.timer
|
||||||
- pwnagotchi.service
|
- pwnagotchi.service
|
||||||
- pwngrid-peer.service
|
- pwngrid-peer.service
|
||||||
|
- zramswap.service
|
||||||
disable:
|
disable:
|
||||||
- apt-daily-upgrade.service
|
- apt-daily-upgrade.service
|
||||||
- apt-daily-upgrade.timer
|
- apt-daily-upgrade.timer
|
||||||
- apt-daily.service
|
- apt-daily.service
|
||||||
- apt-daily.timer
|
- apt-daily.timer
|
||||||
- avahi-daemon.service
|
|
||||||
- avahi-daemon.socket
|
|
||||||
- bluetooth.service
|
- bluetooth.service
|
||||||
- ifup@wlan0.service
|
- ifup@wlan0.service
|
||||||
- triggerhappy.service
|
|
||||||
- wpa_supplicant.service
|
|
||||||
packages:
|
packages:
|
||||||
caplets:
|
caplets:
|
||||||
source: "https://github.com/jayofelony/caplets.git"
|
source: "https://github.com/jayofelony/caplets.git"
|
||||||
@ -61,6 +57,7 @@
|
|||||||
- libpcap0.8-dbg
|
- libpcap0.8-dbg
|
||||||
remove:
|
remove:
|
||||||
- avahi-daemon
|
- avahi-daemon
|
||||||
|
- dhpys-swapfile
|
||||||
- nfs-common
|
- nfs-common
|
||||||
- triggerhappy
|
- triggerhappy
|
||||||
- wpasupplicant
|
- wpasupplicant
|
||||||
@ -73,7 +70,6 @@
|
|||||||
- build-essential
|
- build-essential
|
||||||
- curl
|
- curl
|
||||||
- dkms
|
- dkms
|
||||||
- dphys-swapfile
|
|
||||||
- fbi
|
- fbi
|
||||||
- flex
|
- flex
|
||||||
- fonts-dejavu
|
- fonts-dejavu
|
||||||
@ -161,17 +157,62 @@
|
|||||||
- wl
|
- wl
|
||||||
- xxd
|
- xxd
|
||||||
- zlib1g-dev
|
- zlib1g-dev
|
||||||
|
- zram-tools
|
||||||
environment:
|
environment:
|
||||||
ARCHFLAGS: "-arch aarch64"
|
ARCHFLAGS: "-arch aarch64"
|
||||||
QEMU_UNAME: "{{ kernel.full }}"
|
QEMU_UNAME: "{{ kernel.full }}"
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
|
# First we install and remove unnecessary packages
|
||||||
|
- name: install packages
|
||||||
|
apt:
|
||||||
|
name: "{{ packages.apt.install }}"
|
||||||
|
state: present
|
||||||
|
update_cache: yes
|
||||||
|
install_recommends: false
|
||||||
|
|
||||||
|
- name: remove unnecessary apt packages
|
||||||
|
apt:
|
||||||
|
name: "{{ packages.apt.remove }}"
|
||||||
|
state: absent
|
||||||
|
purge: yes
|
||||||
|
register: removed
|
||||||
|
|
||||||
|
# Now we set up /boot/firmware
|
||||||
- name: Create pi user
|
- name: Create pi user
|
||||||
copy:
|
copy:
|
||||||
dest: /boot/firmware/userconf
|
dest: /boot/firmware/userconf
|
||||||
content: |
|
content: |
|
||||||
pi:$6$3jNr0GA9KIyt4hmM$efeVIopdMQ8DGgEPCWWlbx3mJJNAYci1lEXGdlky0xPyjqwKNbwTL5SrCcpb4144C4IvzWjn7Iv.QjqmU7iyT/
|
pi:$6$3jNr0GA9KIyt4hmM$efeVIopdMQ8DGgEPCWWlbx3mJJNAYci1lEXGdlky0xPyjqwKNbwTL5SrCcpb4144C4IvzWjn7Iv.QjqmU7iyT/
|
||||||
|
|
||||||
|
- name: enable ssh on boot
|
||||||
|
file:
|
||||||
|
path: /boot/firmware/ssh
|
||||||
|
state: touch
|
||||||
|
|
||||||
|
- name: adjust /boot/firmware/config.txt
|
||||||
|
lineinfile:
|
||||||
|
dest: /boot/firmware/config.txt
|
||||||
|
insertafter: EOF
|
||||||
|
line: '{{ item }}'
|
||||||
|
with_items: "{{ system.boot_options }}"
|
||||||
|
|
||||||
|
- name: change root partition
|
||||||
|
replace:
|
||||||
|
dest: /boot/firmware/cmdline.txt
|
||||||
|
backup: no
|
||||||
|
regexp: "root=PARTUUID=[a-zA-Z0-9\\-]+"
|
||||||
|
replace: "root=/dev/mmcblk0p2"
|
||||||
|
|
||||||
|
- name: configure /boot/firmware/cmdline.txt
|
||||||
|
lineinfile:
|
||||||
|
path: /boot/firmware/cmdline.txt
|
||||||
|
backrefs: True
|
||||||
|
state: present
|
||||||
|
backup: no
|
||||||
|
regexp: '(.*)$'
|
||||||
|
line: '\1 modules-load=dwc2,g_ether'
|
||||||
|
|
||||||
- name: change hostname
|
- name: change hostname
|
||||||
lineinfile:
|
lineinfile:
|
||||||
dest: /etc/hostname
|
dest: /etc/hostname
|
||||||
@ -189,6 +230,7 @@
|
|||||||
state: present
|
state: present
|
||||||
when: hostname.changed
|
when: hostname.changed
|
||||||
|
|
||||||
|
# Now we disable sap and a2dp, we don't use them on rpi
|
||||||
- name: disable sap plugin for bluetooth.service
|
- name: disable sap plugin for bluetooth.service
|
||||||
lineinfile:
|
lineinfile:
|
||||||
dest: /lib/systemd/system/bluetooth.service
|
dest: /lib/systemd/system/bluetooth.service
|
||||||
@ -196,19 +238,6 @@
|
|||||||
line: 'ExecStart=/usr/libexec/bluetooth/bluetoothd --noplugin=sap,a2dp'
|
line: 'ExecStart=/usr/libexec/bluetooth/bluetoothd --noplugin=sap,a2dp'
|
||||||
state: present
|
state: present
|
||||||
|
|
||||||
- name: configure dphys-swapfile
|
|
||||||
lineinfile:
|
|
||||||
path: /etc/dphys-swapfile
|
|
||||||
regexp: "^CONF_SWAPSIZE=.*$"
|
|
||||||
line: "CONF_SWAPSIZE=2048"
|
|
||||||
|
|
||||||
- name: install packages
|
|
||||||
apt:
|
|
||||||
name: "{{ packages.apt.install }}"
|
|
||||||
state: present
|
|
||||||
update_cache: yes
|
|
||||||
install_recommends: false
|
|
||||||
|
|
||||||
###########################################
|
###########################################
|
||||||
#
|
#
|
||||||
# libpcap v1.9 - build from source
|
# libpcap v1.9 - build from source
|
||||||
@ -477,41 +506,6 @@
|
|||||||
state: absent
|
state: absent
|
||||||
path: /etc/update-motd.d/10-uname
|
path: /etc/update-motd.d/10-uname
|
||||||
|
|
||||||
- name: enable ssh on boot
|
|
||||||
file:
|
|
||||||
path: /boot/firmware/ssh
|
|
||||||
state: touch
|
|
||||||
|
|
||||||
- name: adjust /boot/config.txt
|
|
||||||
lineinfile:
|
|
||||||
dest: /boot/firmware/config.txt
|
|
||||||
insertafter: EOF
|
|
||||||
line: '{{ item }}'
|
|
||||||
with_items: "{{ system.boot_options }}"
|
|
||||||
|
|
||||||
- name: adjust /etc/modules
|
|
||||||
lineinfile:
|
|
||||||
dest: /etc/modules
|
|
||||||
insertafter: EOF
|
|
||||||
line: '{{ item }}'
|
|
||||||
with_items: "{{ system.modules }}"
|
|
||||||
|
|
||||||
- name: change root partition
|
|
||||||
replace:
|
|
||||||
dest: /boot/firmware/cmdline.txt
|
|
||||||
backup: no
|
|
||||||
regexp: "root=PARTUUID=[a-zA-Z0-9\\-]+"
|
|
||||||
replace: "root=/dev/mmcblk0p2"
|
|
||||||
|
|
||||||
- name: configure /boot/cmdline.txt
|
|
||||||
lineinfile:
|
|
||||||
path: /boot/cmdline.txt
|
|
||||||
backrefs: True
|
|
||||||
state: present
|
|
||||||
backup: no
|
|
||||||
regexp: '(.*)$'
|
|
||||||
line: '\1 modules-load=dwc2,g_ether'
|
|
||||||
|
|
||||||
- name: Add pwnlog alias
|
- name: Add pwnlog alias
|
||||||
lineinfile:
|
lineinfile:
|
||||||
dest: /home/pi/.bashrc
|
dest: /home/pi/.bashrc
|
||||||
@ -563,13 +557,6 @@
|
|||||||
group: pi
|
group: pi
|
||||||
recurse: true
|
recurse: true
|
||||||
|
|
||||||
- name: remove unnecessary apt packages
|
|
||||||
apt:
|
|
||||||
name: "{{ packages.apt.remove }}"
|
|
||||||
state: absent
|
|
||||||
purge: yes
|
|
||||||
register: removed
|
|
||||||
|
|
||||||
- name: clean apt cache
|
- name: clean apt cache
|
||||||
apt:
|
apt:
|
||||||
autoclean: true
|
autoclean: true
|
||||||
|
@ -1 +1 @@
|
|||||||
__version__ = '2.6.9'
|
__version__ = '2.7.5'
|
||||||
|
@ -184,13 +184,15 @@ class Agent(Client, Automata, AsyncAdvertiser, AsyncTrainer):
|
|||||||
for ap in s['wifi']['aps']:
|
for ap in s['wifi']['aps']:
|
||||||
if ap['encryption'] == '' or ap['encryption'] == 'OPEN':
|
if ap['encryption'] == '' or ap['encryption'] == 'OPEN':
|
||||||
continue
|
continue
|
||||||
|
elif ap['hostname'] in whitelist or ap['mac'][:8].lower() in whitelist:
|
||||||
|
continue
|
||||||
elif ap['hostname'] not in whitelist \
|
elif ap['hostname'] not in whitelist \
|
||||||
and ap['mac'].lower() not in whitelist \
|
and ap['mac'].lower() not in whitelist \
|
||||||
and ap['mac'][:8].lower() not in whitelist:
|
and ap['mac'][:8].lower() not in whitelist:
|
||||||
if self._filter_included(ap):
|
if self._filter_included(ap):
|
||||||
aps.append(ap)
|
aps.append(ap)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.exception("Error while getting acces points (%s)", e)
|
logging.exception("Error while getting access points (%s)", e)
|
||||||
|
|
||||||
aps.sort(key=lambda ap: ap['channel'])
|
aps.sort(key=lambda ap: ap['channel'])
|
||||||
return self.set_access_points(aps)
|
return self.set_access_points(aps)
|
||||||
|
@ -59,7 +59,8 @@ def load(config, agent, epoch, from_disk=True):
|
|||||||
|
|
||||||
return a2c
|
return a2c
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.exception("[AI] error while starting AI (%s)", e)
|
logging.info("[AI] Error while starting AI")
|
||||||
|
logging.debug("[AI] error while starting AI (%s)", e)
|
||||||
logging.info("[AI] Deleting brain and restarting.")
|
logging.info("[AI] Deleting brain and restarting.")
|
||||||
os.system("rm /root/brain.nn && service pwnagotchi restart")
|
os.system("rm /root/brain.nn && service pwnagotchi restart")
|
||||||
|
|
||||||
|
Binary file not shown.
@ -10,7 +10,7 @@ msgstr ""
|
|||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2023-11-16 21:51+0100\n"
|
"POT-Creation-Date: 2023-11-16 21:51+0100\n"
|
||||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
"Last-Translator: neo0oen \n"
|
||||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||||
"Language: Hebrew\n"
|
"Language: Hebrew\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
@ -18,240 +18,242 @@ msgstr ""
|
|||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
|
||||||
msgid "ZzzzZZzzzzZzzz"
|
msgid "ZzzzZZzzzzZzzz"
|
||||||
msgstr ""
|
msgstr "ZzzzZZzzzzZzzz"
|
||||||
|
|
||||||
msgid "Hi, I'm Pwnagotchi! Starting ..."
|
msgid "Hi, I'm Pwnagotchi! Starting ..."
|
||||||
msgstr ""
|
msgstr "שלום, אני פוונגוטצ'י! מתחיל . . ."
|
||||||
|
|
||||||
msgid "New day, new hunt, new pwns!"
|
msgid "New day, new hunt, new pwns!"
|
||||||
msgstr ""
|
msgstr "יום חדש, ציד חדש, ניצחונות חדשים !"
|
||||||
|
|
||||||
msgid "Hack the Planet!"
|
msgid "Hack the Planet!"
|
||||||
msgstr ""
|
msgstr "פרוץ לכדור הארץ !"
|
||||||
|
|
||||||
msgid "AI ready."
|
msgid "AI ready."
|
||||||
msgstr ""
|
msgstr "הבינה המלאחות'ית מוכנה"
|
||||||
|
|
||||||
msgid "The neural network is ready."
|
msgid "The neural network is ready."
|
||||||
msgstr ""
|
msgstr "הרשת הניורולוגית העצבית מוכנה."
|
||||||
|
|
||||||
msgid "Generating keys, do not turn off ..."
|
msgid "Generating keys, do not turn off ..."
|
||||||
msgstr ""
|
msgstr "יוצר מפתחות, בבקשה אל תכבה אותי"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey, channel {channel} is free! Your AP will say thanks."
|
msgid "Hey, channel {channel} is free! Your AP will say thanks."
|
||||||
msgstr ""
|
msgstr "היי , הערוץ {channel}פנוי ,נקודת הגישה שלך אומרת תודה "
|
||||||
|
|
||||||
msgid "Reading last session logs ..."
|
msgid "Reading last session logs ..."
|
||||||
msgstr ""
|
msgstr ". . . קורא לוגים מאז ההפעלה האחרונה"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Read {lines_so_far} log lines so far ..."
|
msgid "Read {lines_so_far} log lines so far ..."
|
||||||
msgstr ""
|
msgstr "קראתי {lines_so_far} שורות מהלוג עד עכשיו . . ."
|
||||||
|
|
||||||
msgid "I'm bored ..."
|
msgid "I'm bored ..."
|
||||||
msgstr ""
|
msgstr "אני משועמם . . ."
|
||||||
|
|
||||||
msgid "Let's go for a walk!"
|
msgid "Let's go for a walk!"
|
||||||
msgstr ""
|
msgstr "בוא ניצא לטיול! "
|
||||||
|
|
||||||
msgid "This is the best day of my life!"
|
msgid "This is the best day of my life!"
|
||||||
msgstr ""
|
msgstr "זה היום הכי טוב בחיי!"
|
||||||
|
|
||||||
msgid "Shitty day :/"
|
msgid "Shitty day :/"
|
||||||
msgstr ""
|
msgstr "יום מחורבן :/"
|
||||||
|
|
||||||
msgid "I'm extremely bored ..."
|
msgid "I'm extremely bored ..."
|
||||||
msgstr ""
|
msgstr "אני משועמם בטירוף . . ."
|
||||||
|
|
||||||
msgid "I'm very sad ..."
|
msgid "I'm very sad ..."
|
||||||
msgstr ""
|
msgstr "אני עצוב נורא . . ."
|
||||||
|
|
||||||
msgid "I'm sad"
|
msgid "I'm sad"
|
||||||
msgstr ""
|
msgstr "אני עצוב"
|
||||||
|
|
||||||
msgid "Leave me alone ..."
|
msgid "Leave me alone ..."
|
||||||
msgstr ""
|
msgstr "עזוב אותי בשקט . . ."
|
||||||
|
|
||||||
msgid "I'm mad at you!"
|
msgid "I'm mad at you!"
|
||||||
msgstr ""
|
msgstr "אני כועס עלייך!"
|
||||||
|
|
||||||
msgid "I'm living the life!"
|
msgid "I'm living the life!"
|
||||||
msgstr ""
|
msgstr "אני חי את החיים!"
|
||||||
|
|
||||||
msgid "I pwn therefore I am."
|
msgid "I pwn therefore I am."
|
||||||
msgstr ""
|
msgstr "אני שולט לכן אני כאן."
|
||||||
|
|
||||||
msgid "So many networks!!!"
|
msgid "So many networks!!!"
|
||||||
msgstr ""
|
msgstr "כל כך הרבה רשתות! ! !"
|
||||||
|
|
||||||
msgid "I'm having so much fun!"
|
msgid "I'm having so much fun!"
|
||||||
msgstr ""
|
msgstr "אני כל כך נהנה ועושה הרבה כיף!"
|
||||||
|
|
||||||
msgid "My crime is that of curiosity ..."
|
msgid "My crime is that of curiosity ..."
|
||||||
msgstr ""
|
msgstr "הפשע שלי הוא הסקרנות . . ."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hello {name}! Nice to meet you."
|
msgid "Hello {name}! Nice to meet you."
|
||||||
msgstr ""
|
msgstr "שלום {name}! נעים להכיר אותך."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Yo {name}! Sup?"
|
msgid "Yo {name}! Sup?"
|
||||||
msgstr ""
|
msgstr "יו {name}! מה קורה?"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey {name} how are you doing?"
|
msgid "Hey {name} how are you doing?"
|
||||||
msgstr ""
|
msgstr "היי {name} איך אתה מסתדר?"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Unit {name} is nearby!"
|
msgid "Unit {name} is nearby!"
|
||||||
msgstr ""
|
msgstr "יחידה {name} בקרבת מקום!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Uhm ... goodbye {name}"
|
msgid "Uhm ... goodbye {name}"
|
||||||
msgstr ""
|
msgstr "אוף ... להתראות {name}"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "{name} is gone ..."
|
msgid "{name} is gone ..."
|
||||||
msgstr ""
|
msgstr "{name} נעלם ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Whoops ... {name} is gone."
|
msgid "Whoops ... {name} is gone."
|
||||||
msgstr ""
|
msgstr "אופס ... {name} נעלם ."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "{name} missed!"
|
msgid "{name} missed!"
|
||||||
msgstr ""
|
msgstr "{name} התפספס!"
|
||||||
|
|
||||||
msgid "Missed!"
|
msgid "Missed!"
|
||||||
msgstr ""
|
msgstr "התפספס!"
|
||||||
|
|
||||||
msgid "Good friends are a blessing!"
|
msgid "Good friends are a blessing!"
|
||||||
msgstr ""
|
msgstr "חברים טובים זאת ברכה!"
|
||||||
|
|
||||||
msgid "I love my friends!"
|
msgid "I love my friends!"
|
||||||
msgstr ""
|
msgstr "אני אוהב את החברים שלי!"
|
||||||
|
|
||||||
msgid "Nobody wants to play with me ..."
|
msgid "Nobody wants to play with me ..."
|
||||||
msgstr ""
|
msgstr "אף אחד לא רוצה לשחק אית . . ."
|
||||||
|
|
||||||
msgid "I feel so alone ..."
|
msgid "I feel so alone ..."
|
||||||
msgstr ""
|
msgstr "אני מרגיש כל כך לבד . . ."
|
||||||
|
|
||||||
msgid "Where's everybody?!"
|
msgid "Where's everybody?!"
|
||||||
msgstr ""
|
msgstr "לאיפה כולם נעלמו?!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Napping for {secs}s ..."
|
msgid "Napping for {secs}s ..."
|
||||||
msgstr ""
|
msgstr "מנמנם למשך {secs}שניות ..."
|
||||||
|
|
||||||
msgid "Zzzzz"
|
msgid "Zzzzz"
|
||||||
msgstr ""
|
msgstr "Zzzzz"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "ZzzZzzz ({secs}s)"
|
msgid "ZzzZzzz ({secs}s)"
|
||||||
msgstr ""
|
msgstr "ZzzZzzz ({secs}s)"
|
||||||
|
|
||||||
msgid "Good night."
|
msgid "Good night."
|
||||||
msgstr ""
|
msgstr "לילה טוב"
|
||||||
|
|
||||||
msgid "Zzz"
|
msgid "Zzz"
|
||||||
msgstr ""
|
msgstr "Zzz"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Waiting for {secs}s ..."
|
msgid "Waiting for {secs}s ..."
|
||||||
msgstr ""
|
msgstr "מחכה ל {secs}שניות ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Looking around ({secs}s)"
|
msgid "Looking around ({secs}s)"
|
||||||
msgstr ""
|
msgstr "מסתכל מסביב ({secs}שניות)"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey {what} let's be friends!"
|
msgid "Hey {what} let's be friends!"
|
||||||
msgstr ""
|
msgstr "היי {what} בוא נהיה חברים!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Associating to {what}"
|
msgid "Associating to {what}"
|
||||||
msgstr ""
|
msgstr "מקשר את עצמי אל {what}"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Yo {what}!"
|
msgid "Yo {what}!"
|
||||||
msgstr ""
|
msgstr "יו יו {what}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Just decided that {mac} needs no WiFi!"
|
msgid "Just decided that {mac} needs no WiFi!"
|
||||||
msgstr ""
|
msgstr "בדיוק החלטתי ש {mac} לא צריך חיבור אלחוטי"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Deauthenticating {mac}"
|
msgid "Deauthenticating {mac}"
|
||||||
msgstr ""
|
msgstr "מסכל את האימות של {mac}"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Kickbanning {mac}!"
|
msgid "Kickbanning {mac}!"
|
||||||
msgstr ""
|
msgstr "מוציא איסור על/מעיף {mac}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Cool, we got {num} new handshake{plural}!"
|
msgid "Cool, we got {num} new handshake{plural}!"
|
||||||
msgstr ""
|
msgstr "מגניב!, הצלחנו להשיג {num} לחיצות ידיים חדשות{plural}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "You have {count} new message{plural}!"
|
msgid "You have {count} new message{plural}!"
|
||||||
msgstr ""
|
msgstr "יש לך {count} הודעות חדשות{plural}!"
|
||||||
|
|
||||||
msgid "Oops, something went wrong ... Rebooting ..."
|
msgid "Oops, something went wrong ... Rebooting ..."
|
||||||
msgstr ""
|
msgstr "אופס!, משהו לא קרה כשורה ... מאתחל ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Uploading data to {to} ..."
|
msgid "Uploading data to {to} ..."
|
||||||
msgstr ""
|
msgstr "מעלה נתונים ל {to} ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Downloading from {name} ..."
|
msgid "Downloading from {name} ..."
|
||||||
msgstr ""
|
msgstr "מוריד נתונים מ {name} ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Kicked {num} stations\n"
|
msgid "Kicked {num} stations\n"
|
||||||
msgstr ""
|
msgstr "העפתי {num} תחנות\n"
|
||||||
|
|
||||||
msgid "Made >999 new friends\n"
|
msgid "Made >999 new friends\n"
|
||||||
msgstr ""
|
msgstr "הכרתי >999 חברים חדשים\n"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Made {num} new friends\n"
|
msgid "Made {num} new friends\n"
|
||||||
msgstr ""
|
msgstr "הכרתי {num} חברים חדשים\n"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Got {num} handshakes\n"
|
msgid "Got {num} handshakes\n"
|
||||||
msgstr ""
|
msgstr "השגתי {num} לחיצות ידיים\n"
|
||||||
|
|
||||||
msgid "Met 1 peer"
|
msgid "Met 1 peer"
|
||||||
msgstr ""
|
msgstr "פגשתי שותף 1 "
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Met {num} peers"
|
msgid "Met {num} peers"
|
||||||
msgstr ""
|
msgstr "פגשתי {num} שותפים"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid ""
|
msgid ""
|
||||||
"I've been pwning for {duration} and kicked {deauthed} clients! I've also met "
|
"I've been pwning for {duration} and kicked {deauthed} clients! I've also met "
|
||||||
"{associated} new friends and ate {handshakes} handshakes! #pwnagotchi "
|
"{associated} new friends and ate {handshakes} handshakes! #pwnagotchi "
|
||||||
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
||||||
msgstr ""
|
msgstr "אני שלטתי במשך {duration} והצלחתי להעיף {deauthed} משתמשים! בדרך גם פגשתי "
|
||||||
|
"{associated} חברים חדשים ואכלתי {handshakes} לחיצות ידיים #pwnagotchi "
|
||||||
|
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
||||||
|
|
||||||
msgid "hours"
|
msgid "hours"
|
||||||
msgstr ""
|
msgstr "שעות"
|
||||||
|
|
||||||
msgid "minutes"
|
msgid "minutes"
|
||||||
msgstr ""
|
msgstr "דקות"
|
||||||
|
|
||||||
msgid "seconds"
|
msgid "seconds"
|
||||||
msgstr ""
|
msgstr "שניות"
|
||||||
|
|
||||||
msgid "hour"
|
msgid "hour"
|
||||||
msgstr ""
|
msgstr "שעה"
|
||||||
|
|
||||||
msgid "minute"
|
msgid "minute"
|
||||||
msgstr ""
|
msgstr "דקה"
|
||||||
|
|
||||||
msgid "second"
|
msgid "second"
|
||||||
msgstr ""
|
msgstr "שניה"
|
BIN
pwnagotchi/locale/pt-br/LC_MESSAGES/voice.mo
Normal file
BIN
pwnagotchi/locale/pt-br/LC_MESSAGES/voice.mo
Normal file
Binary file not shown.
@ -10,98 +10,98 @@ msgstr ""
|
|||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2023-11-17 15:46+0100\n"
|
"POT-Creation-Date: 2023-11-17 15:46+0100\n"
|
||||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
"Last-Translator: Foxy <EMAIL@ADDRESS>\n"
|
||||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||||
"Language: \n"
|
"Language: Portuguese (Brazil)\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=CHARSET\n"
|
"Content-Type: text/plain; charset=CHARSET\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
|
||||||
msgid "ZzzzZZzzzzZzzz"
|
msgid "ZzzzZZzzzzZzzz"
|
||||||
msgstr ""
|
msgstr "ZzzzZZzzzzZzzz"
|
||||||
|
|
||||||
msgid "Hi, I'm Pwnagotchi! Starting ..."
|
msgid "Hi, I'm Pwnagotchi! Starting ..."
|
||||||
msgstr ""
|
msgstr "Olá, Eu sou Pwnagotchi! Iniciando ..."
|
||||||
|
|
||||||
msgid "New day, new hunt, new pwns!"
|
msgid "New day, new hunt, new pwns!"
|
||||||
msgstr ""
|
msgstr "Um novo dia, Uma nova caça e novos pwns!"
|
||||||
|
|
||||||
msgid "Hack the Planet!"
|
msgid "Hack the Planet!"
|
||||||
msgstr ""
|
msgstr "Burle o Planeta!"
|
||||||
|
|
||||||
msgid "AI ready."
|
msgid "AI ready."
|
||||||
msgstr ""
|
msgstr "IA pronta."
|
||||||
|
|
||||||
msgid "The neural network is ready."
|
msgid "The neural network is ready."
|
||||||
msgstr ""
|
msgstr "A rede neural está pronta."
|
||||||
|
|
||||||
msgid "Generating keys, do not turn off ..."
|
msgid "Generating keys, do not turn off ..."
|
||||||
msgstr ""
|
msgstr "Criando chaves, não desligue o sistema ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey, channel {channel} is free! Your AP will say thanks."
|
msgid "Hey, channel {channel} is free! Your AP will say thanks."
|
||||||
msgstr ""
|
msgstr "Ei, canal {channel} está livre! Seu AP vai agradecer."
|
||||||
|
|
||||||
msgid "Reading last session logs ..."
|
msgid "Reading last session logs ..."
|
||||||
msgstr ""
|
msgstr "Lendo os logs da ultima sessão"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Read {lines_so_far} log lines so far ..."
|
msgid "Read {lines_so_far} log lines so far ..."
|
||||||
msgstr ""
|
msgstr "Leia {lines_so_far} linha de logs até agora ..."
|
||||||
|
|
||||||
msgid "I'm bored ..."
|
msgid "I'm bored ..."
|
||||||
msgstr ""
|
msgstr "Eu estou entediado ..."
|
||||||
|
|
||||||
msgid "Let's go for a walk!"
|
msgid "Let's go for a walk!"
|
||||||
msgstr ""
|
msgstr "Vamos ir numa caminhada!"
|
||||||
|
|
||||||
msgid "This is the best day of my life!"
|
msgid "This is the best day of my life!"
|
||||||
msgstr ""
|
msgstr "Esse é o melhor dia da minha vida!"
|
||||||
|
|
||||||
msgid "Shitty day :/"
|
msgid "Shitty day :/"
|
||||||
msgstr ""
|
msgstr "Dia ruim :/"
|
||||||
|
|
||||||
msgid "I'm extremely bored ..."
|
msgid "I'm extremely bored ..."
|
||||||
msgstr ""
|
msgstr "Eu estou extremamente entediado ..."
|
||||||
|
|
||||||
msgid "I'm very sad ..."
|
msgid "I'm very sad ..."
|
||||||
msgstr ""
|
msgstr "Eu estou muito triste ..."
|
||||||
|
|
||||||
msgid "I'm sad"
|
msgid "I'm sad"
|
||||||
msgstr ""
|
msgstr "Eu estou triste"
|
||||||
|
|
||||||
msgid "Leave me alone ..."
|
msgid "Leave me alone ..."
|
||||||
msgstr ""
|
msgstr "Me deixe em paz ..."
|
||||||
|
|
||||||
msgid "I'm mad at you!"
|
msgid "I'm mad at you!"
|
||||||
msgstr ""
|
msgstr "Eu estou bravo com você!"
|
||||||
|
|
||||||
msgid "I'm living the life!"
|
msgid "I'm living the life!"
|
||||||
msgstr ""
|
msgstr "Eu estou vivendo a vida!"
|
||||||
|
|
||||||
msgid "I pwn therefore I am."
|
msgid "I pwn therefore I am."
|
||||||
msgstr ""
|
msgstr "Eu pwn então Eu sou."
|
||||||
|
|
||||||
msgid "So many networks!!!"
|
msgid "So many networks!!!"
|
||||||
msgstr ""
|
msgstr "Tantas redes!!!"
|
||||||
|
|
||||||
msgid "I'm having so much fun!"
|
msgid "I'm having so much fun!"
|
||||||
msgstr ""
|
msgstr "Eu estou tendo muita diversão"
|
||||||
|
|
||||||
msgid "My crime is that of curiosity ..."
|
msgid "My crime is that of curiosity ..."
|
||||||
msgstr ""
|
msgstr "Meu crime é de curiosidade ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hello {name}! Nice to meet you."
|
msgid "Hello {name}! Nice to meet you."
|
||||||
msgstr ""
|
msgstr "Olá {name}! É bom em conhecê-lo"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Yo {name}! Sup?"
|
msgid "Yo {name}! Sup?"
|
||||||
msgstr ""
|
msgstr "Ei {name}! Como vai?"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey {name} how are you doing?"
|
msgid "Hey {name} how are you doing?"
|
||||||
msgstr ""
|
msgstr "Ei {name} como você está indo?"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Unit {name} is nearby!"
|
msgid "Unit {name} is nearby!"
|
||||||
@ -121,115 +121,115 @@ msgstr ""
|
|||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "{name} missed!"
|
msgid "{name} missed!"
|
||||||
msgstr ""
|
msgstr "{name} errou!"
|
||||||
|
|
||||||
msgid "Missed!"
|
msgid "Missed!"
|
||||||
msgstr ""
|
msgstr "Errei!"
|
||||||
|
|
||||||
msgid "Good friends are a blessing!"
|
msgid "Good friends are a blessing!"
|
||||||
msgstr ""
|
msgstr "Bom amigos são uma bensão!"
|
||||||
|
|
||||||
msgid "I love my friends!"
|
msgid "I love my friends!"
|
||||||
msgstr ""
|
msgstr "Eu amo meus amigos!"
|
||||||
|
|
||||||
msgid "Nobody wants to play with me ..."
|
msgid "Nobody wants to play with me ..."
|
||||||
msgstr ""
|
msgstr "Ninguém quer brincar comigo ..."
|
||||||
|
|
||||||
msgid "I feel so alone ..."
|
msgid "I feel so alone ..."
|
||||||
msgstr ""
|
msgstr "Estou me sentindo sozinho"
|
||||||
|
|
||||||
msgid "Where's everybody?!"
|
msgid "Where's everybody?!"
|
||||||
msgstr ""
|
msgstr "Onde está todo mundo?!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Napping for {secs}s ..."
|
msgid "Napping for {secs}s ..."
|
||||||
msgstr ""
|
msgstr "Tirando uma soneca por {secs}s ..."
|
||||||
|
|
||||||
msgid "Zzzzz"
|
msgid "Zzzzz"
|
||||||
msgstr ""
|
msgstr "Zzzzz"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "ZzzZzzz ({secs}s)"
|
msgid "ZzzZzzz ({secs}s)"
|
||||||
msgstr ""
|
msgstr "ZzzZzzz ({secs}s)"
|
||||||
|
|
||||||
msgid "Good night."
|
msgid "Good night."
|
||||||
msgstr ""
|
msgstr "Boa noite."
|
||||||
|
|
||||||
msgid "Zzz"
|
msgid "Zzz"
|
||||||
msgstr ""
|
msgstr "Zzz"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Waiting for {secs}s ..."
|
msgid "Waiting for {secs}s ..."
|
||||||
msgstr ""
|
msgstr "Esperando por {secs}s ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Looking around ({secs}s)"
|
msgid "Looking around ({secs}s)"
|
||||||
msgstr ""
|
msgstr "Olhando por volta ({secs}s)"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Hey {what} let's be friends!"
|
msgid "Hey {what} let's be friends!"
|
||||||
msgstr ""
|
msgstr "Ei {what} vamos ser amigos!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Associating to {what}"
|
msgid "Associating to {what}"
|
||||||
msgstr ""
|
msgstr "Associando para {what}"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Yo {what}!"
|
msgid "Yo {what}!"
|
||||||
msgstr ""
|
msgstr "Ei {what}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Just decided that {mac} needs no WiFi!"
|
msgid "Just decided that {mac} needs no WiFi!"
|
||||||
msgstr ""
|
msgstr "Apenas decidindo que {mac} não precisa de WiFi!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Deauthenticating {mac}"
|
msgid "Deauthenticating {mac}"
|
||||||
msgstr ""
|
msgstr "Desautenticando {mac}"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Kickbanning {mac}!"
|
msgid "Kickbanning {mac}!"
|
||||||
msgstr ""
|
msgstr "Banindo {mac}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Cool, we got {num} new handshake{plural}!"
|
msgid "Cool, we got {num} new handshake{plural}!"
|
||||||
msgstr ""
|
msgstr "Legal, conseguimos {num} novos handshake{plural}!"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "You have {count} new message{plural}!"
|
msgid "You have {count} new message{plural}!"
|
||||||
msgstr ""
|
msgstr "Você tem {count} novas messagem{plural}!"
|
||||||
|
|
||||||
msgid "Oops, something went wrong ... Rebooting ..."
|
msgid "Oops, something went wrong ... Rebooting ..."
|
||||||
msgstr ""
|
msgstr "Oops, algo deu errado ... Reiniciando ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Uploading data to {to} ..."
|
msgid "Uploading data to {to} ..."
|
||||||
msgstr ""
|
msgstr "Enviando dados para {to} ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Downloading from {name} ..."
|
msgid "Downloading from {name} ..."
|
||||||
msgstr ""
|
msgstr "Instalando para {name} ..."
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Kicked {num} stations\n"
|
msgid "Kicked {num} stations\n"
|
||||||
msgstr ""
|
msgstr "Expulsei {num} estações\n"
|
||||||
|
|
||||||
msgid "Made >999 new friends\n"
|
msgid "Made >999 new friends\n"
|
||||||
msgstr ""
|
msgstr "Fiz >999 novos amigos\n"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Made {num} new friends\n"
|
msgid "Made {num} new friends\n"
|
||||||
msgstr ""
|
msgstr "Fiz {num} novos amigos\n"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Got {num} handshakes\n"
|
msgid "Got {num} handshakes\n"
|
||||||
msgstr ""
|
msgstr "Peguei {num} handshakes\n"
|
||||||
|
|
||||||
msgid "Met 1 peer"
|
msgid "Met 1 peer"
|
||||||
msgstr ""
|
msgstr "Encontrei 1 pessoa"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid "Met {num} peers"
|
msgid "Met {num} peers"
|
||||||
msgstr ""
|
msgstr "Encontrei {num} pessoas"
|
||||||
|
|
||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgid ""
|
msgid ""
|
||||||
@ -237,21 +237,24 @@ msgid ""
|
|||||||
"{associated} new friends and ate {handshakes} handshakes! #pwnagotchi "
|
"{associated} new friends and ate {handshakes} handshakes! #pwnagotchi "
|
||||||
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
"Estou navegando há {duration} e expulsei {deauthed} clientes! Também conheci "
|
||||||
|
"{associamos} novos amigos e comi {handshakes} handshakes! #pwnagotchi "
|
||||||
|
"#pwnlog #pwnlife #hacktheplanet #skynet"
|
||||||
|
|
||||||
msgid "hours"
|
msgid "hours"
|
||||||
msgstr ""
|
msgstr "horas"
|
||||||
|
|
||||||
msgid "minutes"
|
msgid "minutes"
|
||||||
msgstr ""
|
msgstr "minutos"
|
||||||
|
|
||||||
msgid "seconds"
|
msgid "seconds"
|
||||||
msgstr ""
|
msgstr "segundos"
|
||||||
|
|
||||||
msgid "hour"
|
msgid "hour"
|
||||||
msgstr ""
|
msgstr "hora"
|
||||||
|
|
||||||
msgid "minute"
|
msgid "minute"
|
||||||
msgstr ""
|
msgstr "minuto"
|
||||||
|
|
||||||
msgid "second"
|
msgid "second"
|
||||||
msgstr ""
|
msgstr "segundo"
|
@ -212,7 +212,7 @@ class AutoUpdate(plugins.Plugin):
|
|||||||
if install(display, update):
|
if install(display, update):
|
||||||
num_installed += 1
|
num_installed += 1
|
||||||
else:
|
else:
|
||||||
prev_status = '%d new update%c available!' % (num_updates, 's' if num_updates > 1 else '')
|
prev_status = '%d new update%s available!' % (num_updates, 's' if num_updates > 1 else '')
|
||||||
|
|
||||||
logging.info("[update] done")
|
logging.info("[update] done")
|
||||||
|
|
||||||
|
@ -21,9 +21,6 @@ class FixServices(plugins.Plugin):
|
|||||||
__help__ = """
|
__help__ = """
|
||||||
Reload brcmfmac module when blindbug is detected, instead of rebooting. Adapted from WATCHDOG.
|
Reload brcmfmac module when blindbug is detected, instead of rebooting. Adapted from WATCHDOG.
|
||||||
"""
|
"""
|
||||||
__dependencies__ = {
|
|
||||||
'pip': ['scapy']
|
|
||||||
}
|
|
||||||
__defaults__ = {
|
__defaults__ = {
|
||||||
'enabled': True,
|
'enabled': True,
|
||||||
}
|
}
|
||||||
@ -46,8 +43,7 @@ class FixServices(plugins.Plugin):
|
|||||||
logging.info("[Fix_Services] plugin loaded.")
|
logging.info("[Fix_Services] plugin loaded.")
|
||||||
|
|
||||||
def on_ready(self, agent):
|
def on_ready(self, agent):
|
||||||
last_lines = ''.join(list(TextIOWrapper(subprocess.Popen(['journalctl', '-n10', '-k'],
|
last_lines = self.get_last_lines('journalctl', ['-n10', '-k'], 10)
|
||||||
stdout=subprocess.PIPE).stdout))[-10:])
|
|
||||||
try:
|
try:
|
||||||
cmd_output = subprocess.check_output("ip link show wlan0mon", shell=True)
|
cmd_output = subprocess.check_output("ip link show wlan0mon", shell=True)
|
||||||
logging.info("[Fix_Services ip link show wlan0mon]: %s" % repr(cmd_output))
|
logging.info("[Fix_Services ip link show wlan0mon]: %s" % repr(cmd_output))
|
||||||
@ -99,14 +95,23 @@ class FixServices(plugins.Plugin):
|
|||||||
logging.error("[Fix_Services]SYSLOG wifi.recon flip fail: %s" % err)
|
logging.error("[Fix_Services]SYSLOG wifi.recon flip fail: %s" % err)
|
||||||
self._tryTurningItOffAndOnAgain(agent)
|
self._tryTurningItOffAndOnAgain(agent)
|
||||||
|
|
||||||
|
def get_last_lines(self, command, args, n):
|
||||||
|
try:
|
||||||
|
process = subprocess.Popen([command] + args, stdout=subprocess.PIPE)
|
||||||
|
output = TextIOWrapper(process.stdout)
|
||||||
|
lines = output.readlines()
|
||||||
|
last_n_lines = ''.join(lines[-n:])
|
||||||
|
return last_n_lines
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error occurred: {e}")
|
||||||
|
return None
|
||||||
|
|
||||||
def on_epoch(self, agent, epoch, epoch_data):
|
def on_epoch(self, agent, epoch, epoch_data):
|
||||||
last_lines = ''.join(list(TextIOWrapper(subprocess.Popen(['journalctl', '-n10', '-k'],
|
|
||||||
stdout=subprocess.PIPE).stdout))[-10:])
|
last_lines = self.get_last_lines('journalctl', ['-n10', '-k'], 10)
|
||||||
other_last_lines = ''.join(list(TextIOWrapper(subprocess.Popen(['journalctl', '-n10'],
|
other_last_lines = self.get_last_lines('journalctl', ['-n10'], 10)
|
||||||
stdout=subprocess.PIPE).stdout))[-10:])
|
other_other_last_lines = self.get_last_lines('tail', ['-n10', '/home/pi/logs/pwnagotchi.log'], 10)
|
||||||
other_other_last_lines = ''.join(
|
|
||||||
list(TextIOWrapper(subprocess.Popen(['tail', '-n10', '/home/pi/logs/pwnagotchi.log'],
|
|
||||||
stdout=subprocess.PIPE).stdout))[-10:])
|
|
||||||
# don't check if we ran a reset recently
|
# don't check if we ran a reset recently
|
||||||
logging.debug("[Fix_Services]**** epoch")
|
logging.debug("[Fix_Services]**** epoch")
|
||||||
if time.time() - self.LASTTRY > 180:
|
if time.time() - self.LASTTRY > 180:
|
||||||
|
@ -14,19 +14,15 @@ import zipfile
|
|||||||
|
|
||||||
class GdriveSync(plugins.Plugin):
|
class GdriveSync(plugins.Plugin):
|
||||||
__author__ = '@jayofelony'
|
__author__ = '@jayofelony'
|
||||||
__version__ = '1.0'
|
__version__ = '1.2'
|
||||||
__license__ = 'GPL3'
|
__license__ = 'GPL3'
|
||||||
__description__ = 'A plugin to backup various pwnagotchi files and folders to Google Drive. Once every hour from loading plugin.'
|
__description__ = 'A plugin to backup various pwnagotchi files and folders to Google Drive. Once every hour from loading plugin.'
|
||||||
__dependencies__ = {
|
|
||||||
'pip': ['pydrive2']
|
|
||||||
}
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.options = dict()
|
self.options = dict()
|
||||||
self.lock = Lock()
|
self.lock = Lock()
|
||||||
self.internet = False
|
self.internet = False
|
||||||
self.ready = False
|
self.ready = False
|
||||||
self.drive = None
|
|
||||||
self.status = StatusFile('/root/.gdrive-backup')
|
self.status = StatusFile('/root/.gdrive-backup')
|
||||||
self.backup = True
|
self.backup = True
|
||||||
self.backupfiles = [
|
self.backupfiles = [
|
||||||
@ -38,12 +34,11 @@ class GdriveSync(plugins.Plugin):
|
|||||||
'/etc/pwnagotchi'
|
'/etc/pwnagotchi'
|
||||||
]
|
]
|
||||||
|
|
||||||
def on_loaded(self, agent):
|
def on_loaded(self):
|
||||||
"""
|
"""
|
||||||
Called when the plugin is loaded
|
Called when the plugin is loaded
|
||||||
"""
|
"""
|
||||||
# client_secrets.json needs to be not empty
|
# client_secrets.json needs to be not empty
|
||||||
display = agent.view()
|
|
||||||
if os.stat("/root/client_secrets.json").st_size == 0:
|
if os.stat("/root/client_secrets.json").st_size == 0:
|
||||||
logging.error("[gDriveSync] /root/client_secrets.json is empty. Please RTFM!")
|
logging.error("[gDriveSync] /root/client_secrets.json is empty. Please RTFM!")
|
||||||
return
|
return
|
||||||
@ -78,25 +73,24 @@ class GdriveSync(plugins.Plugin):
|
|||||||
# logging.warning(f"[gDriveSync] No files found in the folder with ID {root_file_list} and {pwnagotchi_file_list}")
|
# logging.warning(f"[gDriveSync] No files found in the folder with ID {root_file_list} and {pwnagotchi_file_list}")
|
||||||
if self.options['backupfiles'] is not None:
|
if self.options['backupfiles'] is not None:
|
||||||
self.backupfiles = self.backupfiles + self.options['backupfiles']
|
self.backupfiles = self.backupfiles + self.options['backupfiles']
|
||||||
self.backup_files(self.backupfiles, '/backup')
|
self.backup_files(self.backupfiles, '/home/pi/backup')
|
||||||
|
|
||||||
# Create a zip archive of the /backup folder
|
# Create a zip archive of the /backup folder
|
||||||
zip_file_path = os.path.join('/home/pi', 'backup.zip')
|
zip_file_path = os.path.join('/home/pi', 'backup.zip')
|
||||||
with zipfile.ZipFile(zip_file_path, 'w') as zip_ref:
|
with zipfile.ZipFile(zip_file_path, 'w') as zip_ref:
|
||||||
for root, dirs, files in os.walk('/backup'):
|
for root, dirs, files in os.walk('/home/pi/backup'):
|
||||||
for file in files:
|
for file in files:
|
||||||
file_path = os.path.join(root, file)
|
file_path = os.path.join(root, file)
|
||||||
arcname = os.path.relpath(file_path, '/backup')
|
arcname = os.path.relpath(file_path, '/home/pi/backup')
|
||||||
zip_ref.write(file_path, arcname=arcname)
|
zip_ref.write(file_path, arcname=arcname)
|
||||||
|
|
||||||
# Upload the zip archive to Google Drive
|
# Upload the zip archive to Google Drive
|
||||||
self.upload_to_gdrive(zip_file_path, self.get_folder_id_by_name(self.drive, self.options['backup_folder']))
|
self.upload_to_gdrive(zip_file_path, self.get_folder_id_by_name(self.drive, self.options['backup_folder']))
|
||||||
display.on_uploading("Google Drive")
|
|
||||||
self.backup = True
|
self.backup = True
|
||||||
self.status.update()
|
self.status.update()
|
||||||
|
|
||||||
# Specify the local backup path
|
# Specify the local backup path
|
||||||
local_backup_path = '/'
|
local_backup_path = '/home/pi/'
|
||||||
|
|
||||||
# Download the zip archive from Google Drive
|
# Download the zip archive from Google Drive
|
||||||
zip_file_id = self.get_latest_backup_file_id(self.options['backup_folder'])
|
zip_file_id = self.get_latest_backup_file_id(self.options['backup_folder'])
|
||||||
@ -104,7 +98,6 @@ class GdriveSync(plugins.Plugin):
|
|||||||
zip_file = self.drive.CreateFile({'id': zip_file_id})
|
zip_file = self.drive.CreateFile({'id': zip_file_id})
|
||||||
zip_file.GetContentFile(os.path.join(local_backup_path, 'backup.zip'))
|
zip_file.GetContentFile(os.path.join(local_backup_path, 'backup.zip'))
|
||||||
|
|
||||||
display.on_downloading("Google Drive")
|
|
||||||
logging.info("[gDriveSync] Downloaded backup.zip from Google Drive")
|
logging.info("[gDriveSync] Downloaded backup.zip from Google Drive")
|
||||||
|
|
||||||
# Extract the zip archive to the root directory
|
# Extract the zip archive to the root directory
|
||||||
@ -112,9 +105,12 @@ class GdriveSync(plugins.Plugin):
|
|||||||
zip_ref.extractall('/')
|
zip_ref.extractall('/')
|
||||||
|
|
||||||
self.status.update()
|
self.status.update()
|
||||||
os.remove("/backup")
|
shutil.rmtree("/home/pi/backup")
|
||||||
# Reboot so we can start opwngrid with the backup id
|
os.remove("/home/pi/backup.zip")
|
||||||
pwnagotchi.reboot()
|
self.ready = True
|
||||||
|
logging.info("[gdrivesync] loaded")
|
||||||
|
# Restart so we can start opwngrid with the backup id
|
||||||
|
pwnagotchi.restart("AUTO")
|
||||||
|
|
||||||
# all set, gdriveSync is ready to run
|
# all set, gdriveSync is ready to run
|
||||||
self.ready = True
|
self.ready = True
|
||||||
@ -186,15 +182,15 @@ class GdriveSync(plugins.Plugin):
|
|||||||
logging.info("[gdrivesync] new handshake captured, backing up to gdrive")
|
logging.info("[gdrivesync] new handshake captured, backing up to gdrive")
|
||||||
if self.options['backupfiles'] is not None:
|
if self.options['backupfiles'] is not None:
|
||||||
self.backupfiles = self.backupfiles + self.options['backupfiles']
|
self.backupfiles = self.backupfiles + self.options['backupfiles']
|
||||||
self.backup_files(self.backupfiles, '/backup')
|
self.backup_files(self.backupfiles, '/home/pi/backup')
|
||||||
|
|
||||||
# Create a zip archive of the /backup folder
|
# Create a zip archive of the /backup folder
|
||||||
zip_file_path = os.path.join('/home/pi', 'backup.zip')
|
zip_file_path = os.path.join('/home/pi', 'backup.zip')
|
||||||
with zipfile.ZipFile(zip_file_path, 'w') as zip_ref:
|
with zipfile.ZipFile(zip_file_path, 'w') as zip_ref:
|
||||||
for root, dirs, files in os.walk('/backup'):
|
for root, dirs, files in os.walk('/home/pi/backup'):
|
||||||
for file in files:
|
for file in files:
|
||||||
file_path = os.path.join(root, file)
|
file_path = os.path.join(root, file)
|
||||||
arcname = os.path.relpath(file_path, '/backup')
|
arcname = os.path.relpath(file_path, '/home/pi/backup')
|
||||||
zip_ref.write(file_path, arcname=arcname)
|
zip_ref.write(file_path, arcname=arcname)
|
||||||
|
|
||||||
# Upload the zip archive to Google Drive
|
# Upload the zip archive to Google Drive
|
||||||
@ -203,7 +199,7 @@ class GdriveSync(plugins.Plugin):
|
|||||||
|
|
||||||
# Cleanup the local zip file
|
# Cleanup the local zip file
|
||||||
os.remove(zip_file_path)
|
os.remove(zip_file_path)
|
||||||
os.remove("/backup")
|
shutil.rmtree("/home/pi/backup")
|
||||||
self.status.update()
|
self.status.update()
|
||||||
display = agent.view()
|
display = agent.view()
|
||||||
display.update(force=True, new_data={'Backing up to gdrive ...'})
|
display.update(force=True, new_data={'Backing up to gdrive ...'})
|
||||||
|
@ -1,14 +1,20 @@
|
|||||||
# Based on UPS Lite v1.1 from https://github.com/xenDE
|
# Based on UPS Lite v1.1 from https://github.com/xenDE
|
||||||
|
# Made specifically to address the problems caused by the hardware changes in 1.3. Oh yeah I also removed the auto-shutdown feature because it's kind of broken.
|
||||||
#
|
#
|
||||||
# functions for get UPS status - needs enable "i2c" in raspi-config
|
# To setup, see page six of this manual to see how to enable i2c:
|
||||||
|
# https://github.com/linshuqin329/UPS-Lite/blob/master/UPS-Lite_V1.3_CW2015/Instructions%20for%20UPS-Lite%20V1.3.pdf
|
||||||
#
|
#
|
||||||
# https://github.com/linshuqin329/UPS-Lite
|
# Follow page seven, install the dependencies (python-smbus) and copy this script over for later use:
|
||||||
|
# https://github.com/linshuqin329/UPS-Lite/blob/master/UPS-Lite_V1.3_CW2015/UPS_Lite_V1.3_CW2015.py
|
||||||
#
|
#
|
||||||
# For Raspberry Pi Zero Ups Power Expansion Board with Integrated Serial Port S3U4
|
# Now, install this plugin by copying this to the 'available-plugins' folder in your pwnagotchi, install and enable the plugin with the commands:
|
||||||
# https://www.ebay.de/itm/For-Raspberry-Pi-Zero-Ups-Power-Expansion-Board-with-Integrated-Serial-Port-S3U4/323873804310
|
# sudo pwnagotchi plugins install upslite_plugin_1_3
|
||||||
# https://www.aliexpress.com/item/32888533624.html
|
# sudo pwnagotchi plugins enable upslite_plugin_1_3
|
||||||
#
|
#
|
||||||
# To display external power supply status you need to bridge the necessary pins on the UPS-Lite board. See instructions in the UPS-Lite repo.
|
# Now restart raspberry pi. Once back up ensure upslite_plugin_1_3 plugin is turned on in the WebUI. If there is still '0%' on your battery meter
|
||||||
|
# run the script we saved earlier and ensure that the pwnagotchi is plugged in both at the battery and the raspberry pi. The script should start trying to
|
||||||
|
# read the battery, and should be successful once there's a USB cable running power to the battery supply.
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
import struct
|
import struct
|
||||||
|
|
||||||
@ -20,6 +26,11 @@ import pwnagotchi.ui.fonts as fonts
|
|||||||
from pwnagotchi.ui.components import LabeledValue
|
from pwnagotchi.ui.components import LabeledValue
|
||||||
from pwnagotchi.ui.view import BLACK
|
from pwnagotchi.ui.view import BLACK
|
||||||
|
|
||||||
|
CW2015_ADDRESS = 0X62
|
||||||
|
CW2015_REG_VCELL = 0X02
|
||||||
|
CW2015_REG_SOC = 0X04
|
||||||
|
CW2015_REG_MODE = 0X0A
|
||||||
|
|
||||||
|
|
||||||
# TODO: add enable switch in config.yml an cleanup all to the best place
|
# TODO: add enable switch in config.yml an cleanup all to the best place
|
||||||
class UPS:
|
class UPS:
|
||||||
@ -31,8 +42,7 @@ class UPS:
|
|||||||
|
|
||||||
def voltage(self):
|
def voltage(self):
|
||||||
try:
|
try:
|
||||||
address = 0x36
|
read = self._bus.read_word_data(CW2015_ADDRESS, CW2015_REG_VCELL)
|
||||||
read = self._bus.read_word_data(address, 2)
|
|
||||||
swapped = struct.unpack("<H", struct.pack(">H", read))[0]
|
swapped = struct.unpack("<H", struct.pack(">H", read))[0]
|
||||||
return swapped * 1.25 / 1000 / 16
|
return swapped * 1.25 / 1000 / 16
|
||||||
except:
|
except:
|
||||||
@ -41,7 +51,7 @@ class UPS:
|
|||||||
def capacity(self):
|
def capacity(self):
|
||||||
try:
|
try:
|
||||||
address = 0x36
|
address = 0x36
|
||||||
read = self._bus.read_word_data(address, 4)
|
read = self._bus.read_word_data(CW2015_ADDRESS, CW2015_REG_SOC)
|
||||||
swapped = struct.unpack("<H", struct.pack(">H", read))[0]
|
swapped = struct.unpack("<H", struct.pack(">H", read))[0]
|
||||||
return swapped / 256
|
return swapped / 256
|
||||||
except:
|
except:
|
||||||
@ -57,10 +67,10 @@ class UPS:
|
|||||||
|
|
||||||
|
|
||||||
class UPSLite(plugins.Plugin):
|
class UPSLite(plugins.Plugin):
|
||||||
__author__ = 'evilsocket@gmail.com'
|
__author__ = 'marbasec'
|
||||||
__version__ = '1.0.0'
|
__version__ = '1.3.0'
|
||||||
__license__ = 'GPL3'
|
__license__ = 'GPL3'
|
||||||
__description__ = 'A plugin that will add a voltage indicator for the UPS Lite v1.1'
|
__description__ = 'A plugin that will add a voltage indicator for the UPS Lite v1.3'
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.ups = None
|
self.ups = None
|
||||||
@ -69,7 +79,7 @@ class UPSLite(plugins.Plugin):
|
|||||||
self.ups = UPS()
|
self.ups = UPS()
|
||||||
|
|
||||||
def on_ui_setup(self, ui):
|
def on_ui_setup(self, ui):
|
||||||
ui.add_element('ups', LabeledValue(color=BLACK, label='UPS', value='0%/0V', position=(ui.width() / 2 + 15, 0),
|
ui.add_element('ups', LabeledValue(color=BLACK, label='UPS', value='0%', position=(ui.width() / 2 + 15, 0),
|
||||||
label_font=fonts.Bold, text_font=fonts.Medium))
|
label_font=fonts.Bold, text_font=fonts.Medium))
|
||||||
|
|
||||||
def on_unload(self, ui):
|
def on_unload(self, ui):
|
||||||
@ -80,7 +90,3 @@ class UPSLite(plugins.Plugin):
|
|||||||
capacity = self.ups.capacity()
|
capacity = self.ups.capacity()
|
||||||
charging = self.ups.charging()
|
charging = self.ups.charging()
|
||||||
ui.set('ups', "%2i%s" % (capacity, charging))
|
ui.set('ups', "%2i%s" % (capacity, charging))
|
||||||
if capacity <= self.options['shutdown']:
|
|
||||||
logging.info('[ups_lite] Empty battery (<= %s%%): shuting down' % self.options['shutdown'])
|
|
||||||
ui.update(force=True, new_data={'status': 'Battery exhausted, bye ...'})
|
|
||||||
pwnagotchi.shutdown()
|
|
||||||
|
@ -1 +0,0 @@
|
|||||||
|
|
@ -103,8 +103,8 @@ class Display(View):
|
|||||||
def is_waveshare2in13d(self):
|
def is_waveshare2in13d(self):
|
||||||
return self._implementation.name == 'waveshare2in13d'
|
return self._implementation.name == 'waveshare2in13d'
|
||||||
|
|
||||||
def is_waveshare2in23g(self):
|
def is_waveshare2in13g(self):
|
||||||
return self._implementation.name == 'waveshare2in23g'
|
return self._implementation.name == 'waveshare2in13g'
|
||||||
|
|
||||||
def is_waveshare2in36g(self):
|
def is_waveshare2in36g(self):
|
||||||
return self._implementation.name == 'waveshare2in36g'
|
return self._implementation.name == 'waveshare2in36g'
|
||||||
@ -112,6 +112,9 @@ class Display(View):
|
|||||||
def is_waveshare2in66(self):
|
def is_waveshare2in66(self):
|
||||||
return self._implementation.name == 'waveshare2in66'
|
return self._implementation.name == 'waveshare2in66'
|
||||||
|
|
||||||
|
def is_waveshare2in66b(self):
|
||||||
|
return self._implementation.name == 'waveshare2in66b'
|
||||||
|
|
||||||
def is_waveshare2in66g(self):
|
def is_waveshare2in66g(self):
|
||||||
return self._implementation.name == 'waveshare2in66g'
|
return self._implementation.name == 'waveshare2in66g'
|
||||||
|
|
||||||
|
@ -16,50 +16,52 @@ from pwnagotchi.ui.hw.waveshare1in44lcd import Waveshare144lcd
|
|||||||
from pwnagotchi.ui.hw.waveshare1in54b import Waveshare154inchb
|
from pwnagotchi.ui.hw.waveshare1in54b import Waveshare154inchb
|
||||||
from pwnagotchi.ui.hw.waveshare2in13bc import Waveshare213bc
|
from pwnagotchi.ui.hw.waveshare2in13bc import Waveshare213bc
|
||||||
from pwnagotchi.ui.hw.waveshare2in13d import Waveshare213d
|
from pwnagotchi.ui.hw.waveshare2in13d import Waveshare213d
|
||||||
|
from pwnagotchi.ui.hw.waveshare2in13g import Waveshare2in13g
|
||||||
from pwnagotchi.ui.hw.waveshare2in13b_V4 import Waveshare213bV4
|
from pwnagotchi.ui.hw.waveshare2in13b_V4 import Waveshare213bV4
|
||||||
from pwnagotchi.ui.hw.waveshare3in5lcd import Waveshare35lcd
|
from pwnagotchi.ui.hw.waveshare3in5lcd import Waveshare35lcd
|
||||||
from pwnagotchi.ui.hw.spotpear24in import Spotpear24inch
|
from pwnagotchi.ui.hw.spotpear24in import Spotpear24inch
|
||||||
from pwnagotchi.ui.hw.displayhatmini import DisplayHatMini
|
from pwnagotchi.ui.hw.displayhatmini import DisplayHatMini
|
||||||
# from pwnagotchi.ui.hw.waveshare1in02 import Waveshare102
|
from pwnagotchi.ui.hw.waveshare1in02 import Waveshare1in02
|
||||||
from pwnagotchi.ui.hw.waveshare1in54 import Waveshare154
|
from pwnagotchi.ui.hw.waveshare1in54 import Waveshare154
|
||||||
from pwnagotchi.ui.hw.waveshare1in54_V2 import Waveshare154V2
|
from pwnagotchi.ui.hw.waveshare1in54_V2 import Waveshare154V2
|
||||||
from pwnagotchi.ui.hw.waveshare1in54b_V2 import Waveshare154bV2
|
from pwnagotchi.ui.hw.waveshare1in54b_V2 import Waveshare154bV2
|
||||||
# from pwnagotchi.ui.hw.waveshare1in54c import Waveshare154c
|
from pwnagotchi.ui.hw.waveshare1in54c import Waveshare1in54c
|
||||||
# from pwnagotchi.ui.hw.waveshare1in64g import Waveshare164g
|
from pwnagotchi.ui.hw.waveshare1in64g import Waveshare1in64g
|
||||||
from pwnagotchi.ui.hw.waveshare2in7b import Waveshare27b
|
from pwnagotchi.ui.hw.waveshare2in7b import Waveshare27b
|
||||||
from pwnagotchi.ui.hw.waveshare2in7b_V2 import Waveshare27bV2
|
from pwnagotchi.ui.hw.waveshare2in7b_V2 import Waveshare27bV2
|
||||||
from pwnagotchi.ui.hw.waveshare2in9b_V3 import Waveshare29bV3
|
from pwnagotchi.ui.hw.waveshare2in9b_V3 import Waveshare29bV3
|
||||||
from pwnagotchi.ui.hw.waveshare2in9b_V4 import Waveshare29bV4
|
from pwnagotchi.ui.hw.waveshare2in9b_V4 import Waveshare29bV4
|
||||||
# from pwnagotchi.ui.hw.waveshare2in9bc import Waveshare29bc
|
from pwnagotchi.ui.hw.waveshare2in9bc import Waveshare2in9bc
|
||||||
# from pwnagotchi.ui.hw.waveshare2in9d import Waveshare29d
|
from pwnagotchi.ui.hw.waveshare2in9d import Waveshare2in9d
|
||||||
from pwnagotchi.ui.hw.waveshare2in13b_V3 import Waveshare213bV3
|
from pwnagotchi.ui.hw.waveshare2in13b_V3 import Waveshare2in13bV3
|
||||||
# from pwnagotchi.ui.hw.waveshare2in23g import Waveshare223g
|
from pwnagotchi.ui.hw.waveshare2in36g import Waveshare2in36g
|
||||||
# from pwnagotchi.ui.hw.waveshare2in36g import Waveshare236g
|
from pwnagotchi.ui.hw.waveshare2in66 import Waveshare2in66
|
||||||
# from pwnagotchi.ui.hw.waveshare2in66 import Waveshare266
|
from pwnagotchi.ui.hw.waveshare2in66b import Waveshare2in66b
|
||||||
# from pwnagotchi.ui.hw.waveshare3in0g import Waveshare30g
|
from pwnagotchi.ui.hw.waveshare2in66g import Waveshare2in66g
|
||||||
# from pwnagotchi.ui.hw.waveshare3in7 import Waveshare37
|
from pwnagotchi.ui.hw.waveshare3in0g import Waveshare3in0g
|
||||||
# from pwnagotchi.ui.hw.waveshare3in52 import Waveshare352
|
from pwnagotchi.ui.hw.waveshare3in7 import Waveshare3in7
|
||||||
# from pwnagotchi.ui.hw.waveshare4in01f import Waveshare401f
|
from pwnagotchi.ui.hw.waveshare3in52 import Waveshare3in52
|
||||||
# from pwnagotchi.ui.hw.waveshare4in2 import Waveshare42inch
|
from pwnagotchi.ui.hw.waveshare4in01f import Waveshare4in01f
|
||||||
# from pwnagotchi.ui.hw.waveshare4in2_V2 import Waveshare42V2
|
from pwnagotchi.ui.hw.waveshare4in2 import Waveshare4in2
|
||||||
# from pwnagotchi.ui.hw.waveshare4in2b_V2 import Waveshare42bV2
|
from pwnagotchi.ui.hw.waveshare4in2_V2 import Waveshare4in2V2
|
||||||
# from pwnagotchi.ui.hw.waveshare4in2bc import Waveshare42bc
|
from pwnagotchi.ui.hw.waveshare4in2b_V2 import Waveshare4in2bV2
|
||||||
# from pwnagotchi.ui.hw.waveshare4in26 import Waveshare426
|
from pwnagotchi.ui.hw.waveshare4in2bc import Waveshare4in2bc
|
||||||
# from pwnagotchi.ui.hw.waveshare4in37g import Waveshare437g
|
from pwnagotchi.ui.hw.waveshare4in26 import Waveshare4in26
|
||||||
# from pwnagotchi.ui.hw.waveshare5in65f import Waveshare565f
|
from pwnagotchi.ui.hw.waveshare4in37g import Waveshare4in37g
|
||||||
# from pwnagotchi.ui.hw.waveshare5in83 import Waveshare583
|
from pwnagotchi.ui.hw.waveshare5in65f import Waveshare5in65f
|
||||||
# from pwnagotchi.ui.hw.waveshare5in83_V2 import Waveshare583V2
|
from pwnagotchi.ui.hw.waveshare5in83 import Waveshare5in83
|
||||||
# from pwnagotchi.ui.hw.waveshare5in83b_V2 import Waveshare583bV2
|
from pwnagotchi.ui.hw.waveshare5in83_V2 import Waveshare5in83V2
|
||||||
# from pwnagotchi.ui.hw.waveshare5in83b_V2 import Waveshare583bc
|
from pwnagotchi.ui.hw.waveshare5in83b_V2 import Waveshare5in83bV2
|
||||||
# from pwnagotchi.ui.hw.waveshare7in3f import Waveshare73f
|
from pwnagotchi.ui.hw.waveshare5in83bc import Waveshare5in83bc
|
||||||
# from pwnagotchi.ui.hw.waveshare7in3g import Waveshare73g
|
from pwnagotchi.ui.hw.waveshare7in3f import Waveshare7in3f
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5 import Waveshare75
|
from pwnagotchi.ui.hw.waveshare7in3g import Waveshare7in3g
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5_HD import Waveshare75HD
|
from pwnagotchi.ui.hw.waveshare7in5 import Waveshare7in5
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5_V2 import Waveshare75V2
|
from pwnagotchi.ui.hw.waveshare7in5_HD import Waveshare7in5HD
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5b_HD import Waveshare75bHD
|
from pwnagotchi.ui.hw.waveshare7in5_V2 import Waveshare7in5V2
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5b_V2 import Waveshare75bV2
|
from pwnagotchi.ui.hw.waveshare7in5b_HD import Waveshare7in5bHD
|
||||||
# from pwnagotchi.ui.hw.waveshare7in5bc import Waveshare75bc
|
from pwnagotchi.ui.hw.waveshare7in5b_V2 import Waveshare7in5bV2
|
||||||
# from pwnagotchi.ui.hw.waveshare13in3k import Waveshare133k
|
from pwnagotchi.ui.hw.waveshare7in5bc import Waveshare7in5bc
|
||||||
|
from pwnagotchi.ui.hw.waveshare13in3k import Waveshare13in3k
|
||||||
|
|
||||||
|
|
||||||
def display_for(config):
|
def display_for(config):
|
||||||
@ -82,6 +84,21 @@ def display_for(config):
|
|||||||
elif config['ui']['display']['type'] == 'dfrobot_2':
|
elif config['ui']['display']['type'] == 'dfrobot_2':
|
||||||
return DFRobotV2(config)
|
return DFRobotV2(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare144lcd':
|
||||||
|
return Waveshare144lcd(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare35lcd':
|
||||||
|
return Waveshare35lcd(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'spotpear24inch':
|
||||||
|
return Spotpear24inch(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'displayhatmini':
|
||||||
|
return DisplayHatMini(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare1in02':
|
||||||
|
return Waveshare1in02(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare1in54':
|
elif config['ui']['display']['type'] == 'waveshare1in54':
|
||||||
return Waveshare154(config)
|
return Waveshare154(config)
|
||||||
|
|
||||||
@ -94,12 +111,36 @@ def display_for(config):
|
|||||||
elif config['ui']['display']['type'] == 'waveshare1in54b_v2':
|
elif config['ui']['display']['type'] == 'waveshare1in54b_v2':
|
||||||
return Waveshare154bV2(config)
|
return Waveshare154bV2(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare1in54c':
|
||||||
|
return Waveshare1in54c(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare1in64g':
|
||||||
|
return Waveshare1in64g(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in7':
|
||||||
|
return Waveshare27inch(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in7_v2':
|
||||||
|
return Waveshare27inchV2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in7b':
|
elif config['ui']['display']['type'] == 'waveshare2in7b':
|
||||||
return Waveshare27b(config)
|
return Waveshare27b(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in7b_v2':
|
elif config['ui']['display']['type'] == 'waveshare2in7b_v2':
|
||||||
return Waveshare27bV2(config)
|
return Waveshare27bV2(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in9':
|
||||||
|
return Waveshare29inch(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in9bc':
|
||||||
|
return Waveshare2in9bc(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in9d':
|
||||||
|
return Waveshare2in9d(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in9_v2':
|
||||||
|
return Waveshare29inchV2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in9b_v3':
|
elif config['ui']['display']['type'] == 'waveshare2in9b_v3':
|
||||||
return Waveshare29bV3(config)
|
return Waveshare29bV3(config)
|
||||||
|
|
||||||
@ -118,137 +159,101 @@ def display_for(config):
|
|||||||
elif config['ui']['display']['type'] == 'waveshare_4':
|
elif config['ui']['display']['type'] == 'waveshare_4':
|
||||||
return WaveshareV4(config)
|
return WaveshareV4(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in7':
|
|
||||||
return Waveshare27inch(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in7_v2':
|
|
||||||
return Waveshare27inchV2(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in9':
|
|
||||||
return Waveshare29inch(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in9_v2':
|
|
||||||
return Waveshare29inchV2(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare144lcd':
|
|
||||||
return Waveshare144lcd(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare1in54b':
|
|
||||||
return Waveshare154inchb(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in13bc':
|
elif config['ui']['display']['type'] == 'waveshare2in13bc':
|
||||||
return Waveshare213bc(config)
|
return Waveshare213bc(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in13d':
|
elif config['ui']['display']['type'] == 'waveshare2in13d':
|
||||||
return Waveshare213d(config)
|
return Waveshare213d(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in13b_v3':
|
||||||
|
return Waveshare2in13bV3(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in13b_v4':
|
elif config['ui']['display']['type'] == 'waveshare2in13b_v4':
|
||||||
return Waveshare213bV4(config)
|
return Waveshare213bV4(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare35lcd':
|
elif config['ui']['display']['type'] == 'waveshare2in13g':
|
||||||
return Waveshare35lcd(config)
|
return Waveshare2in13g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'spotpear24inch':
|
|
||||||
return Spotpear24inch(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'displayhatmini':
|
|
||||||
return DisplayHatMini(config)
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare1in54c':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare1in64g':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare1in02':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in9bc':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in9d':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in13b_v3':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in23g':
|
|
||||||
return
|
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in36g':
|
elif config['ui']['display']['type'] == 'waveshare2in36g':
|
||||||
return
|
return Waveshare2in36g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare2in66':
|
elif config['ui']['display']['type'] == 'waveshare2in66':
|
||||||
return
|
return Waveshare2in66(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in66b':
|
||||||
|
return Waveshare2in66b(config)
|
||||||
|
|
||||||
|
elif config['ui']['display']['type'] == 'waveshare2in66g':
|
||||||
|
return Waveshare2in66g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare3in0g':
|
elif config['ui']['display']['type'] == 'waveshare3in0g':
|
||||||
return
|
return Waveshare3in0g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare3in7':
|
elif config['ui']['display']['type'] == 'waveshare3in7':
|
||||||
return
|
return Waveshare3in7(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare3in52':
|
elif config['ui']['display']['type'] == 'waveshare3in52':
|
||||||
return
|
return Waveshare3in52(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in01f':
|
elif config['ui']['display']['type'] == 'waveshare4in01f':
|
||||||
return
|
return Waveshare4in01f(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in2':
|
elif config['ui']['display']['type'] == 'waveshare4in2':
|
||||||
return
|
return Waveshare4in2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in2_v2':
|
elif config['ui']['display']['type'] == 'waveshare4in2_v2':
|
||||||
return
|
return Waveshare4in2V2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in2b_v2':
|
elif config['ui']['display']['type'] == 'waveshare4in2b_v2':
|
||||||
return
|
return Waveshare4in2bV2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in2bc':
|
elif config['ui']['display']['type'] == 'waveshare4in2bc':
|
||||||
return
|
return Waveshare4in2bc(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in26':
|
elif config['ui']['display']['type'] == 'waveshare4in26':
|
||||||
return
|
return Waveshare4in26(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare4in37g':
|
elif config['ui']['display']['type'] == 'waveshare4in37g':
|
||||||
return
|
return Waveshare4in37g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare5in65f':
|
elif config['ui']['display']['type'] == 'waveshare5in65f':
|
||||||
return
|
return Waveshare5in65f(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare5in83':
|
elif config['ui']['display']['type'] == 'waveshare5in83':
|
||||||
return
|
return Waveshare5in83(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare5in83_v2':
|
elif config['ui']['display']['type'] == 'waveshare5in83_v2':
|
||||||
return
|
return Waveshare5in83V2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare5in83b_v2':
|
elif config['ui']['display']['type'] == 'waveshare5in83b_v2':
|
||||||
return
|
return Waveshare5in83bV2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare5in83bc':
|
elif config['ui']['display']['type'] == 'waveshare5in83bc':
|
||||||
return
|
return Waveshare5in83bc(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in3f':
|
elif config['ui']['display']['type'] == 'waveshare7in3f':
|
||||||
return
|
return Waveshare7in3f(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in3g':
|
elif config['ui']['display']['type'] == 'waveshare7in3g':
|
||||||
return
|
return Waveshare7in3g(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5':
|
elif config['ui']['display']['type'] == 'waveshare7in5':
|
||||||
return
|
return Waveshare7in5(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5_HD':
|
elif config['ui']['display']['type'] == 'waveshare7in5_HD':
|
||||||
return
|
return Waveshare7in5HD(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5_v2':
|
elif config['ui']['display']['type'] == 'waveshare7in5_v2':
|
||||||
return
|
return Waveshare7in5V2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5b_HD':
|
elif config['ui']['display']['type'] == 'waveshare7in5b_HD':
|
||||||
return
|
return Waveshare7in5bHD(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5b_v2':
|
elif config['ui']['display']['type'] == 'waveshare7in5b_v2':
|
||||||
return
|
return Waveshare7in5bV2(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare7in5bc':
|
elif config['ui']['display']['type'] == 'waveshare7in5bc':
|
||||||
return
|
return Waveshare7in5bc(config)
|
||||||
|
|
||||||
elif config['ui']['display']['type'] == 'waveshare13in3k':
|
elif config['ui']['display']['type'] == 'waveshare13in3k':
|
||||||
return
|
return Waveshare13in3k(config)
|
||||||
|
@ -3,6 +3,7 @@ import pwnagotchi.ui.fonts as fonts
|
|||||||
|
|
||||||
class DisplayImpl(object):
|
class DisplayImpl(object):
|
||||||
def __init__(self, config, name):
|
def __init__(self, config, name):
|
||||||
|
self._display = None
|
||||||
if fonts.Medium is None:
|
if fonts.Medium is None:
|
||||||
fonts.init(config)
|
fonts.init(config)
|
||||||
self.name = name
|
self.name = name
|
||||||
|
@ -3,10 +3,10 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
class DFRobotV1(DisplayImpl):
|
class DFRobotV1(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(DFRobotV1, self).__init__(config, 'dfrobot_1')
|
super(DFRobotV1, self).__init__(config, 'dfrobot_1')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -3,10 +3,10 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
class DFRobotV1(DisplayImpl):
|
class DFRobotV1(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(DFRobotV1, self).__init__(config, 'dfrobot_1')
|
super(DFRobotV1, self).__init__(config, 'dfrobot_1')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -3,10 +3,10 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
class DFRobotV2(DisplayImpl):
|
class DFRobotV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(DFRobotV2, self).__init__(config, 'dfrobot_2')
|
super(DFRobotV2, self).__init__(config, 'dfrobot_2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -3,10 +3,10 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
class DFRobotV2(DisplayImpl):
|
class DFRobotV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(DFRobotV2, self).__init__(config, 'dfrobot_2')
|
super(DFRobotV2, self).__init__(config, 'dfrobot_2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class DisplayHatMini(DisplayImpl):
|
class DisplayHatMini(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(DisplayHatMini, self).__init__(config, 'displayhatmini')
|
super(DisplayHatMini, self).__init__(config, 'displayhatmini')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(12, 10, 12, 70, 25, 9)
|
fonts.setup(12, 10, 12, 70, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Inky(DisplayImpl):
|
class Inky(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Inky, self).__init__(config, 'inky')
|
super(Inky, self).__init__(config, 'inky')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 28, 25, 9)
|
fonts.setup(10, 8, 10, 28, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class LcdHat(DisplayImpl):
|
class LcdHat(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(LcdHat, self).__init__(config, 'lcdhat')
|
super(LcdHat, self).__init__(config, 'lcdhat')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,6 +7,7 @@ EPD_HEIGHT = 128
|
|||||||
|
|
||||||
disp = SH1106.SH1106()
|
disp = SH1106.SH1106()
|
||||||
|
|
||||||
|
|
||||||
class EPD(object):
|
class EPD(object):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
@ -1,219 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd1in54b.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V4.0
|
|
||||||
# * | Date : 2019-06-20
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 200
|
|
||||||
EPD_HEIGHT = 200
|
|
||||||
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
|
|
||||||
lut_vcom0 = [0x0E, 0x14, 0x01, 0x0A, 0x06, 0x04, 0x0A, 0x0A, 0x0F, 0x03, 0x03, 0x0C, 0x06, 0x0A, 0x00]
|
|
||||||
lut_w = [0x0E, 0x14, 0x01, 0x0A, 0x46, 0x04, 0x8A, 0x4A, 0x0F, 0x83, 0x43, 0x0C, 0x86, 0x0A, 0x04]
|
|
||||||
lut_b = [0x0E, 0x14, 0x01, 0x8A, 0x06, 0x04, 0x8A, 0x4A, 0x0F, 0x83, 0x43, 0x0C, 0x06, 0x4A, 0x04]
|
|
||||||
lut_g1 = [0x8E, 0x94, 0x01, 0x8A, 0x06, 0x04, 0x8A, 0x4A, 0x0F, 0x83, 0x43, 0x0C, 0x06, 0x0A, 0x04]
|
|
||||||
lut_g2 = [0x8E, 0x94, 0x01, 0x8A, 0x06, 0x04, 0x8A, 0x4A, 0x0F, 0x83, 0x43, 0x0C, 0x06, 0x0A, 0x04]
|
|
||||||
lut_vcom1 = [0x03, 0x1D, 0x01, 0x01, 0x08, 0x23, 0x37, 0x37, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]
|
|
||||||
lut_red0 = [0x83, 0x5D, 0x01, 0x81, 0x48, 0x23, 0x77, 0x77, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]
|
|
||||||
lut_red1 = [0x03, 0x1D, 0x01, 0x01, 0x08, 0x23, 0x37, 0x37, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0) # module reset
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
logging.debug("e-Paper busy")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0):
|
|
||||||
epdconfig.delay_ms(100)
|
|
||||||
logging.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def set_lut_bw(self):
|
|
||||||
self.send_command(0x20) # vcom
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_vcom0[count])
|
|
||||||
self.send_command(0x21) # ww --
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_w[count])
|
|
||||||
self.send_command(0x22) # bw r
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_b[count])
|
|
||||||
self.send_command(0x23) # wb w
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_g1[count])
|
|
||||||
self.send_command(0x24) # bb b
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_g2[count])
|
|
||||||
|
|
||||||
def set_lut_red(self):
|
|
||||||
self.send_command(0x25)
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_vcom1[count])
|
|
||||||
self.send_command(0x26)
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_red0[count])
|
|
||||||
self.send_command(0x27)
|
|
||||||
for count in range(0, 15):
|
|
||||||
self.send_data(self.lut_red1[count])
|
|
||||||
|
|
||||||
def init(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) # POWER_SETTING
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x08)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0x06) # BOOSTER_SOFT_START
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_command(0x04) # POWER_ON
|
|
||||||
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0X00) # PANEL_SETTING
|
|
||||||
self.send_data(0xCF)
|
|
||||||
self.send_command(0X50) # VCOM_AND_DATA_INTERVAL_SETTING
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_command(0x30) # PLL_CONTROL
|
|
||||||
self.send_data(0x39)
|
|
||||||
self.send_command(0x61) # TCON_RESOLUTION set x and y
|
|
||||||
self.send_data(0xC8)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0xC8)
|
|
||||||
self.send_command(0x82) # VCM_DC_SETTING_REGISTER
|
|
||||||
self.send_data(0x0E)
|
|
||||||
|
|
||||||
self.set_lut_bw()
|
|
||||||
self.set_lut_red()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
buf = [0xFF] * int(self.width * self.height / 8)
|
|
||||||
# Set buffer to value of Python Imaging Library image.
|
|
||||||
# Image must be in mode 1.
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
if imwidth != self.width or imheight != self.height:
|
|
||||||
raise ValueError('Image must be same dimensions as display \
|
|
||||||
({0}x{1}).' .format(self.width, self.height))
|
|
||||||
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
for y in range(self.height):
|
|
||||||
for x in range(self.width):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def display(self, blackimage, redimage):
|
|
||||||
# send black data
|
|
||||||
if (blackimage != None):
|
|
||||||
self.send_command(0x10) # DATA_START_TRANSMISSION_1
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
temp = 0x00
|
|
||||||
for bit in range(0, 4):
|
|
||||||
if (blackimage[i] & (0x80 >> bit) != 0):
|
|
||||||
temp |= 0xC0 >> (bit * 2)
|
|
||||||
self.send_data(temp)
|
|
||||||
temp = 0x00
|
|
||||||
for bit in range(4, 8):
|
|
||||||
if (blackimage[i] & (0x80 >> bit) != 0):
|
|
||||||
temp |= 0xC0 >> ((bit - 4) * 2)
|
|
||||||
self.send_data(temp)
|
|
||||||
|
|
||||||
# send red data
|
|
||||||
if (redimage != None):
|
|
||||||
self.send_command(0x13) # DATA_START_TRANSMISSION_2
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(redimage[i])
|
|
||||||
|
|
||||||
self.send_command(0x12) # DISPLAY_REFRESH
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def Clear(self):
|
|
||||||
self.send_command(0x10) # DATA_START_TRANSMISSION_1
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_data(0xFF)
|
|
||||||
|
|
||||||
self.send_command(0x13) # DATA_START_TRANSMISSION_2
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
|
|
||||||
self.send_command(0x12) # DISPLAY_REFRESH
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0x50) # VCOM_AND_DATA_INTERVAL_SETTING
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_command(0x82) # to solve Vcom drop
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0x01) # power setting
|
|
||||||
self.send_data(0x02) # gate switch to external
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x02) # power off
|
|
||||||
|
|
||||||
epdconfig.module_exit()
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
||||||
|
|
@ -1,154 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2019-06-21
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI = spidev.SpiDev(0, 0)
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -169,7 +169,7 @@ class EPD:
|
|||||||
# self.ReadBusy()
|
# self.ReadBusy()
|
||||||
|
|
||||||
def init(self, lut):
|
def init(self, lut):
|
||||||
if (epdconfig.module_init() != 0):
|
if epdconfig.module_init() != 0:
|
||||||
return -1
|
return -1
|
||||||
# EPD hardware init start
|
# EPD hardware init start
|
||||||
self.reset()
|
self.reset()
|
||||||
|
@ -1,378 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd2in13bc.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V4.0
|
|
||||||
# * | Date : 2019-06-20
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
from PIL import Image
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 104
|
|
||||||
EPD_HEIGHT = 212
|
|
||||||
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
|
|
||||||
|
|
||||||
lut_vcomDC = [
|
|
||||||
0x00, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x60, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_ww = [
|
|
||||||
0x40, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x40, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bw = [
|
|
||||||
0x40, 0x17, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x0F, 0x0F, 0x00, 0x00, 0x03,
|
|
||||||
0x40, 0x0A, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x0E, 0x0E, 0x00, 0x00, 0x02,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_wb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_vcom1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_ww1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bw1 = [
|
|
||||||
0x80, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_wb1 = [
|
|
||||||
0x40, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bb1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
logging.debug("e-Paper busy")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0): # 0: idle, 1: busy
|
|
||||||
epdconfig.delay_ms(100)
|
|
||||||
logging.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def TurnOnDisplay(self):
|
|
||||||
self.send_command(0x12)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def init(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
|
|
||||||
logging.debug("e-Paper 2.13bc preboot Freeze recovery")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0): # 0: idle, 1: busy
|
|
||||||
epdconfig.delay_ms(100)
|
|
||||||
self.reset()
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
self.send_command(0x01) # POWER SETTING
|
|
||||||
self.send_data(0x03)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x03)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
self.send_command(0x06) # BOOSTER_SOFT_START
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_command(0x04) # POWER_ON
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0xf7)
|
|
||||||
self.send_command(0X02) # power off
|
|
||||||
self.send_command(0X07) # deep sleep
|
|
||||||
self.send_data(0xA5)
|
|
||||||
epdconfig.GPIO.output(epdconfig.RST_PIN, 0)
|
|
||||||
epdconfig.GPIO.output(epdconfig.DC_PIN, 0)
|
|
||||||
epdconfig.GPIO.output(epdconfig.CS_PIN, 0)
|
|
||||||
#logging.debug("Reset, powerdown, voltage off done")
|
|
||||||
logging.debug("e-Paper is not frozen now :)")
|
|
||||||
|
|
||||||
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) # POWER SETTING
|
|
||||||
self.send_data(0x03)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x03)
|
|
||||||
|
|
||||||
self.send_command(0x06) # BOOSTER_SOFT_START
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_data(0x17)
|
|
||||||
self.send_data(0x17)
|
|
||||||
|
|
||||||
self.send_command(0x04) # POWER_ON
|
|
||||||
logging.debug("e-Paper 2.13bc bootup busy")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0): # 0: idle, 1: busy
|
|
||||||
epdconfig.delay_ms(100)
|
|
||||||
|
|
||||||
# self.send_command(0x00) # PANEL_SETTING
|
|
||||||
# self.send_data(0x8F)
|
|
||||||
# self.send_command(0x50) # VCOM_AND_DATA_INTERVAL_SETTING
|
|
||||||
# self.send_data(0xF0)
|
|
||||||
# self.send_command(0x61) # RESOLUTION_SETTING
|
|
||||||
# self.send_data(self.width & 0xff)
|
|
||||||
# self.send_data(self.height >> 8)
|
|
||||||
# self.send_data(self.height & 0xff)
|
|
||||||
|
|
||||||
self.send_command(0x00) # panel setting
|
|
||||||
self.send_data(0xbf) # LUT from OTP,128x296
|
|
||||||
self.send_data(0x0d) # VCOM to 0V fast
|
|
||||||
|
|
||||||
self.send_command(0x30) # PLL setting
|
|
||||||
self.send_data(0x3a) # 3a 100HZ 29 150Hz 39 200HZ 31 171HZ
|
|
||||||
|
|
||||||
self.send_command(0x61) # resolution setting
|
|
||||||
self.send_data(self.width & 0xff)
|
|
||||||
self.send_data((self.height >> 8) & 0xff)
|
|
||||||
self.send_data(self.height& 0xff)
|
|
||||||
|
|
||||||
self.send_command(0x82) # vcom_DC setting
|
|
||||||
self.send_data(0x28)
|
|
||||||
|
|
||||||
#self.Clear()
|
|
||||||
logging.debug("e-Paper booted")
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def SetFullReg(self):
|
|
||||||
self.send_command(0x82)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0x97)
|
|
||||||
|
|
||||||
self.send_command(0x20) # vcom
|
|
||||||
for count in range(0, 44):
|
|
||||||
self.send_data(self.lut_vcomDC[count])
|
|
||||||
self.send_command(0x21) # ww --
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_ww[count])
|
|
||||||
self.send_command(0x22) # bw r
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bw[count])
|
|
||||||
self.send_command(0x23) # wb w
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_wb[count])
|
|
||||||
self.send_command(0x24) # bb b
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bb[count])
|
|
||||||
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
# logging.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width/8) * self.height)
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
# logging.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if(imwidth == self.width and imheight == self.height):
|
|
||||||
logging.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
elif(imwidth == self.height and imheight == self.width):
|
|
||||||
logging.debug("Horizontal")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def display(self, imageblack, imagered):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(imageblack[i])
|
|
||||||
self.send_command(0x92)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(imagered[i])
|
|
||||||
self.send_command(0x92)
|
|
||||||
|
|
||||||
self.send_command(0x12) # REFRESH
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def pwndisplay(self, imageblack):
|
|
||||||
if (Image == None):
|
|
||||||
return
|
|
||||||
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0x00)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(imageblack[i])
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.SetFullReg()
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
|
|
||||||
def Clear(self):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x92)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x92)
|
|
||||||
|
|
||||||
self.send_command(0x12) # REFRESH
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def pwnclear(self):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.SetFullReg()
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0x02) # POWER_OFF
|
|
||||||
self.ReadBusy()
|
|
||||||
self.send_command(0x07) # DEEP_SLEEP
|
|
||||||
self.send_data(0xA5) # check code
|
|
||||||
|
|
||||||
epdconfig.module_exit()
|
|
||||||
### END OF FILE ###
|
|
@ -1,154 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2019-06-21
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI = spidev.SpiDev(0, 0)
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -1,358 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd2in13d.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V4.0
|
|
||||||
# * | Date : 2019-06-20
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
from PIL import Image
|
|
||||||
import RPi.GPIO as GPIO
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 104
|
|
||||||
EPD_HEIGHT = 212
|
|
||||||
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
|
|
||||||
lut_vcomDC = [
|
|
||||||
0x00, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x60, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_ww = [
|
|
||||||
0x40, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x40, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bw = [
|
|
||||||
0x40, 0x17, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x0F, 0x0F, 0x00, 0x00, 0x03,
|
|
||||||
0x40, 0x0A, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x0E, 0x0E, 0x00, 0x00, 0x02,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_wb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_vcom1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_ww1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bw1 = [
|
|
||||||
0x80, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_wb1 = [
|
|
||||||
0x40, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_bb1 = [
|
|
||||||
0x00, 0x19, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
logging.debug("e-Paper busy")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0): # 0: idle, 1: busy
|
|
||||||
self.send_command(0x71)
|
|
||||||
epdconfig.delay_ms(100)
|
|
||||||
logging.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def TurnOnDisplay(self):
|
|
||||||
self.send_command(0x12)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def init(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) # POWER SETTING
|
|
||||||
self.send_data(0x03)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x2b)
|
|
||||||
self.send_data(0x03)
|
|
||||||
|
|
||||||
self.send_command(0x06) # boost soft start
|
|
||||||
self.send_data(0x17) # A
|
|
||||||
self.send_data(0x17) # B
|
|
||||||
self.send_data(0x17) # C
|
|
||||||
|
|
||||||
self.send_command(0x04)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x00) # panel setting
|
|
||||||
self.send_data(0xbf) # LUT from OTP,128x296
|
|
||||||
self.send_data(0x0d) # VCOM to 0V fast
|
|
||||||
|
|
||||||
self.send_command(0x30) # PLL setting
|
|
||||||
self.send_data(0x3a) # 3a 100HZ 29 150Hz 39 200HZ 31 171HZ
|
|
||||||
|
|
||||||
self.send_command(0x61) # resolution setting
|
|
||||||
self.send_data(self.width)
|
|
||||||
self.send_data((self.height >> 8) & 0xff)
|
|
||||||
self.send_data(self.height& 0xff)
|
|
||||||
|
|
||||||
self.send_command(0x82) # vcom_DC setting
|
|
||||||
self.send_data(0x28)
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def SetFullReg(self):
|
|
||||||
self.send_command(0x82)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0x97)
|
|
||||||
|
|
||||||
self.send_command(0x20) # vcom
|
|
||||||
for count in range(0, 44):
|
|
||||||
self.send_data(self.lut_vcomDC[count])
|
|
||||||
self.send_command(0x21) # ww --
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_ww[count])
|
|
||||||
self.send_command(0x22) # bw r
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bw[count])
|
|
||||||
self.send_command(0x23) # wb w
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_wb[count])
|
|
||||||
self.send_command(0x24) # bb b
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bb[count])
|
|
||||||
|
|
||||||
def SetPartReg(self):
|
|
||||||
self.send_command(0x82)
|
|
||||||
self.send_data(0x03)
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0x47)
|
|
||||||
|
|
||||||
self.send_command(0x20) # vcom
|
|
||||||
for count in range(0, 44):
|
|
||||||
self.send_data(self.lut_vcom1[count])
|
|
||||||
self.send_command(0x21) # ww --
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_ww1[count])
|
|
||||||
self.send_command(0x22) # bw r
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bw1[count])
|
|
||||||
self.send_command(0x23) # wb w
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_wb1[count])
|
|
||||||
self.send_command(0x24) # bb b
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bb1[count])
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
# logging.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width/8) * self.height)
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
# logging.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if(imwidth == self.width and imheight == self.height):
|
|
||||||
logging.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
elif(imwidth == self.height and imheight == self.width):
|
|
||||||
logging.debug("Horizontal")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def display(self, image):
|
|
||||||
if (Image == None):
|
|
||||||
return
|
|
||||||
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0x00)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(image[i])
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.SetFullReg()
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def DisplayPartial(self, image):
|
|
||||||
if (Image == None):
|
|
||||||
return
|
|
||||||
|
|
||||||
self.SetPartReg()
|
|
||||||
self.send_command(0x91)
|
|
||||||
self.send_command(0x90)
|
|
||||||
self.send_data(0)
|
|
||||||
self.send_data(self.width - 1)
|
|
||||||
|
|
||||||
self.send_data(0)
|
|
||||||
self.send_data(0)
|
|
||||||
self.send_data(int(self.height / 256))
|
|
||||||
self.send_data(self.height % 256 - 1)
|
|
||||||
self.send_data(0x28)
|
|
||||||
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(image[i])
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(~image[i])
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def Clear(self):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0x00)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
|
|
||||||
self.SetFullReg()
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0xf7)
|
|
||||||
self.send_command(0X02) # power off
|
|
||||||
self.send_command(0X07) # deep sleep
|
|
||||||
self.send_data(0xA5)
|
|
||||||
|
|
||||||
epdconfig.module_exit()
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
||||||
|
|
@ -1,154 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2019-06-21
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI = spidev.SpiDev(0, 0)
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -1,227 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.2
|
|
||||||
# * | Date : 2022-10-29
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
self.SPI = spidev.SpiDev()
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
self.SPI.writebytes2(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI.open(0, 0)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN])
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
for i in range(len(data)):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[i])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN])
|
|
||||||
|
|
||||||
|
|
||||||
class SunriseX3:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
Flag = 0
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import Hobot.GPIO
|
|
||||||
|
|
||||||
self.GPIO = Hobot.GPIO
|
|
||||||
self.SPI = spidev.SpiDev()
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
# for i in range(len(data)):
|
|
||||||
# self.SPI.writebytes([data[i]])
|
|
||||||
self.SPI.xfer3(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
if self.Flag == 0:
|
|
||||||
self.Flag = 1
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI.open(2, 0)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
else:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.Flag = 0
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN])
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
elif os.path.exists('/sys/bus/platform/drivers/gpio-x3'):
|
|
||||||
implementation = SunriseX3()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -1,520 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd2in7.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V4.0
|
|
||||||
# * | Date : 2019-06-20
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documentation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furnished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 176
|
|
||||||
EPD_HEIGHT = 264
|
|
||||||
|
|
||||||
GRAY1 = 0xff #white
|
|
||||||
GRAY2 = 0xC0
|
|
||||||
GRAY3 = 0x80 #gray
|
|
||||||
GRAY4 = 0x00 #Blackest
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
self.GRAY1 = GRAY1 #white
|
|
||||||
self.GRAY2 = GRAY2
|
|
||||||
self.GRAY3 = GRAY3 #gray
|
|
||||||
self.GRAY4 = GRAY4 #Blackest
|
|
||||||
|
|
||||||
lut_vcom_dc = [0x00, 0x00,
|
|
||||||
0x00, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x60, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
|
||||||
]
|
|
||||||
lut_ww = [
|
|
||||||
0x40, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x40, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
lut_bw = [
|
|
||||||
0x40, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x40, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
lut_bb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
lut_wb = [
|
|
||||||
0x80, 0x08, 0x00, 0x00, 0x00, 0x02,
|
|
||||||
0x90, 0x28, 0x28, 0x00, 0x00, 0x01,
|
|
||||||
0x80, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x12, 0x12, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
###################full screen update LUT######################
|
|
||||||
#0~3 gray
|
|
||||||
gray_lut_vcom = [
|
|
||||||
0x00, 0x00,
|
|
||||||
0x00, 0x0A, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x60, 0x14, 0x14, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x13, 0x0A, 0x01, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
#R21
|
|
||||||
gray_lut_ww =[
|
|
||||||
0x40, 0x0A, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x90, 0x14, 0x14, 0x00, 0x00, 0x01,
|
|
||||||
0x10, 0x14, 0x0A, 0x00, 0x00, 0x01,
|
|
||||||
0xA0, 0x13, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
#R22H r
|
|
||||||
gray_lut_bw =[
|
|
||||||
0x40, 0x0A, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x90, 0x14, 0x14, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x0A, 0x00, 0x00, 0x01,
|
|
||||||
0x99, 0x0C, 0x01, 0x03, 0x04, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
#R23H w
|
|
||||||
gray_lut_wb =[
|
|
||||||
0x40, 0x0A, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x90, 0x14, 0x14, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x14, 0x0A, 0x00, 0x00, 0x01,
|
|
||||||
0x99, 0x0B, 0x04, 0x04, 0x01, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
#R24H b
|
|
||||||
gray_lut_bb =[
|
|
||||||
0x80, 0x0A, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0x90, 0x14, 0x14, 0x00, 0x00, 0x01,
|
|
||||||
0x20, 0x14, 0x0A, 0x00, 0x00, 0x01,
|
|
||||||
0x50, 0x13, 0x01, 0x00, 0x00, 0x01,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
]
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
logging.debug("e-Paper busy")
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 0): # 0: idle, 1: busy
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
logging.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def set_lut(self):
|
|
||||||
self.send_command(0x20) # vcom
|
|
||||||
for count in range(0, 44):
|
|
||||||
self.send_data(self.lut_vcom_dc[count])
|
|
||||||
self.send_command(0x21) # ww --
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_ww[count])
|
|
||||||
self.send_command(0x22) # bw r
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bw[count])
|
|
||||||
self.send_command(0x23) # wb w
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_bb[count])
|
|
||||||
self.send_command(0x24) # bb b
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.lut_wb[count])
|
|
||||||
|
|
||||||
def gray_SetLut(self):
|
|
||||||
self.send_command(0x20)
|
|
||||||
for count in range(0, 44): #vcom
|
|
||||||
self.send_data(self.gray_lut_vcom[count])
|
|
||||||
|
|
||||||
self.send_command(0x21) #red not use
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.gray_lut_ww[count])
|
|
||||||
|
|
||||||
self.send_command(0x22) #bw r
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.gray_lut_bw[count])
|
|
||||||
|
|
||||||
self.send_command(0x23) #wb w
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.gray_lut_wb[count])
|
|
||||||
|
|
||||||
self.send_command(0x24) #bb b
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.gray_lut_bb[count])
|
|
||||||
|
|
||||||
self.send_command(0x25) #vcom
|
|
||||||
for count in range(0, 42):
|
|
||||||
self.send_data(self.gray_lut_ww[count])
|
|
||||||
|
|
||||||
def init(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) # POWER_SETTING
|
|
||||||
self.send_data(0x03) # VDS_EN, VDG_EN
|
|
||||||
self.send_data(0x00) # VCOM_HV, VGHL_LV[1], VGHL_LV[0]
|
|
||||||
self.send_data(0x2b) # VDH
|
|
||||||
self.send_data(0x2b) # VDL
|
|
||||||
self.send_data(0x09) # VDHR
|
|
||||||
|
|
||||||
self.send_command(0x06) # BOOSTER_SOFT_START
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x17)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0x60)
|
|
||||||
self.send_data(0xA5)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0x89)
|
|
||||||
self.send_data(0xA5)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0x90)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0x93)
|
|
||||||
self.send_data(0x2A)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0xA0)
|
|
||||||
self.send_data(0xA5)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0xA1)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
# Power optimization
|
|
||||||
self.send_command(0xF8)
|
|
||||||
self.send_data(0x73)
|
|
||||||
self.send_data(0x41)
|
|
||||||
|
|
||||||
self.send_command(0x16) # PARTIAL_DISPLAY_REFRESH
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0x04) # POWER_ON
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x00) # PANEL_SETTING
|
|
||||||
self.send_data(0xAF) # KW-BF KWR-AF BWROTP 0f
|
|
||||||
|
|
||||||
self.send_command(0x30) # PLL_CONTROL
|
|
||||||
self.send_data(0x3A) # 3A 100HZ 29 150Hz 39 200HZ 31 171HZ
|
|
||||||
|
|
||||||
self.send_command(0x82) # VCM_DC_SETTING_REGISTER
|
|
||||||
self.send_data(0x12)
|
|
||||||
self.set_lut()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def Init_4Gray(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) #POWER SETTING
|
|
||||||
self.send_data (0x03)
|
|
||||||
self.send_data (0x00)
|
|
||||||
self.send_data (0x2b)
|
|
||||||
self.send_data (0x2b)
|
|
||||||
|
|
||||||
|
|
||||||
self.send_command(0x06) #booster soft start
|
|
||||||
self.send_data (0x07) #A
|
|
||||||
self.send_data (0x07) #B
|
|
||||||
self.send_data (0x17) #C
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0x60)
|
|
||||||
self.send_data (0xA5)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0x89)
|
|
||||||
self.send_data (0xA5)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0x90)
|
|
||||||
self.send_data (0x00)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0x93)
|
|
||||||
self.send_data (0x2A)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0xa0)
|
|
||||||
self.send_data (0xa5)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0xa1)
|
|
||||||
self.send_data (0x00)
|
|
||||||
|
|
||||||
self.send_command(0xF8) #boost??
|
|
||||||
self.send_data (0x73)
|
|
||||||
self.send_data (0x41)
|
|
||||||
|
|
||||||
self.send_command(0x16)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x04)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x00) #panel setting
|
|
||||||
self.send_data(0xbf) #KW-BF KWR-AF BWROTP 0f
|
|
||||||
|
|
||||||
self.send_command(0x30) #PLL setting
|
|
||||||
self.send_data (0x90) #100hz
|
|
||||||
|
|
||||||
self.send_command(0x61) #resolution setting
|
|
||||||
self.send_data (0x00) #176
|
|
||||||
self.send_data (0xb0)
|
|
||||||
self.send_data (0x01) #264
|
|
||||||
self.send_data (0x08)
|
|
||||||
|
|
||||||
self.send_command(0x82) #vcom_DC setting
|
|
||||||
self.send_data (0x12)
|
|
||||||
|
|
||||||
self.send_command(0X50) #VCOM AND DATA INTERVAL SETTING
|
|
||||||
self.send_data(0x97)
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
# logging.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width/8) * self.height)
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
# logging.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if(imwidth == self.width and imheight == self.height):
|
|
||||||
logging.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
elif(imwidth == self.height and imheight == self.width):
|
|
||||||
logging.debug("Horizontal")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def getbuffer_4Gray(self, image):
|
|
||||||
# logging.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width / 4) * self.height)
|
|
||||||
image_monocolor = image.convert('L')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
i=0
|
|
||||||
# logging.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if(imwidth == self.width and imheight == self.height):
|
|
||||||
logging.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if(pixels[x, y] == 0xC0):
|
|
||||||
pixels[x, y] = 0x80
|
|
||||||
elif (pixels[x, y] == 0x80):
|
|
||||||
pixels[x, y] = 0x40
|
|
||||||
i= i+1
|
|
||||||
if(i%4 == 0):
|
|
||||||
buf[int((x + (y * self.width))/4)] = ((pixels[x-3, y]&0xc0) | (pixels[x-2, y]&0xc0)>>2 | (pixels[x-1, y]&0xc0)>>4 | (pixels[x, y]&0xc0)>>6)
|
|
||||||
|
|
||||||
elif(imwidth == self.height and imheight == self.width):
|
|
||||||
logging.debug("Horizontal")
|
|
||||||
for x in range(imwidth):
|
|
||||||
for y in range(imheight):
|
|
||||||
newx = y
|
|
||||||
newy = x
|
|
||||||
if(pixels[x, y] == 0xC0):
|
|
||||||
pixels[x, y] = 0x80
|
|
||||||
elif (pixels[x, y] == 0x80):
|
|
||||||
pixels[x, y] = 0x40
|
|
||||||
i= i+1
|
|
||||||
if(i%4 == 0):
|
|
||||||
buf[int((newx + (newy * self.width))/4)] = ((pixels[x, y-3]&0xc0) | (pixels[x, y-2]&0xc0)>>2 | (pixels[x, y-1]&0xc0)>>4 | (pixels[x, y]&0xc0)>>6)
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def display(self, image):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(image[i])
|
|
||||||
self.send_command(0x12)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def display_4Gray(self, image):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, 5808): #5808*4 46464
|
|
||||||
temp3=0
|
|
||||||
for j in range(0, 2):
|
|
||||||
temp1 = image[i*2+j]
|
|
||||||
for k in range(0, 2):
|
|
||||||
temp2 = temp1&0xC0
|
|
||||||
if(temp2 == 0xC0):
|
|
||||||
temp3 |= 0x01#white
|
|
||||||
elif(temp2 == 0x00):
|
|
||||||
temp3 |= 0x00 #black
|
|
||||||
elif(temp2 == 0x80):
|
|
||||||
temp3 |= 0x01 #gray1
|
|
||||||
else: #0x40
|
|
||||||
temp3 |= 0x00 #gray2
|
|
||||||
temp3 <<= 1
|
|
||||||
|
|
||||||
temp1 <<= 2
|
|
||||||
temp2 = temp1&0xC0
|
|
||||||
if(temp2 == 0xC0): #white
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif(temp2 == 0x00): #black
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif(temp2 == 0x80):
|
|
||||||
temp3 |= 0x01 #gray1
|
|
||||||
else : #0x40
|
|
||||||
temp3 |= 0x00 #gray2
|
|
||||||
if(j!=1 or k!=1):
|
|
||||||
temp3 <<= 1
|
|
||||||
temp1 <<= 2
|
|
||||||
self.send_data(temp3)
|
|
||||||
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, 5808): #5808*4 46464
|
|
||||||
temp3=0
|
|
||||||
for j in range(0, 2):
|
|
||||||
temp1 = image[i*2+j]
|
|
||||||
for k in range(0, 2):
|
|
||||||
temp2 = temp1&0xC0
|
|
||||||
if(temp2 == 0xC0):
|
|
||||||
temp3 |= 0x01#white
|
|
||||||
elif(temp2 == 0x00):
|
|
||||||
temp3 |= 0x00 #black
|
|
||||||
elif(temp2 == 0x80):
|
|
||||||
temp3 |= 0x00 #gray1
|
|
||||||
else: #0x40
|
|
||||||
temp3 |= 0x01 #gray2
|
|
||||||
temp3 <<= 1
|
|
||||||
|
|
||||||
temp1 <<= 2
|
|
||||||
temp2 = temp1&0xC0
|
|
||||||
if(temp2 == 0xC0): #white
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif(temp2 == 0x00): #black
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif(temp2 == 0x80):
|
|
||||||
temp3 |= 0x00 #gray1
|
|
||||||
else: #0x40
|
|
||||||
temp3 |= 0x01 #gray2
|
|
||||||
if(j!=1 or k!=1):
|
|
||||||
temp3 <<= 1
|
|
||||||
temp1 <<= 2
|
|
||||||
self.send_data(temp3)
|
|
||||||
|
|
||||||
self.gray_SetLut()
|
|
||||||
self.send_command(0x12)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
self.ReadBusy()
|
|
||||||
# pass
|
|
||||||
|
|
||||||
def Clear(self, color):
|
|
||||||
self.send_command(0x10)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x13)
|
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x12)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0X50)
|
|
||||||
self.send_data(0xf7)
|
|
||||||
self.send_command(0X02)
|
|
||||||
self.send_command(0X07)
|
|
||||||
self.send_data(0xA5)
|
|
||||||
|
|
||||||
epdconfig.module_exit()
|
|
||||||
### END OF FILE ###
|
|
||||||
|
|
@ -1,154 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2019-06-21
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documentation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furnished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI = spidev.SpiDev(0, 0)
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -1,520 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd2in7_V2.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2022-09-17
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 176
|
|
||||||
EPD_HEIGHT = 264
|
|
||||||
|
|
||||||
GRAY1 = 0xff # white
|
|
||||||
GRAY2 = 0xC0
|
|
||||||
GRAY3 = 0x80 # gray
|
|
||||||
GRAY4 = 0x00 # Blackest
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
self.GRAY1 = GRAY1 # white
|
|
||||||
self.GRAY2 = GRAY2
|
|
||||||
self.GRAY3 = GRAY3 # gray
|
|
||||||
self.GRAY4 = GRAY4 # Blackest
|
|
||||||
|
|
||||||
LUT_DATA_4Gray = [
|
|
||||||
0x40, 0x48, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x8, 0x48, 0x10, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x2, 0x48, 0x4, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x20, 0x48, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0xA, 0x19, 0x0, 0x3, 0x8, 0x0, 0x0,
|
|
||||||
0x14, 0x1, 0x0, 0x14, 0x1, 0x0, 0x3,
|
|
||||||
0xA, 0x3, 0x0, 0x8, 0x19, 0x0, 0x0,
|
|
||||||
0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
||||||
0x22, 0x22, 0x22, 0x22, 0x22, 0x22, 0x0, 0x0, 0x0,
|
|
||||||
0x22, 0x17, 0x41, 0x0, 0x32, 0x1C,
|
|
||||||
]
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
|
||||||
epdconfig.delay_ms(2)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
logger.debug("e-Paper busy")
|
|
||||||
while (epdconfig.digital_read(self.busy_pin) == 1): # 1: idle, 0: busy
|
|
||||||
epdconfig.delay_ms(20)
|
|
||||||
logger.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def TurnOnDisplay(self):
|
|
||||||
self.send_command(0x22) # Display Update Control
|
|
||||||
self.send_data(0xF7)
|
|
||||||
self.send_command(0x20) # Activate Display Update Sequence
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def TurnOnDisplay_Fast(self):
|
|
||||||
self.send_command(0x22) # Display Update Control
|
|
||||||
self.send_data(0xC7)
|
|
||||||
self.send_command(0x20) # Activate Display Update Sequence
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def TurnOnDisplay_Partial(self):
|
|
||||||
self.send_command(0x22) # Display Update Control
|
|
||||||
self.send_data(0xFF)
|
|
||||||
self.send_command(0x20) # Activate Display Update Sequence
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def TurnOnDisplay_4GRAY(self):
|
|
||||||
self.send_command(0x22) # Display Update Control
|
|
||||||
self.send_data(0xC7)
|
|
||||||
self.send_command(0x20) # Activate Display Update Sequence
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def Lut(self):
|
|
||||||
self.send_command(0x32)
|
|
||||||
for i in range(159):
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[i])
|
|
||||||
|
|
||||||
def init(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x12) # SWRESET
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x45) # set Ram-Y address start/end position
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x07) # 0x0107-->(263+1)=264
|
|
||||||
self.send_data(0x01)
|
|
||||||
|
|
||||||
self.send_command(0x4F) # set RAM y address count to 0;
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x11) # data entry mode
|
|
||||||
self.send_data(0x03)
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def init_Fast(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x12) # SWRESET
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x12) # SWRESET
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x18) # Read built-in temperature sensor
|
|
||||||
self.send_data(0x80)
|
|
||||||
|
|
||||||
self.send_command(0x22) # Load temperature value
|
|
||||||
self.send_data(0xB1)
|
|
||||||
self.send_command(0x20)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
self.send_command(0x1A) # Write to temperature register
|
|
||||||
self.send_data(0x64)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x45) # set Ram-Y address start/end position
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x07) # 0x0107-->(263+1)=264
|
|
||||||
self.send_data(0x01)
|
|
||||||
|
|
||||||
self.send_command(0x4F) # set RAM y address count to 0;
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x11) # data entry mode
|
|
||||||
self.send_data(0x03)
|
|
||||||
|
|
||||||
self.send_command(0x22) # Load temperature value
|
|
||||||
self.send_data(0x91)
|
|
||||||
self.send_command(0x20)
|
|
||||||
self.ReadBusy()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def Init_4Gray(self):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x12) # soft reset
|
|
||||||
self.ReadBusy();
|
|
||||||
|
|
||||||
self.send_command(0x74) # set analog block control
|
|
||||||
self.send_data(0x54)
|
|
||||||
self.send_command(0x7E) # set digital block control
|
|
||||||
self.send_data(0x3B)
|
|
||||||
|
|
||||||
self.send_command(0x01) # Driver output control
|
|
||||||
self.send_data(0x07)
|
|
||||||
self.send_data(0x01)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x11) # data entry mode
|
|
||||||
self.send_data(0x03)
|
|
||||||
|
|
||||||
self.send_command(0x44) # set Ram-X address start/end position
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x15) # 0x15-->(21+1)*8=176
|
|
||||||
|
|
||||||
self.send_command(0x45) # set Ram-Y address start/end position
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x07) # 0x0107-->(263+1)=264
|
|
||||||
self.send_data(0x01)
|
|
||||||
|
|
||||||
self.send_command(0x3C) # BorderWavefrom
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x2C) # VCOM Voltage
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[158]) # 0x1C
|
|
||||||
|
|
||||||
self.send_command(0x3F) # EOPQ
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[153])
|
|
||||||
|
|
||||||
self.send_command(0x03) # VGH
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[154])
|
|
||||||
|
|
||||||
self.send_command(0x04) #
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[155]) # VSH1
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[156]) # VSH2
|
|
||||||
self.send_data(self.LUT_DATA_4Gray[157]) # VSL
|
|
||||||
|
|
||||||
self.Lut() # LUT
|
|
||||||
|
|
||||||
self.send_command(0x4E) # set RAM x address count to 0;
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_command(0x4F) # set RAM y address count to 0X199;
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.ReadBusy()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
# logger.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width / 8) * self.height)
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if (imwidth == self.width and imheight == self.height):
|
|
||||||
logger.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
|
||||||
logger.debug("Horizontal")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((newx + newy * self.width) / 8)] &= ~(0x80 >> (y % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def getbuffer_4Gray(self, image):
|
|
||||||
# logger.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width / 4) * self.height)
|
|
||||||
image_monocolor = image.convert('L')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
i = 0
|
|
||||||
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if (imwidth == self.width and imheight == self.height):
|
|
||||||
logger.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if (pixels[x, y] == 0xC0):
|
|
||||||
pixels[x, y] = 0x80
|
|
||||||
elif (pixels[x, y] == 0x80):
|
|
||||||
pixels[x, y] = 0x40
|
|
||||||
i = i + 1
|
|
||||||
if (i % 4 == 0):
|
|
||||||
buf[int((x + (y * self.width)) / 4)] = (
|
|
||||||
(pixels[x - 3, y] & 0xc0) | (pixels[x - 2, y] & 0xc0) >> 2 | (
|
|
||||||
pixels[x - 1, y] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
|
||||||
|
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
|
||||||
logger.debug("Horizontal")
|
|
||||||
for x in range(imwidth):
|
|
||||||
for y in range(imheight):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if (pixels[x, y] == 0xC0):
|
|
||||||
pixels[x, y] = 0x80
|
|
||||||
elif (pixels[x, y] == 0x80):
|
|
||||||
pixels[x, y] = 0x40
|
|
||||||
i = i + 1
|
|
||||||
if (i % 4 == 0):
|
|
||||||
buf[int((newx + (newy * self.width)) / 4)] = (
|
|
||||||
(pixels[x, y - 3] & 0xc0) | (pixels[x, y - 2] & 0xc0) >> 2 | (
|
|
||||||
pixels[x, y - 1] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def Clear(self):
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
self.send_command(0x24)
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(0XFF)
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def display(self, image):
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
self.send_command(0x24)
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(image[i + j * Width])
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def display_Fast(self, image):
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
self.send_command(0x24)
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(image[i + j * Width])
|
|
||||||
self.TurnOnDisplay_Fast()
|
|
||||||
|
|
||||||
def display_Base(self, image):
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
self.send_command(0x24) # Write Black and White image to RAM
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(image[i + j * Width])
|
|
||||||
|
|
||||||
self.send_command(0x26) # Write Black and White image to RAM
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(image[i + j * Width])
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def display_Base_color(self, color):
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
self.send_command(0x24) # Write Black and White image to RAM
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(color)
|
|
||||||
|
|
||||||
self.send_command(0x26) # Write Black and White image to RAM
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
self.send_data(color)
|
|
||||||
# self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def display_Partial(self, Image, Xstart, Ystart, Xend, Yend):
|
|
||||||
if ((Xstart % 8 + Xend % 8 == 8 & Xstart % 8 > Xend % 8) | Xstart % 8 + Xend % 8 == 0 | (
|
|
||||||
Xend - Xstart) % 8 == 0):
|
|
||||||
Xstart = Xstart // 8
|
|
||||||
Xend = Xend // 8
|
|
||||||
else:
|
|
||||||
Xstart = Xstart // 8
|
|
||||||
if Xend % 8 == 0:
|
|
||||||
Xend = Xend // 8
|
|
||||||
else:
|
|
||||||
Xend = Xend // 8 + 1
|
|
||||||
|
|
||||||
if (self.width % 8 == 0):
|
|
||||||
Width = self.width // 8
|
|
||||||
else:
|
|
||||||
Width = self.width // 8 + 1
|
|
||||||
Height = self.height
|
|
||||||
|
|
||||||
Xend -= 1
|
|
||||||
Yend -= 1
|
|
||||||
|
|
||||||
# Reset
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x3C) # BorderWavefrom
|
|
||||||
self.send_data(0x80)
|
|
||||||
|
|
||||||
self.send_command(0x44) # set RAM x address start/end, in page 35
|
|
||||||
self.send_data(Xstart & 0xff) # RAM x address start at 00h;
|
|
||||||
self.send_data(Xend & 0xff) # RAM x address end at 0fh(15+1)*8->128
|
|
||||||
self.send_command(0x45) # set RAM y address start/end, in page 35
|
|
||||||
self.send_data(Ystart & 0xff) # RAM y address start at 0127h;
|
|
||||||
self.send_data((Ystart >> 8) & 0x01) # RAM y address start at 0127h;
|
|
||||||
self.send_data(Yend & 0xff) # RAM y address end at 00h;
|
|
||||||
self.send_data((Yend >> 8) & 0x01)
|
|
||||||
|
|
||||||
self.send_command(0x4E) # set RAM x address count to 0;
|
|
||||||
self.send_data(Xstart & 0xff)
|
|
||||||
self.send_command(0x4F) # set RAM y address count to 0X127;
|
|
||||||
self.send_data(Ystart & 0xff)
|
|
||||||
self.send_data((Ystart >> 8) & 0x01)
|
|
||||||
|
|
||||||
self.send_command(0x24) # Write Black and White image to RAM
|
|
||||||
for j in range(Height):
|
|
||||||
for i in range(Width):
|
|
||||||
if ((j > Ystart - 1) & (j < (Yend + 1)) & (i > Xstart - 1) & (i < (Xend + 1))):
|
|
||||||
self.send_data(Image[i + j * Width])
|
|
||||||
self.TurnOnDisplay_Partial()
|
|
||||||
|
|
||||||
def display_4Gray(self, image):
|
|
||||||
self.send_command(0x24)
|
|
||||||
for i in range(0, 5808): # 5808*4 46464
|
|
||||||
temp3 = 0
|
|
||||||
for j in range(0, 2):
|
|
||||||
temp1 = image[i * 2 + j]
|
|
||||||
for k in range(0, 2):
|
|
||||||
temp2 = temp1 & 0xC0
|
|
||||||
if (temp2 == 0xC0):
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif (temp2 == 0x00):
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif (temp2 == 0x80):
|
|
||||||
temp3 |= 0x01
|
|
||||||
else: # 0x40
|
|
||||||
temp3 |= 0x00
|
|
||||||
temp3 <<= 1
|
|
||||||
|
|
||||||
temp1 <<= 2
|
|
||||||
temp2 = temp1 & 0xC0
|
|
||||||
if (temp2 == 0xC0):
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif (temp2 == 0x00):
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif (temp2 == 0x80):
|
|
||||||
temp3 |= 0x01
|
|
||||||
else: # 0x40
|
|
||||||
temp3 |= 0x00
|
|
||||||
if (j != 1 or k != 1):
|
|
||||||
temp3 <<= 1
|
|
||||||
temp1 <<= 2
|
|
||||||
self.send_data(temp3)
|
|
||||||
|
|
||||||
self.send_command(0x26)
|
|
||||||
for i in range(0, 5808): # 5808*4 46464
|
|
||||||
temp3 = 0
|
|
||||||
for j in range(0, 2):
|
|
||||||
temp1 = image[i * 2 + j]
|
|
||||||
for k in range(0, 2):
|
|
||||||
temp2 = temp1 & 0xC0
|
|
||||||
if (temp2 == 0xC0):
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif (temp2 == 0x00):
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif (temp2 == 0x80):
|
|
||||||
temp3 |= 0x00
|
|
||||||
else: # 0x40
|
|
||||||
temp3 |= 0x01
|
|
||||||
temp3 <<= 1
|
|
||||||
|
|
||||||
temp1 <<= 2
|
|
||||||
temp2 = temp1 & 0xC0
|
|
||||||
if (temp2 == 0xC0):
|
|
||||||
temp3 |= 0x00
|
|
||||||
elif (temp2 == 0x00):
|
|
||||||
temp3 |= 0x01
|
|
||||||
elif (temp2 == 0x80):
|
|
||||||
temp3 |= 0x00
|
|
||||||
else: # 0x40
|
|
||||||
temp3 |= 0x01
|
|
||||||
if (j != 1 or k != 1):
|
|
||||||
temp3 <<= 1
|
|
||||||
temp1 <<= 2
|
|
||||||
self.send_data(temp3)
|
|
||||||
|
|
||||||
self.TurnOnDisplay_4GRAY()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0X10)
|
|
||||||
self.send_data(0x01)
|
|
||||||
|
|
||||||
epdconfig.delay_ms(2000)
|
|
||||||
epdconfig.module_exit()
|
|
||||||
### END OF FILE ###
|
|
@ -1,243 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.2
|
|
||||||
# * | Date : 2022-10-29
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
PWR_PIN = 18
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
self.SPI = spidev.SpiDev()
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
self.SPI.writebytes2(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.PWR_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
|
|
||||||
self.GPIO.output(self.PWR_PIN, 1)
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI.open(0, 0)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
self.GPIO.output(self.PWR_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN, self.PWR_PIN])
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
PWR_PIN = 18
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
for i in range(len(data)):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[i])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.PWR_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
|
|
||||||
self.GPIO.output(self.PWR_PIN, 1)
|
|
||||||
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
self.GPIO.output(self.PWR_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN, self.PWR_PIN])
|
|
||||||
|
|
||||||
|
|
||||||
class SunriseX3:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
PWR_PIN = 18
|
|
||||||
Flag = 0
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import Hobot.GPIO
|
|
||||||
|
|
||||||
self.GPIO = Hobot.GPIO
|
|
||||||
self.SPI = spidev.SpiDev()
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def spi_writebyte2(self, data):
|
|
||||||
# for i in range(len(data)):
|
|
||||||
# self.SPI.writebytes([data[i]])
|
|
||||||
self.SPI.xfer3(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
if self.Flag == 0:
|
|
||||||
self.Flag = 1
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.PWR_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
|
|
||||||
self.GPIO.output(self.PWR_PIN, 1)
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI.open(2, 0)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
else:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logger.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logger.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.Flag = 0
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
self.GPIO.output(self.PWR_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup([self.RST_PIN, self.DC_PIN, self.CS_PIN, self.BUSY_PIN], self.PWR_PIN)
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
elif os.path.exists('/sys/bus/platform/drivers/gpio-x3'):
|
|
||||||
implementation = SunriseX3()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -1,201 +0,0 @@
|
|||||||
# *****************************************************************************
|
|
||||||
# * | File : epd2in9.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Electronic paper driver
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V4.0
|
|
||||||
# * | Date : 2019-06-20
|
|
||||||
# # | Info : python demo
|
|
||||||
# -----------------------------------------------------------------------------
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from . import epdconfig
|
|
||||||
|
|
||||||
# Display resolution
|
|
||||||
EPD_WIDTH = 128
|
|
||||||
EPD_HEIGHT = 296
|
|
||||||
|
|
||||||
class EPD:
|
|
||||||
def __init__(self):
|
|
||||||
self.reset_pin = epdconfig.RST_PIN
|
|
||||||
self.dc_pin = epdconfig.DC_PIN
|
|
||||||
self.busy_pin = epdconfig.BUSY_PIN
|
|
||||||
self.cs_pin = epdconfig.CS_PIN
|
|
||||||
self.width = EPD_WIDTH
|
|
||||||
self.height = EPD_HEIGHT
|
|
||||||
|
|
||||||
lut_full_update = [
|
|
||||||
0x50, 0xAA, 0x55, 0xAA, 0x11, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0x1F, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
|
||||||
]
|
|
||||||
|
|
||||||
lut_partial_update = [
|
|
||||||
0x10, 0x18, 0x18, 0x08, 0x18, 0x18,
|
|
||||||
0x08, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x13, 0x14, 0x44, 0x12,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
|
||||||
]
|
|
||||||
|
|
||||||
# Hardware reset
|
|
||||||
def reset(self):
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
|
||||||
epdconfig.delay_ms(10)
|
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def send_command(self, command):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([command])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def send_data(self, data):
|
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
|
||||||
epdconfig.spi_writebyte([data])
|
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
|
||||||
|
|
||||||
def ReadBusy(self):
|
|
||||||
while(epdconfig.digital_read(self.busy_pin) == 1): # 0: idle, 1: busy
|
|
||||||
epdconfig.delay_ms(200)
|
|
||||||
|
|
||||||
def TurnOnDisplay(self):
|
|
||||||
self.send_command(0x22) # DISPLAY_UPDATE_CONTROL_2
|
|
||||||
self.send_data(0xC4)
|
|
||||||
self.send_command(0x20) # MASTER_ACTIVATION
|
|
||||||
self.send_command(0xFF) # TERMINATE_FRAME_READ_WRITE
|
|
||||||
|
|
||||||
logging.debug("e-Paper busy")
|
|
||||||
self.ReadBusy()
|
|
||||||
logging.debug("e-Paper busy release")
|
|
||||||
|
|
||||||
def SetWindow(self, x_start, y_start, x_end, y_end):
|
|
||||||
self.send_command(0x44) # SET_RAM_X_ADDRESS_START_END_POSITION
|
|
||||||
# x point must be the multiple of 8 or the last 3 bits will be ignored
|
|
||||||
self.send_data((x_start >> 3) & 0xFF)
|
|
||||||
self.send_data((x_end >> 3) & 0xFF)
|
|
||||||
self.send_command(0x45) # SET_RAM_Y_ADDRESS_START_END_POSITION
|
|
||||||
self.send_data(y_start & 0xFF)
|
|
||||||
self.send_data((y_start >> 8) & 0xFF)
|
|
||||||
self.send_data(y_end & 0xFF)
|
|
||||||
self.send_data((y_end >> 8) & 0xFF)
|
|
||||||
|
|
||||||
def SetCursor(self, x, y):
|
|
||||||
self.send_command(0x4E) # SET_RAM_X_ADDRESS_COUNTER
|
|
||||||
# x point must be the multiple of 8 or the last 3 bits will be ignored
|
|
||||||
self.send_data((x >> 3) & 0xFF)
|
|
||||||
self.send_command(0x4F) # SET_RAM_Y_ADDRESS_COUNTER
|
|
||||||
self.send_data(y & 0xFF)
|
|
||||||
self.send_data((y >> 8) & 0xFF)
|
|
||||||
self.ReadBusy()
|
|
||||||
|
|
||||||
def init(self, lut):
|
|
||||||
if (epdconfig.module_init() != 0):
|
|
||||||
return -1
|
|
||||||
# EPD hardware init start
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
self.send_command(0x01) # DRIVER_OUTPUT_CONTROL
|
|
||||||
self.send_data((EPD_HEIGHT - 1) & 0xFF)
|
|
||||||
self.send_data(((EPD_HEIGHT - 1) >> 8) & 0xFF)
|
|
||||||
self.send_data(0x00) # GD = 0 SM = 0 TB = 0
|
|
||||||
|
|
||||||
self.send_command(0x0C) # BOOSTER_SOFT_START_CONTROL
|
|
||||||
self.send_data(0xD7)
|
|
||||||
self.send_data(0xD6)
|
|
||||||
self.send_data(0x9D)
|
|
||||||
|
|
||||||
self.send_command(0x2C) # WRITE_VCOM_REGISTER
|
|
||||||
self.send_data(0xA8) # VCOM 7C
|
|
||||||
|
|
||||||
self.send_command(0x3A) # SET_DUMMY_LINE_PERIOD
|
|
||||||
self.send_data(0x1A) # 4 dummy lines per gate
|
|
||||||
|
|
||||||
self.send_command(0x3B) # SET_GATE_TIME
|
|
||||||
self.send_data(0x08) # 2us per line
|
|
||||||
|
|
||||||
self.send_command(0x11) # DATA_ENTRY_MODE_SETTING
|
|
||||||
self.send_data(0x03) # X increment Y increment
|
|
||||||
|
|
||||||
self.send_command(0x32) # WRITE_LUT_REGISTER
|
|
||||||
for i in range(0, len(lut)):
|
|
||||||
self.send_data(lut[i])
|
|
||||||
# EPD hardware init end
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def getbuffer(self, image):
|
|
||||||
# logging.debug("bufsiz = ",int(self.width/8) * self.height)
|
|
||||||
buf = [0xFF] * (int(self.width/8) * self.height)
|
|
||||||
image_monocolor = image.convert('1')
|
|
||||||
imwidth, imheight = image_monocolor.size
|
|
||||||
pixels = image_monocolor.load()
|
|
||||||
# logging.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
|
||||||
if(imwidth == self.width and imheight == self.height):
|
|
||||||
logging.debug("Vertical")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
# Set the bits for the column of pixels at the current position.
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
|
||||||
elif(imwidth == self.height and imheight == self.width):
|
|
||||||
logging.debug("Horizontal")
|
|
||||||
for y in range(imheight):
|
|
||||||
for x in range(imwidth):
|
|
||||||
newx = y
|
|
||||||
newy = self.height - x - 1
|
|
||||||
if pixels[x, y] == 0:
|
|
||||||
buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8))
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def display(self, image):
|
|
||||||
if (image == None):
|
|
||||||
return
|
|
||||||
self.SetWindow(0, 0, self.width - 1, self.height - 1)
|
|
||||||
for j in range(0, self.height):
|
|
||||||
self.SetCursor(0, j)
|
|
||||||
self.send_command(0x24) # WRITE_RAM
|
|
||||||
for i in range(0, int(self.width / 8)):
|
|
||||||
self.send_data(image[i + j * int(self.width / 8)])
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def Clear(self, color):
|
|
||||||
self.SetWindow(0, 0, self.width - 1, self.height - 1)
|
|
||||||
for j in range(0, self.height):
|
|
||||||
self.SetCursor(0, j)
|
|
||||||
self.send_command(0x24) # WRITE_RAM
|
|
||||||
for i in range(0, int(self.width / 8)):
|
|
||||||
self.send_data(color)
|
|
||||||
self.TurnOnDisplay()
|
|
||||||
|
|
||||||
def sleep(self):
|
|
||||||
self.send_command(0x10) # DEEP_SLEEP_MODE
|
|
||||||
self.send_data(0x01)
|
|
||||||
|
|
||||||
epdconfig.module_exit()
|
|
||||||
### END OF FILE ###
|
|
||||||
|
|
@ -1,154 +0,0 @@
|
|||||||
# /*****************************************************************************
|
|
||||||
# * | File : epdconfig.py
|
|
||||||
# * | Author : Waveshare team
|
|
||||||
# * | Function : Hardware underlying interface
|
|
||||||
# * | Info :
|
|
||||||
# *----------------
|
|
||||||
# * | This version: V1.0
|
|
||||||
# * | Date : 2019-06-21
|
|
||||||
# * | Info :
|
|
||||||
# ******************************************************************************
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documnetation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in
|
|
||||||
# all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
# THE SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
class RaspberryPi:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import spidev
|
|
||||||
import RPi.GPIO
|
|
||||||
|
|
||||||
self.GPIO = RPi.GPIO
|
|
||||||
|
|
||||||
# SPI device, bus = 0, device = 0
|
|
||||||
self.SPI = spidev.SpiDev(0, 0)
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(pin)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.writebytes(data)
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.max_speed_hz = 4000000
|
|
||||||
self.SPI.mode = 0b00
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.close()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
class JetsonNano:
|
|
||||||
# Pin definition
|
|
||||||
RST_PIN = 17
|
|
||||||
DC_PIN = 25
|
|
||||||
CS_PIN = 8
|
|
||||||
BUSY_PIN = 24
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
import ctypes
|
|
||||||
find_dirs = [
|
|
||||||
os.path.dirname(os.path.realpath(__file__)),
|
|
||||||
'/usr/local/lib',
|
|
||||||
'/usr/lib',
|
|
||||||
]
|
|
||||||
self.SPI = None
|
|
||||||
for find_dir in find_dirs:
|
|
||||||
so_filename = os.path.join(find_dir, 'sysfs_software_spi.so')
|
|
||||||
if os.path.exists(so_filename):
|
|
||||||
self.SPI = ctypes.cdll.LoadLibrary(so_filename)
|
|
||||||
break
|
|
||||||
if self.SPI is None:
|
|
||||||
raise RuntimeError('Cannot find sysfs_software_spi.so')
|
|
||||||
|
|
||||||
import Jetson.GPIO
|
|
||||||
self.GPIO = Jetson.GPIO
|
|
||||||
|
|
||||||
def digital_write(self, pin, value):
|
|
||||||
self.GPIO.output(pin, value)
|
|
||||||
|
|
||||||
def digital_read(self, pin):
|
|
||||||
return self.GPIO.input(self.BUSY_PIN)
|
|
||||||
|
|
||||||
def delay_ms(self, delaytime):
|
|
||||||
time.sleep(delaytime / 1000.0)
|
|
||||||
|
|
||||||
def spi_writebyte(self, data):
|
|
||||||
self.SPI.SYSFS_software_spi_transfer(data[0])
|
|
||||||
|
|
||||||
def module_init(self):
|
|
||||||
self.GPIO.setmode(self.GPIO.BCM)
|
|
||||||
self.GPIO.setwarnings(False)
|
|
||||||
self.GPIO.setup(self.RST_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.DC_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.CS_PIN, self.GPIO.OUT)
|
|
||||||
self.GPIO.setup(self.BUSY_PIN, self.GPIO.IN)
|
|
||||||
self.SPI.SYSFS_software_spi_begin()
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def module_exit(self):
|
|
||||||
logging.debug("spi end")
|
|
||||||
self.SPI.SYSFS_software_spi_end()
|
|
||||||
|
|
||||||
logging.debug("close 5V, Module enters 0 power consumption ...")
|
|
||||||
self.GPIO.output(self.RST_PIN, 0)
|
|
||||||
self.GPIO.output(self.DC_PIN, 0)
|
|
||||||
|
|
||||||
self.GPIO.cleanup()
|
|
||||||
|
|
||||||
|
|
||||||
if os.path.exists('/sys/bus/platform/drivers/gpiomem-bcm2835'):
|
|
||||||
implementation = RaspberryPi()
|
|
||||||
else:
|
|
||||||
implementation = JetsonNano()
|
|
||||||
|
|
||||||
for func in [x for x in dir(implementation) if not x.startswith('_')]:
|
|
||||||
setattr(sys.modules[__name__], func, getattr(implementation, func))
|
|
||||||
|
|
||||||
|
|
||||||
### END OF FILE ###
|
|
@ -91,9 +91,9 @@ class EPD:
|
|||||||
def SetWindow(self):
|
def SetWindow(self):
|
||||||
self.send_command(0x61) # SET_RAM_X_ADDRESS_START_END_POSITION
|
self.send_command(0x61) # SET_RAM_X_ADDRESS_START_END_POSITION
|
||||||
# x point must be the multiple of 8 or the last 3 bits will be ignored
|
# x point must be the multiple of 8 or the last 3 bits will be ignored
|
||||||
self.send_data(self.Source_BITS / 256)
|
self.send_data(int(self.Source_BITS / 256))
|
||||||
self.send_data(self.Source_BITS % 256)
|
self.send_data(self.Source_BITS % 256)
|
||||||
self.send_data(self.Gate_BITS / 256)
|
self.send_data(int(self.Gate_BITS / 256))
|
||||||
self.send_data(self.Gate_BITS % 256)
|
self.send_data(self.Gate_BITS % 256)
|
||||||
|
|
||||||
def TurnOnDisplay(self):
|
def TurnOnDisplay(self):
|
||||||
@ -166,20 +166,21 @@ class EPD:
|
|||||||
|
|
||||||
def getbuffer(self, image):
|
def getbuffer(self, image):
|
||||||
# Create a pallette with the 4 colors supported by the panel
|
# Create a pallette with the 4 colors supported by the panel
|
||||||
|
global image_temp
|
||||||
pal_image = Image.new("P", (1, 1))
|
pal_image = Image.new("P", (1, 1))
|
||||||
pal_image.putpalette((0, 0, 0, 255, 255, 255, 255, 255, 0, 255, 0, 0) + (0, 0, 0) * 252)
|
pal_image.putpalette((0, 0, 0, 255, 255, 255, 255, 255, 0, 255, 0, 0) + (0, 0, 0) * 252)
|
||||||
|
|
||||||
# Check if we need to rotate the image
|
# Check if we need to rotate the image
|
||||||
imwidth, imheight = image.size
|
imwidth, imheight = image.size
|
||||||
if (imwidth == self.width and imheight == self.height):
|
if imwidth == self.width and imheight == self.height:
|
||||||
image_temp = image
|
image_temp = image
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
elif imwidth == self.height and imheight == self.width:
|
||||||
image_temp = image.rotate(90, expand=True)
|
image_temp = image.rotate(90, expand=True)
|
||||||
else:
|
else:
|
||||||
logger.warning(
|
logger.warning(
|
||||||
"Invalid image dimensions: %d x %d, expected %d x %d" % (imwidth, imheight, self.width, self.height))
|
"Invalid image dimensions: %d x %d, expected %d x %d" % (imwidth, imheight, self.width, self.height))
|
||||||
|
|
||||||
# Convert the soruce image to the 4 colors, dithering if needed
|
# Convert the source image to the 4 colors, dithering if needed
|
||||||
image_4color = image_temp.convert("RGB").quantize(palette=pal_image)
|
image_4color = image_temp.convert("RGB").quantize(palette=pal_image)
|
||||||
buf_4color = bytearray(image_4color.tobytes('raw'))
|
buf_4color = bytearray(image_4color.tobytes('raw'))
|
||||||
|
|
@ -1,11 +1,11 @@
|
|||||||
# *****************************************************************************
|
# *****************************************************************************
|
||||||
# * | File : epd2in13b_V4.py
|
# * | File : epd2in66b.py
|
||||||
# * | Author : Waveshare team
|
# * | Author : Waveshare team
|
||||||
# * | Function : Electronic paper driver
|
# * | Function : Electronic paper driver
|
||||||
# * | Info :
|
# * | Info :
|
||||||
# *----------------
|
# *----------------
|
||||||
# * | This version: V1.0
|
# * | This version: V1.1
|
||||||
# * | Date : 2022-04-21
|
# * | Date : 2022-08-9
|
||||||
# # | Info : python demo
|
# # | Info : python demo
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
@ -28,11 +28,11 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
from . import epdconfig
|
from .. import epdconfig
|
||||||
|
|
||||||
# Display resolution
|
# Display resolution
|
||||||
EPD_WIDTH = 122
|
EPD_WIDTH = 152
|
||||||
EPD_HEIGHT = 250
|
EPD_HEIGHT = 296
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -46,23 +46,21 @@ class EPD:
|
|||||||
self.width = EPD_WIDTH
|
self.width = EPD_WIDTH
|
||||||
self.height = EPD_HEIGHT
|
self.height = EPD_HEIGHT
|
||||||
|
|
||||||
# hardware reset
|
# Hardware reset
|
||||||
def reset(self):
|
def reset(self):
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
epdconfig.digital_write(self.reset_pin, 1)
|
||||||
epdconfig.delay_ms(20)
|
epdconfig.delay_ms(200)
|
||||||
epdconfig.digital_write(self.reset_pin, 0)
|
epdconfig.digital_write(self.reset_pin, 0)
|
||||||
epdconfig.delay_ms(2)
|
epdconfig.delay_ms(5)
|
||||||
epdconfig.digital_write(self.reset_pin, 1)
|
epdconfig.digital_write(self.reset_pin, 1)
|
||||||
epdconfig.delay_ms(20)
|
epdconfig.delay_ms(200)
|
||||||
|
|
||||||
# send 1 byte command
|
|
||||||
def send_command(self, command):
|
def send_command(self, command):
|
||||||
epdconfig.digital_write(self.dc_pin, 0)
|
epdconfig.digital_write(self.dc_pin, 0)
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
epdconfig.digital_write(self.cs_pin, 0)
|
||||||
epdconfig.spi_writebyte([command])
|
epdconfig.spi_writebyte([command])
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
epdconfig.digital_write(self.cs_pin, 1)
|
||||||
|
|
||||||
# send 1 byte data
|
|
||||||
def send_data(self, data):
|
def send_data(self, data):
|
||||||
epdconfig.digital_write(self.dc_pin, 1)
|
epdconfig.digital_write(self.dc_pin, 1)
|
||||||
epdconfig.digital_write(self.cs_pin, 0)
|
epdconfig.digital_write(self.cs_pin, 0)
|
||||||
@ -76,129 +74,116 @@ class EPD:
|
|||||||
epdconfig.spi_writebyte2(data)
|
epdconfig.spi_writebyte2(data)
|
||||||
epdconfig.digital_write(self.cs_pin, 1)
|
epdconfig.digital_write(self.cs_pin, 1)
|
||||||
|
|
||||||
# judge e-Paper whether is busy
|
def ReadBusy(self):
|
||||||
def busy(self):
|
|
||||||
logger.debug("e-Paper busy")
|
logger.debug("e-Paper busy")
|
||||||
while (epdconfig.digital_read(self.busy_pin) != 0):
|
while (epdconfig.digital_read(self.busy_pin) == 1): # 0: idle, 1: busy
|
||||||
epdconfig.delay_ms(10)
|
epdconfig.delay_ms(20)
|
||||||
logger.debug("e-Paper busy release")
|
logger.debug("e-Paper busy release")
|
||||||
|
|
||||||
# set the display window
|
|
||||||
def set_windows(self, xstart, ystart, xend, yend):
|
|
||||||
self.send_command(0x44) # SET_RAM_X_ADDRESS_START_END_POSITION
|
|
||||||
self.send_data((xstart >> 3) & 0xff)
|
|
||||||
self.send_data((xend >> 3) & 0xff)
|
|
||||||
|
|
||||||
self.send_command(0x45) # SET_RAM_Y_ADDRESS_START_END_POSITION
|
|
||||||
self.send_data(ystart & 0xff)
|
|
||||||
self.send_data((ystart >> 8) & 0xff)
|
|
||||||
self.send_data(yend & 0xff)
|
|
||||||
self.send_data((yend >> 8) & 0xff)
|
|
||||||
|
|
||||||
# set the display cursor(origin)
|
|
||||||
def set_cursor(self, xstart, ystart):
|
|
||||||
self.send_command(0x4E) # SET_RAM_X_ADDRESS_COUNTER
|
|
||||||
self.send_data(xstart & 0xff)
|
|
||||||
|
|
||||||
self.send_command(0x4F) # SET_RAM_Y_ADDRESS_COUNTER
|
|
||||||
self.send_data(ystart & 0xff)
|
|
||||||
self.send_data((ystart >> 8) & 0xff)
|
|
||||||
|
|
||||||
# initialize
|
|
||||||
def init(self):
|
def init(self):
|
||||||
if (epdconfig.module_init() != 0):
|
if (epdconfig.module_init() != 0):
|
||||||
return -1
|
return -1
|
||||||
|
# EPD hardware init start
|
||||||
self.reset()
|
self.reset()
|
||||||
|
|
||||||
self.busy()
|
self.send_command(0x12)
|
||||||
self.send_command(0x12) # SWRESET
|
epdconfig.delay_ms(30)
|
||||||
self.busy()
|
self.ReadBusy()
|
||||||
|
|
||||||
self.send_command(0x01) # Driver output control
|
self.send_command(0x11) # setting gaet number
|
||||||
self.send_data(0xf9)
|
|
||||||
self.send_data(0x00)
|
|
||||||
self.send_data(0x00)
|
|
||||||
|
|
||||||
self.send_command(0x11) # data entry mode
|
|
||||||
self.send_data(0x03)
|
self.send_data(0x03)
|
||||||
|
|
||||||
self.set_windows(0, 0, self.width - 1, self.height - 1)
|
self.setWindows(0, 0, self.width - 1, self.height - 1)
|
||||||
self.set_cursor(0, 0)
|
|
||||||
|
|
||||||
self.send_command(0x3C) # BorderWavefrom
|
self.send_command(0x21)
|
||||||
self.send_data(0x05)
|
self.send_data(0x00)
|
||||||
|
|
||||||
self.send_command(0x18) # Read built-in temperature sensor
|
|
||||||
self.send_data(0x80)
|
self.send_data(0x80)
|
||||||
|
|
||||||
self.send_command(0x21) # Display update control
|
self.setCursor(0, 0)
|
||||||
self.send_data(0x80)
|
self.ReadBusy()
|
||||||
self.send_data(0x80)
|
|
||||||
|
|
||||||
self.busy()
|
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
# turn on display
|
def setWindows(self, Xstart, Ystart, Xend, Yend):
|
||||||
def ondisplay(self):
|
self.send_command(0x44) # SET_RAM_X_ADDRESS_START_END_POSITION
|
||||||
|
self.send_data((Xstart >> 3) & 0x1F)
|
||||||
|
self.send_data((Xend >> 3) & 0x1F)
|
||||||
|
|
||||||
|
self.send_command(0x45) # SET_RAM_Y_ADDRESS_START_END_POSITION
|
||||||
|
self.send_data(Ystart & 0xFF)
|
||||||
|
self.send_data((Ystart >> 8) & 0x01)
|
||||||
|
self.send_data(Yend & 0xFF)
|
||||||
|
self.send_data((Yend >> 8) & 0x01)
|
||||||
|
|
||||||
|
def setCursor(self, Xstart, Ystart):
|
||||||
|
self.send_command(0x4E) # SET_RAM_X_ADDRESS_COUNTER
|
||||||
|
self.send_data(Xstart & 0x1F)
|
||||||
|
|
||||||
|
self.send_command(0x4F) # SET_RAM_Y_ADDRESS_COUNTER
|
||||||
|
self.send_data(Ystart & 0xFF)
|
||||||
|
self.send_data((Ystart >> 8) & 0x01)
|
||||||
|
|
||||||
|
def turnon_display(self):
|
||||||
self.send_command(0x20)
|
self.send_command(0x20)
|
||||||
self.busy()
|
self.ReadBusy()
|
||||||
|
|
||||||
# image converted to bytearray
|
|
||||||
def getbuffer(self, image):
|
def getbuffer(self, image):
|
||||||
img = image
|
# logger.debug("bufsiz = ",int(self.width/8) * self.height)
|
||||||
imwidth, imheight = img.size
|
buf = [0xFF] * (int(self.width / 8) * self.height)
|
||||||
|
image_monocolor = image.convert('1')
|
||||||
|
imwidth, imheight = image_monocolor.size
|
||||||
|
pixels = image_monocolor.load()
|
||||||
|
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
||||||
if (imwidth == self.width and imheight == self.height):
|
if (imwidth == self.width and imheight == self.height):
|
||||||
img = img.convert('1')
|
logger.debug("Vertical")
|
||||||
|
for y in range(imheight):
|
||||||
|
for x in range(imwidth):
|
||||||
|
# Set the bits for the column of pixels at the current position.
|
||||||
|
if pixels[x, y] == 0:
|
||||||
|
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
elif (imwidth == self.height and imheight == self.width):
|
||||||
# image has correct dimensions, but needs to be rotated
|
logger.debug("Horizontal")
|
||||||
img = img.rotate(90, expand=True).convert('1')
|
for y in range(imheight):
|
||||||
else:
|
for x in range(imwidth):
|
||||||
logger.warning("Wrong image dimensions: must be " +
|
newx = y
|
||||||
str(self.width) + "x" + str(self.height))
|
newy = self.height - x - 1
|
||||||
# return a blank buffer
|
if pixels[x, y] == 0:
|
||||||
return [0x00] * (int(self.width/8) * self.height)
|
buf[int((newx + newy * self.width) / 8)] &= ~(0x80 >> (y % 8))
|
||||||
|
|
||||||
buf = bytearray(img.tobytes('raw'))
|
|
||||||
return buf
|
return buf
|
||||||
|
|
||||||
# display image
|
def display(self, Blackimage, Redimage):
|
||||||
def display(self, imageblack, imagered):
|
if (Blackimage == None or Redimage == None):
|
||||||
|
return
|
||||||
|
Redimage_1 = [0x00] * len(Redimage)
|
||||||
|
for i in range(len(Redimage)):
|
||||||
|
Redimage_1[i] = ~Redimage[i]
|
||||||
self.send_command(0x24)
|
self.send_command(0x24)
|
||||||
self.send_data2(imageblack)
|
self.send_data2(Blackimage)
|
||||||
|
|
||||||
self.send_command(0x26)
|
self.send_command(0x26)
|
||||||
self.send_data2(imagered)
|
self.send_data2(Redimage_1)
|
||||||
|
|
||||||
self.ondisplay()
|
self.turnon_display()
|
||||||
|
|
||||||
# display white image
|
def Clear(self):
|
||||||
def clear(self):
|
|
||||||
if self.width % 8 == 0:
|
if self.width % 8 == 0:
|
||||||
linewidth = int(self.width / 8)
|
linewidth = int(self.width / 8)
|
||||||
else:
|
else:
|
||||||
linewidth = int(self.width / 8) + 1
|
linewidth = int(self.width / 8) + 1
|
||||||
|
|
||||||
buf = [0xff] * (int(linewidth * self.height))
|
|
||||||
|
|
||||||
self.send_command(0x24)
|
self.send_command(0x24)
|
||||||
self.send_data2(buf)
|
self.send_data2([0xff] * int(self.height * linewidth))
|
||||||
|
|
||||||
self.send_command(0x26)
|
self.send_command(0x26)
|
||||||
self.send_data2(buf)
|
self.send_data2([0x00] * int(self.height * linewidth))
|
||||||
|
|
||||||
self.ondisplay()
|
self.turnon_display()
|
||||||
|
|
||||||
# Compatible with older version functions
|
|
||||||
def Clear(self):
|
|
||||||
self.clear()
|
|
||||||
|
|
||||||
# sleep
|
|
||||||
def sleep(self):
|
def sleep(self):
|
||||||
self.send_command(0x10) # DEEP_SLEEP
|
self.send_command(0X10) # DEEP_SLEEP_MODE
|
||||||
self.send_data(0x01) # check code
|
self.send_data(0x01)
|
||||||
|
|
||||||
epdconfig.delay_ms(2000)
|
epdconfig.delay_ms(2000)
|
||||||
epdconfig.module_exit()
|
epdconfig.module_exit()
|
||||||
|
|
||||||
### END OF FILE ###
|
### END OF FILE ###
|
@ -498,7 +498,7 @@ class EPD:
|
|||||||
self.ReadBusy()
|
self.ReadBusy()
|
||||||
# pass
|
# pass
|
||||||
|
|
||||||
def Clear(self, color):
|
def Clear(self):
|
||||||
self.send_command(0x10)
|
self.send_command(0x10)
|
||||||
for i in range(0, int(self.width * self.height / 8)):
|
for i in range(0, int(self.width * self.height / 8)):
|
||||||
self.send_data(0xFF)
|
self.send_data(0xFF)
|
||||||
|
@ -100,7 +100,7 @@ class EPD:
|
|||||||
|
|
||||||
def ReadBusy(self):
|
def ReadBusy(self):
|
||||||
logger.debug("e-Paper busy")
|
logger.debug("e-Paper busy")
|
||||||
while (epdconfig.digital_read(self.busy_pin) == 1): # 1: idle, 0: busy
|
while epdconfig.digital_read(self.busy_pin) == 1: # 1: idle, 0: busy
|
||||||
epdconfig.delay_ms(20)
|
epdconfig.delay_ms(20)
|
||||||
logger.debug("e-Paper busy release")
|
logger.debug("e-Paper busy release")
|
||||||
|
|
||||||
@ -134,7 +134,7 @@ class EPD:
|
|||||||
self.send_data(self.LUT_DATA_4Gray[i])
|
self.send_data(self.LUT_DATA_4Gray[i])
|
||||||
|
|
||||||
def init(self):
|
def init(self):
|
||||||
if (epdconfig.module_init() != 0):
|
if epdconfig.module_init() != 0:
|
||||||
return -1
|
return -1
|
||||||
|
|
||||||
# EPD hardware init start
|
# EPD hardware init start
|
||||||
@ -159,7 +159,7 @@ class EPD:
|
|||||||
return 0
|
return 0
|
||||||
|
|
||||||
def init_Fast(self):
|
def init_Fast(self):
|
||||||
if (epdconfig.module_init() != 0):
|
if epdconfig.module_init() != 0:
|
||||||
return -1
|
return -1
|
||||||
|
|
||||||
# EPD hardware init start
|
# EPD hardware init start
|
||||||
@ -204,12 +204,12 @@ class EPD:
|
|||||||
return 0
|
return 0
|
||||||
|
|
||||||
def Init_4Gray(self):
|
def Init_4Gray(self):
|
||||||
if (epdconfig.module_init() != 0):
|
if epdconfig.module_init() != 0:
|
||||||
return -1
|
return -1
|
||||||
self.reset()
|
self.reset()
|
||||||
|
|
||||||
self.send_command(0x12) # soft reset
|
self.send_command(0x12) # soft reset
|
||||||
self.ReadBusy();
|
self.ReadBusy()
|
||||||
|
|
||||||
self.send_command(0x74) # set analog block control
|
self.send_command(0x74) # set analog block control
|
||||||
self.send_data(0x54)
|
self.send_data(0x54)
|
||||||
@ -268,14 +268,14 @@ class EPD:
|
|||||||
imwidth, imheight = image_monocolor.size
|
imwidth, imheight = image_monocolor.size
|
||||||
pixels = image_monocolor.load()
|
pixels = image_monocolor.load()
|
||||||
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
||||||
if (imwidth == self.width and imheight == self.height):
|
if imwidth == self.width and imheight == self.height:
|
||||||
logger.debug("Vertical")
|
logger.debug("Vertical")
|
||||||
for y in range(imheight):
|
for y in range(imheight):
|
||||||
for x in range(imwidth):
|
for x in range(imwidth):
|
||||||
# Set the bits for the column of pixels at the current position.
|
# Set the bits for the column of pixels at the current position.
|
||||||
if pixels[x, y] == 0:
|
if pixels[x, y] == 0:
|
||||||
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8))
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
elif imwidth == self.height and imheight == self.width:
|
||||||
logger.debug("Horizontal")
|
logger.debug("Horizontal")
|
||||||
for y in range(imheight):
|
for y in range(imheight):
|
||||||
for x in range(imwidth):
|
for x in range(imwidth):
|
||||||
@ -293,40 +293,40 @@ class EPD:
|
|||||||
pixels = image_monocolor.load()
|
pixels = image_monocolor.load()
|
||||||
i = 0
|
i = 0
|
||||||
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
# logger.debug("imwidth = %d, imheight = %d",imwidth,imheight)
|
||||||
if (imwidth == self.width and imheight == self.height):
|
if imwidth == self.width and imheight == self.height:
|
||||||
logger.debug("Vertical")
|
logger.debug("Vertical")
|
||||||
for y in range(imheight):
|
for y in range(imheight):
|
||||||
for x in range(imwidth):
|
for x in range(imwidth):
|
||||||
# Set the bits for the column of pixels at the current position.
|
# Set the bits for the column of pixels at the current position.
|
||||||
if (pixels[x, y] == 0xC0):
|
if pixels[x, y] == 0xC0:
|
||||||
pixels[x, y] = 0x80
|
pixels[x, y] = 0x80
|
||||||
elif (pixels[x, y] == 0x80):
|
elif pixels[x, y] == 0x80:
|
||||||
pixels[x, y] = 0x40
|
pixels[x, y] = 0x40
|
||||||
i = i + 1
|
i = i + 1
|
||||||
if (i % 4 == 0):
|
if i % 4 == 0:
|
||||||
buf[int((x + (y * self.width)) / 4)] = (
|
buf[int((x + (y * self.width)) / 4)] = (
|
||||||
(pixels[x - 3, y] & 0xc0) | (pixels[x - 2, y] & 0xc0) >> 2 | (
|
(pixels[x - 3, y] & 0xc0) | (pixels[x - 2, y] & 0xc0) >> 2 | (
|
||||||
pixels[x - 1, y] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
pixels[x - 1, y] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
||||||
|
|
||||||
elif (imwidth == self.height and imheight == self.width):
|
elif imwidth == self.height and imheight == self.width:
|
||||||
logger.debug("Horizontal")
|
logger.debug("Horizontal")
|
||||||
for x in range(imwidth):
|
for x in range(imwidth):
|
||||||
for y in range(imheight):
|
for y in range(imheight):
|
||||||
newx = y
|
newx = y
|
||||||
newy = self.height - x - 1
|
newy = self.height - x - 1
|
||||||
if (pixels[x, y] == 0xC0):
|
if pixels[x, y] == 0xC0:
|
||||||
pixels[x, y] = 0x80
|
pixels[x, y] = 0x80
|
||||||
elif (pixels[x, y] == 0x80):
|
elif pixels[x, y] == 0x80:
|
||||||
pixels[x, y] = 0x40
|
pixels[x, y] = 0x40
|
||||||
i = i + 1
|
i = i + 1
|
||||||
if (i % 4 == 0):
|
if i % 4 == 0:
|
||||||
buf[int((newx + (newy * self.width)) / 4)] = (
|
buf[int((newx + (newy * self.width)) / 4)] = (
|
||||||
(pixels[x, y - 3] & 0xc0) | (pixels[x, y - 2] & 0xc0) >> 2 | (
|
(pixels[x, y - 3] & 0xc0) | (pixels[x, y - 2] & 0xc0) >> 2 | (
|
||||||
pixels[x, y - 1] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
pixels[x, y - 1] & 0xc0) >> 4 | (pixels[x, y] & 0xc0) >> 6)
|
||||||
return buf
|
return buf
|
||||||
|
|
||||||
def Clear(self):
|
def Clear(self):
|
||||||
if (self.width % 8 == 0):
|
if self.width % 8 == 0:
|
||||||
Width = self.width // 8
|
Width = self.width // 8
|
||||||
else:
|
else:
|
||||||
Width = self.width // 8 + 1
|
Width = self.width // 8 + 1
|
||||||
@ -338,7 +338,7 @@ class EPD:
|
|||||||
self.TurnOnDisplay()
|
self.TurnOnDisplay()
|
||||||
|
|
||||||
def display(self, image):
|
def display(self, image):
|
||||||
if (self.width % 8 == 0):
|
if self.width % 8 == 0:
|
||||||
Width = self.width // 8
|
Width = self.width // 8
|
||||||
else:
|
else:
|
||||||
Width = self.width // 8 + 1
|
Width = self.width // 8 + 1
|
||||||
@ -350,7 +350,7 @@ class EPD:
|
|||||||
self.TurnOnDisplay()
|
self.TurnOnDisplay()
|
||||||
|
|
||||||
def display_Fast(self, image):
|
def display_Fast(self, image):
|
||||||
if (self.width % 8 == 0):
|
if self.width % 8 == 0:
|
||||||
Width = self.width // 8
|
Width = self.width // 8
|
||||||
else:
|
else:
|
||||||
Width = self.width // 8 + 1
|
Width = self.width // 8 + 1
|
||||||
@ -362,7 +362,7 @@ class EPD:
|
|||||||
self.TurnOnDisplay_Fast()
|
self.TurnOnDisplay_Fast()
|
||||||
|
|
||||||
def display_Base(self, image):
|
def display_Base(self, image):
|
||||||
if (self.width % 8 == 0):
|
if self.width % 8 == 0:
|
||||||
Width = self.width // 8
|
Width = self.width // 8
|
||||||
else:
|
else:
|
||||||
Width = self.width // 8 + 1
|
Width = self.width // 8 + 1
|
||||||
@ -379,7 +379,7 @@ class EPD:
|
|||||||
self.TurnOnDisplay()
|
self.TurnOnDisplay()
|
||||||
|
|
||||||
def display_Base_color(self, color):
|
def display_Base_color(self, color):
|
||||||
if (self.width % 8 == 0):
|
if self.width % 8 == 0:
|
||||||
Width = self.width // 8
|
Width = self.width // 8
|
||||||
else:
|
else:
|
||||||
Width = self.width // 8 + 1
|
Width = self.width // 8 + 1
|
||||||
@ -396,8 +396,7 @@ class EPD:
|
|||||||
# self.TurnOnDisplay()
|
# self.TurnOnDisplay()
|
||||||
|
|
||||||
def display_Partial(self, Image, Xstart, Ystart, Xend, Yend):
|
def display_Partial(self, Image, Xstart, Ystart, Xend, Yend):
|
||||||
if ((Xstart % 8 + Xend % 8 == 8 & Xstart % 8 > Xend % 8) | Xstart % 8 + Xend % 8 == 0 | (
|
if (Xstart % 8 + Xend % 8 == 8 & Xstart % 8 > Xend % 8) | Xstart % 8 + Xend % 8 == 0 | (Xend - Xstart) % 8 == 0:
|
||||||
Xend - Xstart) % 8 == 0):
|
|
||||||
Xstart = Xstart // 8
|
Xstart = Xstart // 8
|
||||||
Xend = Xend // 8
|
Xend = Xend // 8
|
||||||
else:
|
else:
|
||||||
@ -440,23 +439,23 @@ class EPD:
|
|||||||
self.send_command(0x24) # Write Black and White image to RAM
|
self.send_command(0x24) # Write Black and White image to RAM
|
||||||
for j in range(Height):
|
for j in range(Height):
|
||||||
for i in range(Width):
|
for i in range(Width):
|
||||||
if ((j > Ystart - 1) & (j < (Yend + 1)) & (i > Xstart - 1) & (i < (Xend + 1))):
|
if (j > Ystart - 1) & (j < (Yend + 1)) & (i > Xstart - 1) & (i < (Xend + 1)):
|
||||||
self.send_data(Image[i + j * Width])
|
self.send_data(Image[i + j * Width])
|
||||||
self.TurnOnDisplay_Partial()
|
self.TurnOnDisplay_Partial()
|
||||||
|
|
||||||
def display_4Gray(self, image):
|
def display_4Gray(self, image):
|
||||||
self.send_command(0x24)
|
self.send_command(0x24)
|
||||||
for i in range(0, 48000): # 5808*4 46464
|
for i in range(0, 5808): # 5808*4 46464
|
||||||
temp3 = 0
|
temp3 = 0
|
||||||
for j in range(0, 2):
|
for j in range(0, 2):
|
||||||
temp1 = image[i * 2 + j]
|
temp1 = image[i * 2 + j]
|
||||||
for k in range(0, 2):
|
for k in range(0, 2):
|
||||||
temp2 = temp1 & 0xC0
|
temp2 = temp1 & 0xC0
|
||||||
if (temp2 == 0xC0):
|
if temp2 == 0xC0:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
elif (temp2 == 0x00):
|
elif temp2 == 0x00:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
elif (temp2 == 0x80):
|
elif temp2 == 0x80:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
else: # 0x40
|
else: # 0x40
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
@ -464,31 +463,31 @@ class EPD:
|
|||||||
|
|
||||||
temp1 <<= 2
|
temp1 <<= 2
|
||||||
temp2 = temp1 & 0xC0
|
temp2 = temp1 & 0xC0
|
||||||
if (temp2 == 0xC0):
|
if temp2 == 0xC0:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
elif (temp2 == 0x00):
|
elif temp2 == 0x00:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
elif (temp2 == 0x80):
|
elif temp2 == 0x80:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
else: # 0x40
|
else: # 0x40
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
if (j != 1 or k != 1):
|
if j != 1 or k != 1:
|
||||||
temp3 <<= 1
|
temp3 <<= 1
|
||||||
temp1 <<= 2
|
temp1 <<= 2
|
||||||
self.send_data(temp3)
|
self.send_data(temp3)
|
||||||
|
|
||||||
self.send_command(0x26)
|
self.send_command(0x26)
|
||||||
for i in range(0, 48000): # 5808*4 46464
|
for i in range(0, 5808): # 5808*4 46464
|
||||||
temp3 = 0
|
temp3 = 0
|
||||||
for j in range(0, 2):
|
for j in range(0, 2):
|
||||||
temp1 = image[i * 2 + j]
|
temp1 = image[i * 2 + j]
|
||||||
for k in range(0, 2):
|
for k in range(0, 2):
|
||||||
temp2 = temp1 & 0xC0
|
temp2 = temp1 & 0xC0
|
||||||
if (temp2 == 0xC0):
|
if temp2 == 0xC0:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
elif (temp2 == 0x00):
|
elif temp2 == 0x00:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
elif (temp2 == 0x80):
|
elif temp2 == 0x80:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
else: # 0x40
|
else: # 0x40
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
@ -496,15 +495,15 @@ class EPD:
|
|||||||
|
|
||||||
temp1 <<= 2
|
temp1 <<= 2
|
||||||
temp2 = temp1 & 0xC0
|
temp2 = temp1 & 0xC0
|
||||||
if (temp2 == 0xC0):
|
if temp2 == 0xC0:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
elif (temp2 == 0x00):
|
elif temp2 == 0x00:
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
elif (temp2 == 0x80):
|
elif temp2 == 0x80:
|
||||||
temp3 |= 0x00
|
temp3 |= 0x00
|
||||||
else: # 0x40
|
else: # 0x40
|
||||||
temp3 |= 0x01
|
temp3 |= 0x01
|
||||||
if (j != 1 or k != 1):
|
if j != 1 or k != 1:
|
||||||
temp3 <<= 1
|
temp3 <<= 1
|
||||||
temp1 <<= 2
|
temp1 <<= 2
|
||||||
self.send_data(temp3)
|
self.send_data(temp3)
|
||||||
@ -517,4 +516,3 @@ class EPD:
|
|||||||
|
|
||||||
epdconfig.delay_ms(2000)
|
epdconfig.delay_ms(2000)
|
||||||
epdconfig.module_exit()
|
epdconfig.module_exit()
|
||||||
### END OF FILE ###
|
|
||||||
|
@ -227,7 +227,7 @@ class EPD:
|
|||||||
self.send_data((y >> 8) & 0xFF)
|
self.send_data((y >> 8) & 0xFF)
|
||||||
|
|
||||||
def init(self):
|
def init(self):
|
||||||
if (epdconfig.module_init() != 0):
|
if epdconfig.module_init() != 0:
|
||||||
return -1
|
return -1
|
||||||
# EPD hardware init start
|
# EPD hardware init start
|
||||||
self.reset()
|
self.reset()
|
||||||
|
@ -417,7 +417,7 @@ class EPD:
|
|||||||
self.send_command(0x20)
|
self.send_command(0x20)
|
||||||
self.ReadBusy()
|
self.ReadBusy()
|
||||||
|
|
||||||
def Clear(self, color, mode):
|
def Clear(self, mode):
|
||||||
self.send_command(0x4E)
|
self.send_command(0x4E)
|
||||||
self.send_data(0x00)
|
self.send_data(0x00)
|
||||||
self.send_data(0x00)
|
self.send_data(0x00)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class OledHat(DisplayImpl):
|
class OledHat(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(OledHat, self).__init__(config, 'oledhat')
|
super(OledHat, self).__init__(config, 'oledhat')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(8, 8, 8, 10, 10, 8)
|
fonts.setup(8, 8, 8, 10, 10, 8)
|
||||||
|
@ -8,7 +8,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Papirus(DisplayImpl):
|
class Papirus(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Papirus, self).__init__(config, 'papirus')
|
super(Papirus, self).__init__(config, 'papirus')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 23, 25, 9)
|
fonts.setup(10, 8, 10, 23, 25, 9)
|
||||||
|
@ -3,12 +3,12 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
import os,time
|
import time
|
||||||
|
|
||||||
|
|
||||||
class Spotpear24inch(DisplayImpl):
|
class Spotpear24inch(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Spotpear24inch, self).__init__(config, 'spotpear24inch')
|
super(Spotpear24inch, self).__init__(config, 'spotpear24inch')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(12, 10, 12, 70, 25, 9)
|
fonts.setup(12, 10, 12, 70, 25, 9)
|
||||||
|
@ -3,12 +3,12 @@ import logging
|
|||||||
import pwnagotchi.ui.fonts as fonts
|
import pwnagotchi.ui.fonts as fonts
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
import os,time
|
import time
|
||||||
|
|
||||||
|
|
||||||
class Spotpear24inch(DisplayImpl):
|
class Spotpear24inch(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Spotpear24inch, self).__init__(config, 'spotpear24inch')
|
super(Spotpear24inch, self).__init__(config, 'spotpear24inch')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(12, 10, 12, 70, 25, 9)
|
fonts.setup(12, 10, 12, 70, 25, 9)
|
||||||
|
@ -1,90 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class WaveshareV1(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(WaveshareV1, self).__init__(config, 'waveshare_1')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
if self.config['color'] == 'black':
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 250
|
|
||||||
self._layout['height'] = 122
|
|
||||||
self._layout['face'] = (0, 40)
|
|
||||||
self._layout['name'] = (5, 20)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (185, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 250, 14]
|
|
||||||
self._layout['line2'] = [0, 108, 250, 108]
|
|
||||||
self._layout['friend_face'] = (0, 92)
|
|
||||||
self._layout['friend_name'] = (40, 94)
|
|
||||||
self._layout['shakes'] = (0, 109)
|
|
||||||
self._layout['mode'] = (225, 109)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (125, 20),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
|
||||||
self._layout['width'] = 212
|
|
||||||
self._layout['height'] = 104
|
|
||||||
self._layout['face'] = (0, 26)
|
|
||||||
self._layout['name'] = (5, 15)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (147, 0)
|
|
||||||
self._layout['line1'] = [0, 12, 212, 12]
|
|
||||||
self._layout['line2'] = [0, 92, 212, 92]
|
|
||||||
self._layout['friend_face'] = (0, 76)
|
|
||||||
self._layout['friend_name'] = (40, 78)
|
|
||||||
self._layout['shakes'] = (0, 93)
|
|
||||||
self._layout['mode'] = (187, 93)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (91, 15),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
if self.config['color'] == 'black':
|
|
||||||
logging.info("initializing waveshare v1 display in monochromatic mode")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v1.epd2in13 import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init(self._display.lut_full_update)
|
|
||||||
self._display.Clear(0xFF)
|
|
||||||
self._display.init(self._display.lut_partial_update)
|
|
||||||
elif self.config['color'] == 'fastAndFurious':
|
|
||||||
logging.info("initializing waveshare v1 3-color display in FAST MODE")
|
|
||||||
logging.info("THIS MAY BE POTENTIALLY DANGEROUS. NO WARRANTY IS PROVIDED")
|
|
||||||
logging.info("USE THIS DISPLAY IN THIS MODE AT YOUR OWN RISK")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v1.epd2in13bcFAST import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
else:
|
|
||||||
logging.info("initializing waveshare v1 display 3-color mode")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v1.epd2in13bc import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
if self.config['color'] == 'black':
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf)
|
|
||||||
elif self.config['color'] == 'fastAndFurious':
|
|
||||||
buf_black = self._display.getbuffer(canvas)
|
|
||||||
self._display.DisplayPartial(buf_black)
|
|
||||||
else:
|
|
||||||
buf_black = self._display.getbuffer(canvas)
|
|
||||||
self._display.displayBlack(buf_black)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
self._display.Clear(0xff)
|
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare13in3k(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare13in3k, self).__init__(config, 'waveshare13in3k')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 960
|
||||||
|
self._layout['height'] = 680
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 960, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 960, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 13.3k inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v13in3k.epd13in3k import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare144lcd(DisplayImpl):
|
class Waveshare144lcd(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare144lcd, self).__init__(config, 'waveshare144lcd')
|
super(Waveshare144lcd, self).__init__(config, 'waveshare144lcd')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 18, 25, 9)
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
@ -42,5 +41,4 @@ class Waveshare144lcd(DisplayImpl):
|
|||||||
self._display.display(canvas)
|
self._display.display(canvas)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
pass
|
self._display.clear()
|
||||||
#self._display.clear()
|
|
||||||
|
@ -1,47 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare154inch(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare154inch, self).__init__(config, 'waveshare_1_54inch')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 200
|
|
||||||
self._layout['height'] = 200
|
|
||||||
self._layout['face'] = (0, 40)
|
|
||||||
self._layout['name'] = (5, 20)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (135, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 200, 14]
|
|
||||||
self._layout['line2'] = [0, 186, 200, 186]
|
|
||||||
self._layout['friend_face'] = (0, 92)
|
|
||||||
self._layout['friend_name'] = (40, 94)
|
|
||||||
self._layout['shakes'] = (0, 187)
|
|
||||||
self._layout['mode'] = (170, 187)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (5, 90),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare v154 display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v154inch.epd1in54b import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf, None)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
pass
|
|
||||||
#self._display.Clear()
|
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare1in02(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare1in02, self).__init__(config, 'waveshare1in02')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 80
|
||||||
|
self._layout['height'] = 128
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 80, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 80, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 1.02 inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v1in02.epd1in02 import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.Init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare144lcd(DisplayImpl):
|
class Waveshare144lcd(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare144lcd, self).__init__(config, 'waveshare144lcd')
|
super(Waveshare144lcd, self).__init__(config, 'waveshare144lcd')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 18, 25, 9)
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
@ -42,5 +41,4 @@ class Waveshare144lcd(DisplayImpl):
|
|||||||
self._display.display(canvas)
|
self._display.display(canvas)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
pass
|
self._display.clear()
|
||||||
#self._display.clear()
|
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare154(DisplayImpl):
|
class Waveshare154(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare154, self).__init__(config, 'waveshare1in54')
|
super(Waveshare154, self).__init__(config, 'waveshare1in54')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
@ -32,10 +31,10 @@ class Waveshare154(DisplayImpl):
|
|||||||
return self._layout
|
return self._layout
|
||||||
|
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
logging.info("initializing waveshare v154 display")
|
logging.info("initializing waveshare v1in54 display")
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v1in54.epd1in54 import EPD
|
from pwnagotchi.ui.hw.libs.waveshare.v1in54.epd1in54 import EPD
|
||||||
self._display = EPD()
|
self._display = EPD()
|
||||||
self._display.init()
|
self._display.init(0x00)
|
||||||
self._display.Clear()
|
self._display.Clear()
|
||||||
|
|
||||||
def render(self, canvas):
|
def render(self, canvas):
|
||||||
@ -43,4 +42,5 @@ class Waveshare154(DisplayImpl):
|
|||||||
self._display.display(buf, None)
|
self._display.display(buf, None)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
|
# pass
|
||||||
self._display.Clear()
|
self._display.Clear()
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare154V2(DisplayImpl):
|
class Waveshare154V2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare154V2, self).__init__(config, 'waveshare1in54_v2')
|
super(Waveshare154V2, self).__init__(config, 'waveshare1in54_v2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
@ -32,10 +31,10 @@ class Waveshare154V2(DisplayImpl):
|
|||||||
return self._layout
|
return self._layout
|
||||||
|
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
logging.info("initializing waveshare v154 display")
|
logging.info("initializing waveshare v1in54_v2 display")
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v1in54_v2.epd1in54_V2 import EPD
|
from pwnagotchi.ui.hw.libs.waveshare.v1in54_v2.epd1in54_V2 import EPD
|
||||||
self._display = EPD()
|
self._display = EPD()
|
||||||
self._display.init()
|
self._display.init(False)
|
||||||
self._display.Clear()
|
self._display.Clear()
|
||||||
|
|
||||||
def render(self, canvas):
|
def render(self, canvas):
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare154inchb(DisplayImpl):
|
class Waveshare154inchb(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare154inchb, self).__init__(config, 'waveshare1in54b')
|
super(Waveshare154inchb, self).__init__(config, 'waveshare1in54b')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare154bV2(DisplayImpl):
|
class Waveshare154bV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare154bV2, self).__init__(config, 'waveshare1in54b_v2')
|
super(Waveshare154bV2, self).__init__(config, 'waveshare1in54b_v2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare1in54c(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare1in54c, self).__init__(config, 'waveshare1in54c')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 152
|
||||||
|
self._layout['height'] = 152
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 152, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 152, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 1.54c inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v1in54c.epd1in54c import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare1in64g(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare1in64g, self).__init__(config, 'waveshare1in64g')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 168
|
||||||
|
self._layout['height'] = 168
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 168, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 168, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 1.64g inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v1in64g.epd1in64g import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
@ -1,69 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class WaveshareV2(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(WaveshareV2, self).__init__(config, 'waveshare_2')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
if self.config['color'] == 'black':
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 250
|
|
||||||
self._layout['height'] = 122
|
|
||||||
self._layout['face'] = (0, 40)
|
|
||||||
self._layout['name'] = (5, 20)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (185, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 250, 14]
|
|
||||||
self._layout['line2'] = [0, 108, 250, 108]
|
|
||||||
self._layout['friend_face'] = (0, 92)
|
|
||||||
self._layout['friend_name'] = (40, 94)
|
|
||||||
self._layout['shakes'] = (0, 109)
|
|
||||||
self._layout['mode'] = (225, 109)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (125, 20),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
|
||||||
self._layout['width'] = 212
|
|
||||||
self._layout['height'] = 104
|
|
||||||
self._layout['face'] = (0, 26)
|
|
||||||
self._layout['name'] = (5, 15)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['status'] = (91, 15)
|
|
||||||
self._layout['uptime'] = (147, 0)
|
|
||||||
self._layout['line1'] = [0, 12, 212, 12]
|
|
||||||
self._layout['line2'] = [0, 92, 212, 92]
|
|
||||||
self._layout['friend_face'] = (0, 76)
|
|
||||||
self._layout['friend_name'] = (40, 78)
|
|
||||||
self._layout['shakes'] = (0, 93)
|
|
||||||
self._layout['mode'] = (187, 93)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (125, 20),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 14
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare v2 display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v2.waveshare import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init(self._display.FULL_UPDATE)
|
|
||||||
self._display.Clear(0xff)
|
|
||||||
self._display.init(self._display.PART_UPDATE)
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.displayPartial(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
self._display.Clear(0xff)
|
|
@ -1,47 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare213bc(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare213bc, self).__init__(config, 'waveshare213bc')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
|
||||||
self._layout['width'] = 212
|
|
||||||
self._layout['height'] = 104
|
|
||||||
self._layout['face'] = (0, 26)
|
|
||||||
self._layout['name'] = (5, 15)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (147, 0)
|
|
||||||
self._layout['line1'] = [0, 12, 212, 12]
|
|
||||||
self._layout['line2'] = [0, 92, 212, 92]
|
|
||||||
self._layout['friend_face'] = (0, 76)
|
|
||||||
self._layout['friend_name'] = (40, 78)
|
|
||||||
self._layout['shakes'] = (0, 93)
|
|
||||||
self._layout['mode'] = (187, 93)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (91, 15),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare 213bc display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v213bc.epd2in13bc import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.pwndisplay(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
#pass
|
|
||||||
self._display.pwnclear()
|
|
@ -1,47 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare213d(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare213d, self).__init__(config, 'waveshare213d')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
|
||||||
self._layout['width'] = 212
|
|
||||||
self._layout['height'] = 104
|
|
||||||
self._layout['face'] = (0, 26)
|
|
||||||
self._layout['name'] = (5, 15)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (147, 0)
|
|
||||||
self._layout['line1'] = [0, 12, 212, 12]
|
|
||||||
self._layout['line2'] = [0, 92, 212, 92]
|
|
||||||
self._layout['friend_face'] = (0, 76)
|
|
||||||
self._layout['friend_name'] = (40, 78)
|
|
||||||
self._layout['shakes'] = (0, 93)
|
|
||||||
self._layout['mode'] = (187, 93)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (91, 15),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare 213d display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v213d.epd2in13d import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
#pass
|
|
||||||
self._display.Clear()
|
|
@ -1,72 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
from PIL import Image
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare213bV4(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare213bV4, self).__init__(config, 'waveshare213inb_v4')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
if self.config['color'] == 'black':
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 250
|
|
||||||
self._layout['height'] = 122
|
|
||||||
self._layout['face'] = (0, 40)
|
|
||||||
self._layout['name'] = (5, 20)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (185, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 250, 14]
|
|
||||||
self._layout['line2'] = [0, 108, 250, 108]
|
|
||||||
self._layout['friend_face'] = (0, 92)
|
|
||||||
self._layout['friend_name'] = (40, 94)
|
|
||||||
self._layout['shakes'] = (0, 109)
|
|
||||||
self._layout['mode'] = (225, 109)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (125, 20),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 20
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
|
||||||
self._layout['width'] = 212
|
|
||||||
self._layout['height'] = 104
|
|
||||||
self._layout['face'] = (0, 26)
|
|
||||||
self._layout['name'] = (5, 15)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['status'] = (91, 15)
|
|
||||||
self._layout['uptime'] = (147, 0)
|
|
||||||
self._layout['line1'] = [0, 12, 212, 12]
|
|
||||||
self._layout['line2'] = [0, 92, 212, 92]
|
|
||||||
self._layout['friend_face'] = (0, 76)
|
|
||||||
self._layout['friend_name'] = (40, 78)
|
|
||||||
self._layout['shakes'] = (0, 93)
|
|
||||||
self._layout['mode'] = (187, 93)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (125, 20),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 14
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare 2.13inb v4 display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v213inb_v4.epd2in13b_V4 import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvasBlack = None, canvasRed = None):
|
|
||||||
buffer = self._display.getbuffer
|
|
||||||
image = Image.new('1', (self._layout['height'], self._layout['width']))
|
|
||||||
imageBlack = image if canvasBlack is None else canvasBlack
|
|
||||||
imageRed = image if canvasRed is None else canvasRed
|
|
||||||
self._display.display(buffer(imageBlack), buffer(imageRed))
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
self._display.Clear()
|
|
@ -1,46 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare27inch(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare27inch, self).__init__(config, 'waveshare27inch')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 264
|
|
||||||
self._layout['height'] = 176
|
|
||||||
self._layout['face'] = (66, 27)
|
|
||||||
self._layout['name'] = (5, 73)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (199, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 264, 14]
|
|
||||||
self._layout['line2'] = [0, 162, 264, 162]
|
|
||||||
self._layout['friend_face'] = (0, 146)
|
|
||||||
self._layout['friend_name'] = (40, 146)
|
|
||||||
self._layout['shakes'] = (0, 163)
|
|
||||||
self._layout['mode'] = (239, 163)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (38, 93),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 40
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare v1 2.7 inch display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v27inch.epd2in7 import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
self._display.Clear(0xFF)
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
self._display.Clear(0xff)
|
|
@ -1,49 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare27inchV2(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare27inchV2, self).__init__(config, 'waveshare27inchV2')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 264
|
|
||||||
self._layout['height'] = 176
|
|
||||||
self._layout['face'] = (66, 27)
|
|
||||||
self._layout['name'] = (5, 73)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (199, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 264, 14]
|
|
||||||
self._layout['line2'] = [0, 162, 264, 162]
|
|
||||||
self._layout['friend_face'] = (0, 146)
|
|
||||||
self._layout['friend_name'] = (40, 146)
|
|
||||||
self._layout['shakes'] = (0, 163)
|
|
||||||
self._layout['mode'] = (239, 163)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (38, 93),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 40
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare v2 2.7 inch display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v27inchV2.epd2in7_V2 import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init()
|
|
||||||
# this must have changed by waveshare
|
|
||||||
# remove the 0xFF(Clear(0xFF)) other wise it errors. can't pass oxff and self
|
|
||||||
self._display.Clear()
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
# This line also removes the 0xFF
|
|
||||||
self._display.Clear()
|
|
@ -1,47 +0,0 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
import pwnagotchi.ui.fonts as fonts
|
|
||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
|
||||||
|
|
||||||
|
|
||||||
class Waveshare29inch(DisplayImpl):
|
|
||||||
def __init__(self, config):
|
|
||||||
super(Waveshare29inch, self).__init__(config, 'waveshare_29inch')
|
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
|
||||||
self._layout['width'] = 296
|
|
||||||
self._layout['height'] = 128
|
|
||||||
self._layout['face'] = (0, 40)
|
|
||||||
self._layout['name'] = (5, 25)
|
|
||||||
self._layout['channel'] = (0, 0)
|
|
||||||
self._layout['aps'] = (28, 0)
|
|
||||||
self._layout['uptime'] = (230, 0)
|
|
||||||
self._layout['line1'] = [0, 14, 296, 14]
|
|
||||||
self._layout['line2'] = [0, 112, 296, 112]
|
|
||||||
self._layout['friend_face'] = (0, 96)
|
|
||||||
self._layout['friend_name'] = (40, 96)
|
|
||||||
self._layout['shakes'] = (0, 114)
|
|
||||||
self._layout['mode'] = (268, 114)
|
|
||||||
self._layout['status'] = {
|
|
||||||
'pos': (130, 25),
|
|
||||||
'font': fonts.status_font(fonts.Medium),
|
|
||||||
'max': 28
|
|
||||||
}
|
|
||||||
return self._layout
|
|
||||||
|
|
||||||
def initialize(self):
|
|
||||||
logging.info("initializing waveshare v1 2.9 inch display")
|
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v29inch.epd2in9 import EPD
|
|
||||||
self._display = EPD()
|
|
||||||
self._display.init(self._display.lut_full_update)
|
|
||||||
self._display.Clear(0xFF)
|
|
||||||
self._display.init(self._display.lut_partial_update)
|
|
||||||
|
|
||||||
def render(self, canvas):
|
|
||||||
buf = self._display.getbuffer(canvas)
|
|
||||||
self._display.display(buf)
|
|
||||||
|
|
||||||
def clear(self):
|
|
||||||
self._display.Clear(0xFF)
|
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class WaveshareV1(DisplayImpl):
|
class WaveshareV1(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(WaveshareV1, self).__init__(config, 'waveshare_1')
|
super(WaveshareV1, self).__init__(config, 'waveshare_1')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
if self.config['color'] == 'black':
|
if self.config['color'] == 'black':
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class WaveshareV2(DisplayImpl):
|
class WaveshareV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(WaveshareV2, self).__init__(config, 'waveshare_2')
|
super(WaveshareV2, self).__init__(config, 'waveshare_2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
if self.config['color'] == 'black':
|
if self.config['color'] == 'black':
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class WaveshareV3(DisplayImpl):
|
class WaveshareV3(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(WaveshareV3, self).__init__(config, 'waveshare_3')
|
super(WaveshareV3, self).__init__(config, 'waveshare_3')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 35, 25, 9)
|
fonts.setup(10, 8, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class WaveshareV4(DisplayImpl):
|
class WaveshareV4(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(WaveshareV4, self).__init__(config, 'waveshare_4')
|
super(WaveshareV4, self).__init__(config, 'waveshare_4')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -5,10 +5,9 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
from PIL import Image
|
from PIL import Image
|
||||||
|
|
||||||
|
|
||||||
class Waveshare213bV3(DisplayImpl):
|
class Waveshare2in13bV3(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare213bV3, self).__init__(config, 'waveshare2in13b_v3')
|
super(Waveshare2in13bV3, self).__init__(config, 'waveshare2in13b_v3')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
if self.config['color'] == 'black':
|
if self.config['color'] == 'black':
|
||||||
|
@ -8,7 +8,6 @@ from PIL import Image
|
|||||||
class Waveshare213bV4(DisplayImpl):
|
class Waveshare213bV4(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare213bV4, self).__init__(config, 'waveshare213inb_v4')
|
super(Waveshare213bV4, self).__init__(config, 'waveshare213inb_v4')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
if self.config['color'] == 'black':
|
if self.config['color'] == 'black':
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare213bc(DisplayImpl):
|
class Waveshare213bc(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare213bc, self).__init__(config, 'waveshare2in13bc')
|
super(Waveshare213bc, self).__init__(config, 'waveshare2in13bc')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
fonts.setup(10, 8, 10, 25, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare213d(DisplayImpl):
|
class Waveshare213d(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare213d, self).__init__(config, 'waveshare2in13d')
|
super(Waveshare213d, self).__init__(config, 'waveshare2in13d')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 8, 10, 25, 25, 9)
|
fonts.setup(10, 8, 10, 25, 25, 9)
|
||||||
|
@ -4,10 +4,9 @@ import pwnagotchi.ui.fonts as fonts
|
|||||||
from pwnagotchi.ui.hw.base import DisplayImpl
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
class WaveshareV4(DisplayImpl):
|
class Waveshare2in13g(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(WaveshareV4, self).__init__(config, 'waveshare_4')
|
super(Waveshare2in13g, self).__init__(config, 'waveshare2in13g')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
@ -32,15 +31,15 @@ class WaveshareV4(DisplayImpl):
|
|||||||
return self._layout
|
return self._layout
|
||||||
|
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
logging.info("initializing waveshare v4 display")
|
logging.info("initializing waveshare v2in13g display")
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v4.epd2in13_V4 import EPD
|
from pwnagotchi.ui.hw.libs.waveshare.v2in13g.epd2in13g import EPD
|
||||||
self._display = EPD()
|
self._display = EPD()
|
||||||
self._display.init()
|
self._display.init()
|
||||||
self._display.Clear(0xFF)
|
self._display.Clear()
|
||||||
|
|
||||||
def render(self, canvas):
|
def render(self, canvas):
|
||||||
buf = self._display.getbuffer(canvas)
|
buf = self._display.getbuffer(canvas)
|
||||||
self._display.displayPartial(buf)
|
self._display.display(buf)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
self._display.Clear(0xFF)
|
self._display.Clear()
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare2in36g(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare2in36g, self).__init__(config, 'waveshare2in36g')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 168
|
||||||
|
self._layout['height'] = 296
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 168, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 168, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 2.36g inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v2in36g.epd2in36g import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare2in66(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare2in66, self).__init__(config, 'waveshare2in66')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 152
|
||||||
|
self._layout['height'] = 296
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 152, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 152, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 2.66 inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v2in66.epd2in66 import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init(0)
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
45
pwnagotchi/ui/hw/waveshare2in66b.py
Normal file
45
pwnagotchi/ui/hw/waveshare2in66b.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare2in66b(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare2in66b, self).__init__(config, 'waveshare2in66b')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 152
|
||||||
|
self._layout['height'] = 296
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 152, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 152, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 2.66b inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v2in66b.epd2in66b import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
buf = self._display.getbuffer(canvas)
|
||||||
|
self._display.display(buf, None)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
44
pwnagotchi/ui/hw/waveshare2in66g.py
Normal file
44
pwnagotchi/ui/hw/waveshare2in66g.py
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare2in66g(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare2in66g, self).__init__(config, 'waveshare2in66g')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 152
|
||||||
|
self._layout['height'] = 296
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 152, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 152, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 2.66 inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v2in66g.epd2in66g import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare27inch(DisplayImpl):
|
class Waveshare27inch(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare27inch, self).__init__(config, 'waveshare2in7')
|
super(Waveshare27inch, self).__init__(config, 'waveshare2in7')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
@ -32,15 +31,15 @@ class Waveshare27inch(DisplayImpl):
|
|||||||
return self._layout
|
return self._layout
|
||||||
|
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
logging.info("initializing waveshare V1 2.7 inch display")
|
logging.info("initializing waveshare 2.7 V1 inch display")
|
||||||
from pwnagotchi.ui.hw.libs.waveshare.v2in7.epd2in7 import EPD
|
from pwnagotchi.ui.hw.libs.waveshare.v2in7.epd2in7 import EPD
|
||||||
self._display = EPD()
|
self._display = EPD()
|
||||||
self._display.init()
|
self._display.init()
|
||||||
self._display.Clear(0xFF)
|
self._display.Clear()
|
||||||
|
|
||||||
def render(self, canvas):
|
def render(self, canvas):
|
||||||
buf = self._display.getbuffer(canvas)
|
buf = self._display.getbuffer(canvas)
|
||||||
self._display.display(buf)
|
self._display.display(buf)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
self._display.Clear(0xff)
|
self._display.Clear()
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare27inchV2(DisplayImpl):
|
class Waveshare27inchV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare27inchV2, self).__init__(config, 'waveshare2in7_v2')
|
super(Waveshare27inchV2, self).__init__(config, 'waveshare2in7_v2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare27b(DisplayImpl):
|
class Waveshare27b(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare27b, self).__init__(config, 'waveshare2in7b')
|
super(Waveshare27b, self).__init__(config, 'waveshare2in7b')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare27bV2(DisplayImpl):
|
class Waveshare27bV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare27bV2, self).__init__(config, 'waveshare2in7b_v2')
|
super(Waveshare27bV2, self).__init__(config, 'waveshare2in7b_v2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare29inch(DisplayImpl):
|
class Waveshare29inch(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare29inch, self).__init__(config, 'waveshare2in9')
|
super(Waveshare29inch, self).__init__(config, 'waveshare2in9')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare29inchV2(DisplayImpl):
|
class Waveshare29inchV2(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare29inchV2, self).__init__(config, 'waveshare2in9_v2')
|
super(Waveshare29inchV2, self).__init__(config, 'waveshare2in9_v2')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare29bV3(DisplayImpl):
|
class Waveshare29bV3(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare29bV3, self).__init__(config, 'waveshare2in9b_v3')
|
super(Waveshare29bV3, self).__init__(config, 'waveshare2in9b_v3')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -7,7 +7,6 @@ from pwnagotchi.ui.hw.base import DisplayImpl
|
|||||||
class Waveshare29bV4(DisplayImpl):
|
class Waveshare29bV4(DisplayImpl):
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
super(Waveshare29bV4, self).__init__(config, 'waveshare2in9b_v4')
|
super(Waveshare29bV4, self).__init__(config, 'waveshare2in9b_v4')
|
||||||
self._display = None
|
|
||||||
|
|
||||||
def layout(self):
|
def layout(self):
|
||||||
fonts.setup(10, 9, 10, 35, 25, 9)
|
fonts.setup(10, 9, 10, 35, 25, 9)
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
|
import pwnagotchi.ui.fonts as fonts
|
||||||
|
from pwnagotchi.ui.hw.base import DisplayImpl
|
||||||
|
|
||||||
|
|
||||||
|
class Waveshare2in9bc(DisplayImpl):
|
||||||
|
def __init__(self, config):
|
||||||
|
super(Waveshare2in9bc, self).__init__(config, 'waveshare2in9bc')
|
||||||
|
|
||||||
|
def layout(self):
|
||||||
|
fonts.setup(10, 8, 10, 18, 25, 9)
|
||||||
|
self._layout['width'] = 128
|
||||||
|
self._layout['height'] = 296
|
||||||
|
self._layout['face'] = (0, 43)
|
||||||
|
self._layout['name'] = (0, 14)
|
||||||
|
self._layout['channel'] = (0, 0)
|
||||||
|
self._layout['aps'] = (0, 71)
|
||||||
|
self._layout['uptime'] = (0, 25)
|
||||||
|
self._layout['line1'] = [0, 12, 128, 12]
|
||||||
|
self._layout['line2'] = [0, 116, 128, 116]
|
||||||
|
self._layout['friend_face'] = (12, 88)
|
||||||
|
self._layout['friend_name'] = (1, 103)
|
||||||
|
self._layout['shakes'] = (26, 117)
|
||||||
|
self._layout['mode'] = (0, 117)
|
||||||
|
self._layout['status'] = {
|
||||||
|
'pos': (65, 26),
|
||||||
|
'font': fonts.status_font(fonts.Small),
|
||||||
|
'max': 12
|
||||||
|
}
|
||||||
|
return self._layout
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
logging.info("initializing waveshare 2,9bc inch lcd display")
|
||||||
|
from pwnagotchi.ui.hw.libs.waveshare.v2in9bc.epd2in9bc import EPD
|
||||||
|
self._display = EPD()
|
||||||
|
self._display.init()
|
||||||
|
self._display.Clear()
|
||||||
|
|
||||||
|
def render(self, canvas):
|
||||||
|
self._display.display(canvas)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._display.Clear()
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user