Compare commits
20 Commits
v.2.1
...
900b1f197e
| Author | SHA1 | Date | |
|---|---|---|---|
| 900b1f197e | |||
| fb91bd3b30 | |||
| 8b9a708cb4 | |||
| 6afc62a87f | |||
| d4cc8f2616 | |||
| 333bb3b995 | |||
| 77db85cc71 | |||
| 1c6060d8d5 | |||
|
|
8da056350a | ||
| 0ab3297054 | |||
|
|
6ac34770ff | ||
| 45e95d0e2b | |||
| d2d954b25f | |||
| 72144f76c6 | |||
| 755664230a | |||
| aae46e4b1e | |||
| 14497e21f2 | |||
| 34e44f4b3c | |||
| 53f5984eb5 | |||
| 8acc18c7fe |
1
.gitignore
vendored
@@ -2,6 +2,7 @@
|
||||
*.glade#
|
||||
*.txt
|
||||
*.json
|
||||
*.tar.gz
|
||||
extracted_images/
|
||||
selected/
|
||||
__pycache__
|
||||
|
||||
21
CHANGELOG
@@ -1,3 +1,24 @@
|
||||
# Version 2.5
|
||||
- Switch from libheif to imagemagick, HEIC conversion is now possible on Arch and Fedora!
|
||||
- Three new image sets: desert, island, lake
|
||||
|
||||
# Version 2.4
|
||||
- Adding Login window support
|
||||
- Adding Russian translation (Thanks to blogdron)
|
||||
- Adding Hungarian translation (Thanks to vajdao)
|
||||
- Bugfix: Time period calculation on locations near the dateline, like Japan
|
||||
|
||||
# Version 2.3
|
||||
- Adding Dutch translation
|
||||
- Adding German translation
|
||||
- Adding Spanish translation (Thanks to rcalixte)
|
||||
|
||||
# Version 2.2
|
||||
- Bugfixes
|
||||
- Migrate code to MVVM pattern
|
||||
- Adding option to change the location provider
|
||||
- Adding example image for dynamic background color
|
||||
|
||||
# Version 2.1
|
||||
- Bugfixes
|
||||
- Smaller UI for displays with reduced resolution (< 1000px height)
|
||||
|
||||
62
README.md
@@ -1,5 +1,5 @@
|
||||
# Cinnamon Dynamic Wallpaper
|
||||
<img src="cinnamon-dynamic-wallpaper@TobiZog/5.4/icons/icon.svg" alt="drawing" width="200" style="margin-left:auto; margin-right:auto; width:50%; display:block"/>
|
||||
<img src="cinnamon-dynamic-wallpaper@TobiZog/5.4/res/icons/icon.svg" alt="drawing" width="200" style="margin-left:auto; margin-right:auto; width:50%; display:block"/>
|
||||
|
||||

|
||||
|
||||
@@ -11,17 +11,18 @@ This extension switches the background image of your Cinnamon desktop multiple t
|
||||
- 10 day periods
|
||||
- HEIF converter
|
||||
- Image configuration assistent with simple one-click setup for image choices
|
||||
- Online location estimation or offline with manual latitude and longitude input
|
||||
- Online location estimation (three provider) or offline with manual latitude and longitude input
|
||||
- Time periods individual configured by user
|
||||
- Offline sun angles estimation
|
||||
- Image stretching over multiple displays or repeat image for every display
|
||||
- Show image on lock screen
|
||||
- Creating a color gradient based on the current wallpaper for images which not fill the whole screen
|
||||
|
||||
### Tested Cinnamon versions
|
||||
- 5.4 (Mint 21)
|
||||
- 5.6 (Mint 21.1)
|
||||
- 5.8 (Mint 21.2)
|
||||
- 6.0 (Mint 21.3)
|
||||
- 6.2 (Mint 22)
|
||||
|
||||
### Technology
|
||||
- `JavaScript`
|
||||
@@ -43,7 +44,7 @@ This extension switches the background image of your Cinnamon desktop multiple t
|
||||
3. Search and download it
|
||||
|
||||
### From the repo
|
||||
1. Download the Repository
|
||||
1. Download the latest from the Releases page on GitHub: https://github.com/TobiZog/cinnamon-dynamic-wallpaper/releases
|
||||
2. Extract the files
|
||||
3. Copy the folder `cinnamon-dynamic-wallpaper@TobiZog` to `~/.local/share/cinnamon/extensions/`
|
||||
---
|
||||
@@ -60,18 +61,61 @@ All configuration will be handled there. You can choose between included image s
|
||||
|
||||

|
||||
|
||||
---
|
||||
## Troubleshooting
|
||||
### General
|
||||
At first: Check if the extension is installed AND activated (check symbol on the left in Extension window).
|
||||
|
||||

|
||||
|
||||
Many errors on Linux Mint/Cinnamon Desktop will be printed to Melange. You can open it by pressing `Super Key` + `L`.
|
||||
|
||||
### The Preference Window doesn't open!
|
||||
Go to your home directory, open the terminal. Execute the command:
|
||||
|
||||
```
|
||||
python3 .local/share/cinnamon/extensions/cinnamon-dynamic-wallpaper@TobiZog/5.4/src/main.py
|
||||
```
|
||||
|
||||
This will manually start the window. If there are any problems, the terminal will show it. Use these informations to fix the problem or add them to the Issue.
|
||||
|
||||
### The extension don't change the wallpaper
|
||||
Go to your home directory, open the terminal. Execute the command:
|
||||
|
||||
```
|
||||
python3 .local/share/cinnamon/extensions/cinnamon-dynamic-wallpaper@TobiZog/5.4/src/main.py loop
|
||||
```
|
||||
|
||||
It executes the loop-function which handles the wallpaper change. Terminal will show errors. Use these informations to fix the problem or add them to the Issue.
|
||||
|
||||
---
|
||||
## Contribute
|
||||
### Translation
|
||||
You want to contribute a language which isn't supported yet? Here is how to do:
|
||||
|
||||
1. Fork the cinnamon-spices-extensions project: https://github.com/linuxmint/cinnamon-spices-extensions
|
||||
2. Pull the repository
|
||||
3. Open `cinnamon-dynamic-wallpaper@TobiZog/files/cinnamon-dynamic-wallpaper@TobiZog/po/cinnamon-dynamic-wallpaper@TobiZog.pot` with a po-tool like poedit (https://poedit.net/).
|
||||
4. Create a new translation in your language
|
||||
5. Push the changes to your repository
|
||||
6. Create a pull request
|
||||
|
||||
---
|
||||
## Included image sets
|
||||
The image sets are from https://github.com/adi1090x/dynamic-wallpaper
|
||||
|
||||
| Aurora | Beach | Bitday |
|
||||
| ------ | ----- | ------ |
|
||||
|  |  |  |
|
||||
|  |  |  |
|
||||
|
||||
| Cliffs | Earth | Gradient |
|
||||
| -------- | --------- | ------ |
|
||||
|  |  |  |
|
||||
| Cliffs | Desert | Earth |
|
||||
| -------- | --------- | --------- |
|
||||
|  |  |  |
|
||||
|
||||
| Gradient | Island | Lake |
|
||||
| --- | --- | --- |
|
||||
 |  |  |
|
||||
|
||||
| Lakeside | Mountains | Sahara |
|
||||
| --------- | ------ | ------ |
|
||||
|  |  |  |
|
||||
|  |  |  |
|
||||
@@ -17,7 +17,8 @@ const Gio = imports.gi.Gio;
|
||||
const MessageTray = imports.ui.messageTray;
|
||||
const St = imports.gi.St;
|
||||
const Main = imports.ui.main;
|
||||
|
||||
const Gettext = imports.gettext;
|
||||
const GLib = imports.gi.GLib;
|
||||
|
||||
|
||||
/******************** Constants ********************/
|
||||
@@ -44,6 +45,15 @@ function CinnamonDynamicWallpaperExtension(uuid) {
|
||||
}
|
||||
|
||||
|
||||
function _(str) {
|
||||
let customTranslation = Gettext.dgettext(UUID, str);
|
||||
if (customTranslation !== str) {
|
||||
return customTranslation;
|
||||
}
|
||||
return Gettext.gettext(str);
|
||||
}
|
||||
|
||||
|
||||
CinnamonDynamicWallpaperExtension.prototype = {
|
||||
|
||||
/******************** Lifecycle ********************/
|
||||
@@ -56,19 +66,33 @@ CinnamonDynamicWallpaperExtension.prototype = {
|
||||
_init: function(uuid) {
|
||||
this.settings = new Settings.ExtensionSettings(this, uuid);
|
||||
|
||||
Gettext.bindtextdomain(UUID, GLib.get_home_dir() + '/.local/share/locale');
|
||||
|
||||
// Check for the first startup
|
||||
if (this.settings.getValue("first_start")) {
|
||||
|
||||
// Welcome notification
|
||||
this.showNotification("Welcome to Cinnamon Dynamic Wallpaper",
|
||||
"Check the preferences to choose a dynamic wallpaper", true)
|
||||
this.showNotification(_("Welcome to Cinnamon Dynamic Wallpaper"),
|
||||
_("Check the preferences to choose a dynamic wallpaper"), true)
|
||||
|
||||
// Check for necessary software
|
||||
if (!find_program_in_path('convert')) {
|
||||
// Run on Ubuntu/Debian based distros with APT package manager
|
||||
if(GLib.find_program_in_path("apturl")) {
|
||||
Util.spawnCommandLine("apturl apt://imagemagick");
|
||||
} else {
|
||||
// Notification on other distros
|
||||
this.showNotification(_("imagemagick is not installed"),
|
||||
_("Please install the package manually for the full range of functions"), true)
|
||||
}
|
||||
}
|
||||
|
||||
// Hide the notification on system restart
|
||||
this.settings.setValue("first_start", false)
|
||||
this.settings.setValue("source_folder", DIRECTORY["path"] + "/images/included_image_sets/lakeside/")
|
||||
this.settings.setValue("source_folder", DIRECTORY["path"] + "/res/images/included_image_sets/lakeside/")
|
||||
}
|
||||
|
||||
// Start the main loop, checks in fixed time periods the
|
||||
// Start the main loop, checks in fixed time periods
|
||||
this._loop()
|
||||
},
|
||||
|
||||
@@ -96,10 +120,10 @@ CinnamonDynamicWallpaperExtension.prototype = {
|
||||
_loop: function () {
|
||||
if (looping) {
|
||||
try {
|
||||
Util.spawnCommandLine("/usr/bin/env python3 " + DIRECTORY.path + "/loop.py")
|
||||
Util.spawnCommandLine("/usr/bin/env python3 " + DIRECTORY.path + "/src/main.py loop")
|
||||
} catch(e) {
|
||||
this.showNotification("Error!",
|
||||
"Cinnamon Dynamic Wallpaper got an error while running the loop script. Please create an issue on GitHub.")
|
||||
this.showNotification(_("Error!"),
|
||||
_("Cinnamon Dynamic Wallpaper got an error while running the loop script. Please create an issue on GitHub."))
|
||||
}
|
||||
|
||||
// Refresh every 60 seconds
|
||||
@@ -127,8 +151,7 @@ CinnamonDynamicWallpaperExtension.prototype = {
|
||||
notification.addButton("open-settings", _("Open settings"));
|
||||
|
||||
notification.connect("action-invoked", () =>
|
||||
Util.spawnCommandLine("/usr/bin/env python3 " +
|
||||
DIRECTORY.path + "/preferences.py"));
|
||||
Util.spawnCommandLine("/usr/bin/env python3 " + DIRECTORY.path + "/src/main.py"));
|
||||
}
|
||||
|
||||
// Put all together
|
||||
@@ -159,11 +182,6 @@ function init(extensionMeta) {
|
||||
* @returns The extension object
|
||||
*/
|
||||
function enable() {
|
||||
// Check for necessary software
|
||||
if (!find_program_in_path('heif-convert')) {
|
||||
Util.spawnCommandLine("apturl apt://libheif-examples");
|
||||
}
|
||||
|
||||
return extension;
|
||||
}
|
||||
|
||||
|
||||
@@ -1 +1 @@
|
||||
icons/icon.svg
|
||||
res/icons/icon.svg
|
||||
@@ -1,110 +0,0 @@
|
||||
#!/usr/bin/python3
|
||||
|
||||
from scripts.cinnamon_pref_handler import *
|
||||
from scripts.suntimes import *
|
||||
from datetime import datetime, time
|
||||
from enums.PeriodSourceEnum import *
|
||||
from scripts.location import *
|
||||
from gi.repository import Gio
|
||||
from PIL import Image
|
||||
|
||||
|
||||
class Loop():
|
||||
def __init__(self) -> None:
|
||||
self.prefs = Cinnamon_Pref_Handler()
|
||||
|
||||
self.suntimes = Suntimes()
|
||||
self.location = Location()
|
||||
self.background_settings = Gio.Settings.new("org.cinnamon.desktop.background")
|
||||
|
||||
# Position should estimate by network
|
||||
if self.prefs.period_source == PeriodSourceEnum.NETWORKLOCATION:
|
||||
current_location = self.location.get_location()
|
||||
|
||||
self.suntimes.calc_suntimes(float(current_location["latitude"]), float(current_location["longitude"]))
|
||||
self.start_times = self.suntimes.day_periods
|
||||
|
||||
# Position is given by user
|
||||
elif self.prefs.period_source == PeriodSourceEnum.CUSTOMLOCATION:
|
||||
self.suntimes.calc_suntimes(float(self.prefs.latitude_custom), float(self.prefs.longitude_custom))
|
||||
self.start_times = self.suntimes.day_periods
|
||||
|
||||
# No position, concrete times
|
||||
else:
|
||||
def string_to_time_converter(raw_str: str) -> time:
|
||||
hour = raw_str[0:raw_str.find(":")]
|
||||
minute = raw_str[raw_str.find(":") + 1:]
|
||||
|
||||
return time(hour=int(hour), minute=int(minute))
|
||||
|
||||
self.start_times = [
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[0]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[1]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[2]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[3]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[4]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[5]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[6]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[7]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[8]),
|
||||
string_to_time_converter(self.prefs.period_custom_start_time[9])
|
||||
]
|
||||
|
||||
|
||||
def exchange_image(self):
|
||||
""" Replace the desktop image
|
||||
"""
|
||||
# Get the time of day
|
||||
time_now = time(datetime.now().hour, datetime.now().minute)
|
||||
|
||||
# Assign the last image as fallback
|
||||
self.current_image_uri = self.prefs.source_folder + self.prefs.period_images[9]
|
||||
|
||||
for i in range(0, 9):
|
||||
# Replace the image URI, if it's not the last time period of the day
|
||||
if self.start_times[i] <= time_now and time_now < self.start_times[i + 1]:
|
||||
self.current_image_uri = self.prefs.source_folder + self.prefs.period_images[i]
|
||||
break
|
||||
|
||||
# Set the background
|
||||
self.background_settings['picture-uri'] = "file://" + self.current_image_uri
|
||||
|
||||
# Set background stretching
|
||||
self.background_settings['picture-options'] = self.prefs.picture_aspect
|
||||
|
||||
self.set_background_gradient()
|
||||
|
||||
|
||||
def set_background_gradient(self):
|
||||
""" Setting a gradient background to hide images, which are not high enough
|
||||
"""
|
||||
# Load the image
|
||||
try:
|
||||
im = Image.open(self.current_image_uri)
|
||||
pix = im.load()
|
||||
|
||||
# Width and height of the current setted image
|
||||
width, height = im.size
|
||||
|
||||
# Color of the top and bottom pixel in the middle of the image
|
||||
top_color = pix[width / 2,0]
|
||||
bottom_color = pix[width / 2, height - 1]
|
||||
|
||||
# Create the gradient
|
||||
self.background_settings['color-shading-type'] = "vertical"
|
||||
|
||||
if self.prefs.dynamic_background_color:
|
||||
self.background_settings['primary-color'] = f"#{top_color[0]:x}{top_color[1]:x}{top_color[2]:x}"
|
||||
self.background_settings['secondary-color'] = f"#{bottom_color[0]:x}{bottom_color[1]:x}{bottom_color[2]:x}"
|
||||
else:
|
||||
self.background_settings['primary-color'] = "#000000"
|
||||
self.background_settings['secondary-color'] = "#000000"
|
||||
except:
|
||||
self.background_settings['primary-color'] = "#000000"
|
||||
self.background_settings['secondary-color'] = "#000000"
|
||||
|
||||
|
||||
# Needed for JavaScript
|
||||
if __name__ == "__main__":
|
||||
l = Loop()
|
||||
l.exchange_image()
|
||||
@@ -1,642 +0,0 @@
|
||||
#!/usr/bin/python3
|
||||
|
||||
############################################################
|
||||
# Imports #
|
||||
############################################################
|
||||
|
||||
# Packages
|
||||
import gi, os, subprocess, time
|
||||
from datetime import timedelta
|
||||
|
||||
# Local scripts
|
||||
from scripts import cinnamon_pref_handler, dialogs, display, images, location, suntimes, time_bar_chart, ui
|
||||
from loop import *
|
||||
from enums.ImageSourceEnum import ImageSourceEnum
|
||||
from enums.PeriodSourceEnum import PeriodSourceEnum
|
||||
|
||||
gi.require_version("Gtk", "3.0")
|
||||
from gi.repository import Gtk, GdkPixbuf
|
||||
|
||||
|
||||
# Global definitions
|
||||
PREFERENCES_URI = os.path.dirname(os.path.abspath(__file__))
|
||||
GLADE_URI = PREFERENCES_URI + "/preferences.glade"
|
||||
|
||||
|
||||
class Preferences:
|
||||
""" Preference window class
|
||||
"""
|
||||
|
||||
############################################################
|
||||
# Lifecycle #
|
||||
############################################################
|
||||
|
||||
def __init__(self) -> None:
|
||||
# Objects from external scripts
|
||||
self.prefs = cinnamon_pref_handler.Cinnamon_Pref_Handler()
|
||||
self.dialogs = dialogs.Dialogs()
|
||||
self.display = display.Display()
|
||||
self.images = images.Images()
|
||||
self.location = location.Location()
|
||||
self.suntimes = suntimes.Suntimes()
|
||||
self.time_bar_chart = time_bar_chart.Time_Bar_Chart()
|
||||
|
||||
# Glade
|
||||
self.builder = Gtk.Builder()
|
||||
self.builder.add_from_file(GLADE_URI)
|
||||
self.builder.connect_signals(self)
|
||||
|
||||
self.ui = ui.UI(self.builder)
|
||||
|
||||
|
||||
# Local Config
|
||||
self.smaller_ui_height = 1000
|
||||
|
||||
|
||||
def show(self):
|
||||
""" Display the window to the screen
|
||||
"""
|
||||
window = self.builder.get_object("window_main")
|
||||
window.show_all()
|
||||
|
||||
# Load from preferences
|
||||
if self.prefs.image_source == ImageSourceEnum.IMAGESET:
|
||||
self.ui.tb_image_set.set_active(True)
|
||||
elif self.prefs.image_source == ImageSourceEnum.HEICFILE:
|
||||
self.ui.tb_heic_file.set_active(True)
|
||||
elif self.prefs.image_source == ImageSourceEnum.SOURCEFOLDER:
|
||||
self.ui.tb_source_folder.set_active(True)
|
||||
|
||||
|
||||
# Remove icons in the ToggleButtons if the screen height resolution < 1000 px
|
||||
if self.display.get_screen_height() < self.smaller_ui_height:
|
||||
self.ui.img_tb_image_set.clear()
|
||||
self.ui.img_tb_heic_file.clear()
|
||||
self.ui.img_tb_source_folder.clear()
|
||||
self.ui.img_tb_network_location.clear()
|
||||
self.ui.img_tb_custom_location.clear()
|
||||
self.ui.img_tb_time_periods.clear()
|
||||
|
||||
|
||||
picture_aspects = ["centered", "scaled", "stretched", "zoom", "spanned"]
|
||||
self.ui.add_items_to_combo_box(self.ui.cb_picture_aspect, picture_aspects)
|
||||
self.ui.set_active_combobox_item(self.ui.cb_picture_aspect, self.prefs.picture_aspect)
|
||||
|
||||
self.ui.sw_dynamic_background_color.set_active(self.prefs.dynamic_background_color)
|
||||
|
||||
|
||||
if self.prefs.period_source == PeriodSourceEnum.NETWORKLOCATION:
|
||||
self.ui.tb_network_location.set_active(True)
|
||||
elif self.prefs.period_source == PeriodSourceEnum.CUSTOMLOCATION:
|
||||
self.ui.tb_custom_location.set_active(True)
|
||||
elif self.prefs.period_source == PeriodSourceEnum.CUSTOMTIMEPERIODS:
|
||||
self.ui.tb_time_periods.set_active(True)
|
||||
|
||||
|
||||
# Time diagram
|
||||
try:
|
||||
self.refresh_chart()
|
||||
except:
|
||||
pass
|
||||
|
||||
# Show the main window
|
||||
Gtk.main()
|
||||
|
||||
|
||||
def on_destroy(self, *args):
|
||||
""" Lifecycle handler when window will be destroyed
|
||||
"""
|
||||
Gtk.main_quit()
|
||||
|
||||
|
||||
|
||||
############################################################
|
||||
# Local methods #
|
||||
############################################################
|
||||
|
||||
def refresh_chart(self):
|
||||
""" Recomputes both time bar charts and load them to the UI
|
||||
"""
|
||||
# Stores the start times of the periods in minutes since midnight
|
||||
time_periods_min = []
|
||||
|
||||
if self.prefs.period_source == PeriodSourceEnum.CUSTOMTIMEPERIODS:
|
||||
for i in range(0, 10):
|
||||
time_str = self.prefs.period_custom_start_time[i]
|
||||
|
||||
time_periods_min.append(int(time_str[0:2]) * 60 + int(time_str[3:5]))
|
||||
else:
|
||||
if self.prefs.period_source == PeriodSourceEnum.NETWORKLOCATION:
|
||||
self.suntimes.calc_suntimes(float(self.prefs.latitude_auto), float(self.prefs.longitude_auto))
|
||||
else:
|
||||
self.suntimes.calc_suntimes(float(self.ui.etr_latitude.get_text()), float(self.ui.etr_longitude.get_text()))
|
||||
|
||||
|
||||
# Get all time periods. Store the minutes to the list and print the values to the text views
|
||||
for i in range(0, 10):
|
||||
time_range_now = self.suntimes.day_periods[i]
|
||||
|
||||
if i != 9:
|
||||
time_range_next = self.suntimes.day_periods[i + 1]
|
||||
else:
|
||||
time_range_next = time(hour=23, minute=59)
|
||||
|
||||
self.ui.etr_periods[i].set_text(
|
||||
str(time_range_now.hour).rjust(2, '0') + ":" + str(time_range_now.minute).rjust(2, '0') + \
|
||||
" - " + str(time_range_next.hour).rjust(2, '0') + ":" + str(time_range_next.minute).rjust(2, '0'))
|
||||
|
||||
time_periods_min.append(time_range_now.hour * 60 + time_range_now.minute)
|
||||
|
||||
# Create time bar
|
||||
# Reduce size for small displays
|
||||
if self.display.get_screen_height() < self.smaller_ui_height:
|
||||
bar_width = 1150
|
||||
bar_height = 110
|
||||
else:
|
||||
bar_width = 1300
|
||||
bar_height = 150
|
||||
|
||||
self.time_bar_chart.create_bar_chart_with_polylines(PREFERENCES_URI, bar_width, bar_height, time_periods_min)
|
||||
self.time_bar_chart.create_bar_chart(PREFERENCES_URI, bar_width, bar_height, time_periods_min)
|
||||
|
||||
# Load to the views
|
||||
pixbuf = GdkPixbuf.Pixbuf.new_from_file(PREFERENCES_URI + "/time_bar_polylines.svg")
|
||||
self.ui.img_bar_images.set_from_pixbuf(pixbuf)
|
||||
|
||||
pixbuf2 = GdkPixbuf.Pixbuf.new_from_file(PREFERENCES_URI + "/time_bar.svg")
|
||||
self.ui.img_bar_times.set_from_pixbuf(pixbuf2)
|
||||
|
||||
|
||||
def load_image_options_to_combo_boxes(self, options: list):
|
||||
""" Add a list of Strings to all image option comboboxes
|
||||
|
||||
Args:
|
||||
options (list): All possible options
|
||||
"""
|
||||
options.insert(0, "")
|
||||
|
||||
for combobox in self.ui.cb_periods:
|
||||
self.ui.add_items_to_combo_box(combobox, options)
|
||||
|
||||
|
||||
def load_image_to_preview(self, image_preview: Gtk.Image, image_src: str):
|
||||
""" Scales the image to a lower resoultion and put them into the time bar chart
|
||||
|
||||
Args:
|
||||
image_preview (Gtk.Image): Gtk Image where it will be displayed
|
||||
image_src (str): Absolute path to the image
|
||||
"""
|
||||
try:
|
||||
pixbuf = GdkPixbuf.Pixbuf.new_from_file(image_src)
|
||||
|
||||
screen_height = self.display.get_screen_height()
|
||||
|
||||
# Scaling the images smaller for screens
|
||||
if screen_height < self.smaller_ui_height:
|
||||
pixbuf = pixbuf.scale_simple(221, 128, GdkPixbuf.InterpType.BILINEAR)
|
||||
else:
|
||||
pixbuf = pixbuf.scale_simple(260, 150, GdkPixbuf.InterpType.BILINEAR)
|
||||
|
||||
image_preview.set_from_pixbuf(pixbuf)
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
############################################################
|
||||
# Callbacks #
|
||||
############################################################
|
||||
|
||||
## Image Configuration
|
||||
|
||||
# +-----------+-----------+---------------+
|
||||
# | Image Set | HEIC file | Source Folder |
|
||||
# +-----------+-----------+---------------+
|
||||
|
||||
def on_toggle_button_image_set_clicked(self, button: Gtk.ToggleButton):
|
||||
""" Clicked on ToggleButton "Image Set"
|
||||
|
||||
Args:
|
||||
button (Gtk.ToggleButton): Clicked ToggleButton
|
||||
"""
|
||||
if button.get_active():
|
||||
self.prefs.image_source = ImageSourceEnum.IMAGESET
|
||||
self.ui.tb_heic_file.set_active(False)
|
||||
self.ui.tb_source_folder.set_active(False)
|
||||
|
||||
self.ui.lbr_image_set.set_visible(True)
|
||||
self.ui.lbr_heic_file.set_visible(False)
|
||||
self.ui.lbr_source_folder.set_visible(False)
|
||||
|
||||
image_set_choices = [
|
||||
"aurora", "beach",
|
||||
"bitday", "cliffs",
|
||||
"earth", "gradient",
|
||||
"lakeside", "mountains",
|
||||
"sahara"
|
||||
]
|
||||
self.ui.add_items_to_combo_box(self.ui.cb_image_set, image_set_choices)
|
||||
|
||||
self.ui.set_active_combobox_item(self.ui.cb_image_set, self.prefs.selected_image_set)
|
||||
|
||||
for i, combobox in enumerate(self.ui.cb_periods):
|
||||
selected_image_name = self.prefs.period_images[i]
|
||||
self.ui.set_active_combobox_item(combobox, selected_image_name)
|
||||
|
||||
# Make the comboboxes invisible
|
||||
for combobox in self.ui.cb_periods:
|
||||
combobox.set_visible(False)
|
||||
|
||||
|
||||
def on_toggle_button_heic_file_clicked(self, button: Gtk.ToggleButton):
|
||||
""" Clicked on ToggleButton "Heic file"
|
||||
|
||||
Args:
|
||||
button (Gtk.ToggleButton): Clicked ToggleButton
|
||||
"""
|
||||
if button.get_active():
|
||||
self.prefs.image_source = ImageSourceEnum.HEICFILE
|
||||
self.ui.tb_image_set.set_active(False)
|
||||
self.ui.tb_source_folder.set_active(False)
|
||||
|
||||
self.ui.lbr_image_set.set_visible(False)
|
||||
self.ui.lbr_heic_file.set_visible(True)
|
||||
self.ui.lbr_source_folder.set_visible(False)
|
||||
|
||||
# Make the comboboxes visible
|
||||
for combobox in self.ui.cb_periods:
|
||||
combobox.set_visible(True)
|
||||
|
||||
# Load images from source folder
|
||||
files = self.images.get_images_from_folder(self.prefs.source_folder)
|
||||
|
||||
if len(files) != 0:
|
||||
self.load_image_options_to_combo_boxes(files)
|
||||
|
||||
# Load the values for the images from the preferences
|
||||
for i in range(0, 10):
|
||||
self.ui.set_active_combobox_item(self.ui.cb_periods[i], self.prefs.period_images[i])
|
||||
else:
|
||||
print("No image files!")
|
||||
|
||||
|
||||
def on_toggle_button_source_folder_clicked(self, button: Gtk.ToggleButton):
|
||||
""" Clicked on ToggleButton "Source Folder"
|
||||
|
||||
Args:
|
||||
button (Gtk.ToggleButton): Clicked ToggleButton
|
||||
"""
|
||||
if button.get_active():
|
||||
self.prefs.image_source = ImageSourceEnum.SOURCEFOLDER
|
||||
self.ui.tb_image_set.set_active(False)
|
||||
self.ui.tb_heic_file.set_active(False)
|
||||
|
||||
self.ui.lbr_image_set.set_visible(False)
|
||||
self.ui.lbr_heic_file.set_visible(False)
|
||||
self.ui.lbr_source_folder.set_visible(True)
|
||||
|
||||
# Make the comboboxes visible
|
||||
for combobox in self.ui.cb_periods:
|
||||
combobox.set_visible(True)
|
||||
|
||||
# Load the source folder to the view
|
||||
# This will update the comboboxes in the preview to contain the right items
|
||||
self.ui.lbl_source_folder.set_label(self.prefs.source_folder)
|
||||
|
||||
# Load files from saved source folder
|
||||
files = self.images.get_images_from_folder(self.prefs.source_folder)
|
||||
|
||||
if len(files) != 0:
|
||||
self.load_image_options_to_combo_boxes(files)
|
||||
|
||||
# Load the values for the images from the preferences
|
||||
for i in range(0, 10):
|
||||
self.ui.set_active_combobox_item(self.ui.cb_periods[i], self.prefs.period_images[i])
|
||||
else:
|
||||
print("No image files!")
|
||||
|
||||
|
||||
|
||||
# +------------------------------------+
|
||||
# | Select an image set | aurora ▼ |
|
||||
# +------------------------------------+
|
||||
|
||||
def on_cb_image_set_changed(self, combobox: Gtk.ComboBox):
|
||||
""" User select on of the included image sets
|
||||
|
||||
Args:
|
||||
combobox (Gtk.ComboBox): The used ComboBox
|
||||
"""
|
||||
tree_iter = combobox.get_active_iter()
|
||||
|
||||
if tree_iter is not None and self.prefs.image_source == ImageSourceEnum.IMAGESET:
|
||||
# Get the selected value
|
||||
model = combobox.get_model()
|
||||
selected_image_set = model[tree_iter][0]
|
||||
|
||||
# Store to the preferences
|
||||
self.prefs.selected_image_set = selected_image_set
|
||||
self.prefs.source_folder = os.path.abspath(os.path.join(PREFERENCES_URI, os.pardir)) + \
|
||||
"/5.4/images/included_image_sets/" + selected_image_set + "/"
|
||||
|
||||
# Load all possible options to the comboboxes
|
||||
image_names = self.images.get_images_from_folder(self.prefs.source_folder)
|
||||
self.load_image_options_to_combo_boxes(image_names)
|
||||
|
||||
# Image sets have the same names for the images:
|
||||
# 9.jpg = Period 0
|
||||
# 1.jpg = Period 1
|
||||
# 2.jpg = Period 2
|
||||
# and so on....
|
||||
for i in range(0, 10):
|
||||
self.ui.cb_periods[i].set_active(i + 1)
|
||||
|
||||
|
||||
# +----------------------------------------------+
|
||||
# | Select a heic file to import | (None) 📄 |
|
||||
# +----------------------------------------------+
|
||||
|
||||
def on_fc_heic_file_file_set(self, fc_button: Gtk.FileChooser):
|
||||
""" User has a heic file selected with the FileChooserDialog
|
||||
|
||||
Args:
|
||||
fc_button (Gtk.FileChooser): Parameter about the selected file
|
||||
"""
|
||||
# The the absolute path to the heic file
|
||||
file_path: str = fc_button.get_filename()
|
||||
|
||||
# Extract the heic file
|
||||
result = self.images.extract_heic_file(file_path)
|
||||
|
||||
# Update the preferences
|
||||
self.prefs.selected_image_set = ""
|
||||
self.prefs.source_folder = PREFERENCES_URI + "/images/extracted_images/"
|
||||
|
||||
# Load images only if the extraction was successfully
|
||||
if result:
|
||||
# Collect all extracted images and push them to the comboboxes
|
||||
image_names = self.images.get_images_from_folder(self.prefs.source_folder)
|
||||
self.load_image_options_to_combo_boxes(image_names)
|
||||
else:
|
||||
self.dialogs.message_dialog("Error during extraction")
|
||||
|
||||
|
||||
# +------------------------------------------------------------+
|
||||
# | Select a source folder | 📂 Open file selection dialog |
|
||||
# | /home/developer/Downloads/
|
||||
# +------------------------------------------------------------+
|
||||
|
||||
def on_btn_source_folder_clicked(self, button: Gtk.Button):
|
||||
""" Button to choose an image source folder was clicked
|
||||
|
||||
Args:
|
||||
button (Gtk.Button): The clicked button
|
||||
"""
|
||||
folder = self.dialogs.source_folder_dialog()
|
||||
files = self.images.get_images_from_folder(folder)
|
||||
|
||||
# Update the preferences
|
||||
self.prefs.selected_image_set = ""
|
||||
self.prefs.source_folder = folder + "/"
|
||||
|
||||
# Update the label
|
||||
self.ui.lbl_source_folder.set_label(folder)
|
||||
|
||||
# Update the image comboboxes
|
||||
self.load_image_options_to_combo_boxes(files)
|
||||
|
||||
# Load the values for the images from the preferences
|
||||
for i in range(0, 10):
|
||||
self.ui.cb_periods[i].set_active(0)
|
||||
|
||||
if len(files) == 1:
|
||||
self.dialogs.message_dialog("No image files found!")
|
||||
|
||||
|
||||
def on_cb_period_preview_changed(self, combobox: Gtk.ComboBox):
|
||||
""" User select an image from the ComboBox for the time period
|
||||
|
||||
Args:
|
||||
combobox (Gtk.ComboBox): The used ComboBox
|
||||
"""
|
||||
tree_iter = combobox.get_active_iter()
|
||||
|
||||
combobox_name = Gtk.Buildable.get_name(combobox)
|
||||
period_index = int(combobox_name[10:11])
|
||||
|
||||
if tree_iter is not None:
|
||||
# Get the selected value
|
||||
model = combobox.get_model()
|
||||
image_file_name = model[tree_iter][0]
|
||||
|
||||
# Store selection to preferences
|
||||
self.prefs.period_images[period_index] = image_file_name
|
||||
|
||||
# Build up image path
|
||||
image_path = self.prefs.source_folder + image_file_name
|
||||
|
||||
self.load_image_to_preview(self.ui.img_periods[period_index], image_path)
|
||||
|
||||
|
||||
## Location & Times
|
||||
|
||||
def on_toggle_button_network_location_clicked(self, button: Gtk.ToggleButton):
|
||||
""" User clicks on the ToggleButton for the network location
|
||||
|
||||
Args:
|
||||
button (Gtk.ToggleButton): Clicked ToggleButton
|
||||
"""
|
||||
if button.get_active():
|
||||
self.prefs.period_source = PeriodSourceEnum.NETWORKLOCATION
|
||||
self.ui.tb_custom_location.set_active(False)
|
||||
self.ui.tb_time_periods.set_active(False)
|
||||
|
||||
self.ui.lbr_network_location.set_visible(True)
|
||||
self.ui.lbr_current_location.set_visible(True)
|
||||
self.ui.lbr_custom_location_longitude.set_visible(False)
|
||||
self.ui.lbr_custom_location_latitude.set_visible(False)
|
||||
self.ui.lbr_time_periods.set_visible(False)
|
||||
|
||||
self.ui.spb_network_location_refresh_time.set_value(self.prefs.location_refresh_intervals)
|
||||
|
||||
|
||||
# Display the location in the UI
|
||||
current_location = self.location.get_location()
|
||||
self.ui.lb_current_location.set_text("Latitude: " + current_location["latitude"] + \
|
||||
", Longitude: " + current_location["longitude"])
|
||||
|
||||
# Store the location to the preferences
|
||||
self.prefs.latitude_auto = float(current_location["latitude"])
|
||||
self.prefs.longitude_auto = float(current_location["longitude"])
|
||||
|
||||
self.refresh_chart()
|
||||
|
||||
|
||||
def on_toggle_button_custom_location_clicked(self, button: Gtk.ToggleButton):
|
||||
if button.get_active():
|
||||
self.prefs.period_source = PeriodSourceEnum.CUSTOMLOCATION
|
||||
self.ui.tb_network_location.set_active(False)
|
||||
self.ui.tb_time_periods.set_active(False)
|
||||
|
||||
self.ui.lbr_network_location.set_visible(False)
|
||||
self.ui.lbr_current_location.set_visible(False)
|
||||
self.ui.lbr_custom_location_longitude.set_visible(True)
|
||||
self.ui.lbr_custom_location_latitude.set_visible(True)
|
||||
self.ui.lbr_time_periods.set_visible(False)
|
||||
|
||||
self.ui.etr_latitude.set_text(str(self.prefs.latitude_custom))
|
||||
self.ui.etr_longitude.set_text(str(self.prefs.longitude_custom))
|
||||
|
||||
|
||||
def on_toggle_button_time_periods_clicked(self, button: Gtk.ToggleButton):
|
||||
if button.get_active():
|
||||
self.prefs.period_source = PeriodSourceEnum.CUSTOMTIMEPERIODS
|
||||
self.ui.tb_network_location.set_active(False)
|
||||
self.ui.tb_custom_location.set_active(False)
|
||||
|
||||
self.ui.lbr_network_location.set_visible(False)
|
||||
self.ui.lbr_current_location.set_visible(False)
|
||||
self.ui.lbr_custom_location_longitude.set_visible(False)
|
||||
self.ui.lbr_custom_location_latitude.set_visible(False)
|
||||
self.ui.lbr_time_periods.set_visible(True)
|
||||
|
||||
|
||||
for i in range(0, 9):
|
||||
pref_value = self.prefs.period_custom_start_time[i + 1]
|
||||
time_parts = [int(pref_value[0:pref_value.find(":")]), int(pref_value[pref_value.find(":") + 1:])]
|
||||
|
||||
self.ui.spb_periods_hour[i].set_value(time_parts[0])
|
||||
self.ui.spb_periods_minute[i].set_value(time_parts[1])
|
||||
|
||||
|
||||
|
||||
def on_spb_period_value_changed(self, spin_button: Gtk.SpinButton):
|
||||
""" Callback if one of the time spinners (minute or hour) will be clicked
|
||||
|
||||
(1) (2) (3)
|
||||
Previous period Current period Next period
|
||||
12:34 - 14:40 14:41 - 16:20 16:21 - 17:30
|
||||
^
|
||||
Variable to change
|
||||
|
||||
Args:
|
||||
spin_button (Gtk.SpinButton): SpinButton which was changed
|
||||
"""
|
||||
spin_button_name = Gtk.Buildable.get_name(spin_button)
|
||||
index = int(spin_button_name[11:12]) - 1
|
||||
|
||||
# Determe time string and store to prefs
|
||||
time_current_start = datetime(2024,1,1, int(self.ui.spb_periods_hour[index].get_value()), int(self.ui.spb_periods_minute[index].get_value()))
|
||||
time_current_start_str = str(time_current_start.hour).rjust(2, '0') + ":" + str(time_current_start.minute).rjust(2, '0')
|
||||
|
||||
self.prefs.period_custom_start_time[index + 1] = time_current_start_str
|
||||
|
||||
|
||||
time_previous_end = time_current_start - timedelta(minutes=1)
|
||||
self.ui.lb_period_end[index].set_text(str(time_previous_end.hour).rjust(2, '0') + ":" + str(time_previous_end.minute).rjust(2, '0'))
|
||||
|
||||
|
||||
self.refresh_chart()
|
||||
|
||||
|
||||
def on_spb_network_location_refresh_time_changed(self, spin_button: Gtk.SpinButton):
|
||||
""" User changed the refresh time of network location estimation
|
||||
|
||||
Args:
|
||||
spin_button (Gtk.SpinButton): The used SpinButton
|
||||
"""
|
||||
self.prefs.location_refresh_intervals = spin_button.get_value()
|
||||
|
||||
|
||||
def on_etr_longitude_changed(self, entry: Gtk.Entry):
|
||||
""" User changes the value of the longitude Entry
|
||||
|
||||
Args:
|
||||
entry (Gtk.Entry): The manipulated Entry object
|
||||
"""
|
||||
try:
|
||||
self.prefs.longitude_custom = float(entry.get_text())
|
||||
self.refresh_chart()
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
def on_etr_latitude_changed(self, entry: Gtk.Entry):
|
||||
""" User changes the value of the latitude Entry
|
||||
|
||||
Args:
|
||||
entry (Gtk.Entry): The manipulated Entry object
|
||||
"""
|
||||
try:
|
||||
self.prefs.latitude_custom = float(entry.get_text())
|
||||
self.refresh_chart()
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
# Behaviour
|
||||
|
||||
def on_cb_picture_aspect_changed(self, combobox: Gtk.ComboBox):
|
||||
tree_iter = combobox.get_active_iter()
|
||||
|
||||
if tree_iter is not None:
|
||||
model = combobox.get_model()
|
||||
self.prefs.picture_aspect = model[tree_iter][0]
|
||||
|
||||
def on_sw_dynamic_background_color_state_set(self, switch: Gtk.Switch, state):
|
||||
self.prefs.dynamic_background_color = state
|
||||
|
||||
|
||||
# About
|
||||
|
||||
def on_cinnamon_spices_website_button_clicked(self, button: Gtk.Button):
|
||||
""" Callback for the button to navigate to the Cinnamon Spices web page of this project
|
||||
|
||||
Args:
|
||||
button (Gtk.Button): Button which was clicked
|
||||
"""
|
||||
subprocess.Popen(["xdg-open", "https://cinnamon-spices.linuxmint.com/extensions/view/97"])
|
||||
|
||||
|
||||
def on_github_website_button_clicked(self, button: Gtk.Button):
|
||||
""" Callback for the button to navigate to the GitHub web page of this project
|
||||
|
||||
Args:
|
||||
button (Gtk.Button): Button which was clicked
|
||||
"""
|
||||
subprocess.Popen(["xdg-open", "https://github.com/TobiZog/cinnamon-dynamic-wallpaper"])
|
||||
|
||||
|
||||
def on_create_issue_button_clicked(self, button):
|
||||
""" Callback for the button to navigate to the Issues page on GitHub of this project
|
||||
|
||||
Args:
|
||||
button (Gtk.Button): Button which was clicked
|
||||
"""
|
||||
subprocess.Popen(["xdg-open", "https://github.com/TobiZog/cinnamon-dynamic-wallpaper/issues/new"])
|
||||
|
||||
|
||||
def on_ok(self, *args):
|
||||
""" Callback for the OK button in the top bar
|
||||
"""
|
||||
try:
|
||||
self.on_apply()
|
||||
except:
|
||||
pass
|
||||
|
||||
# Close the window
|
||||
self.on_destroy()
|
||||
|
||||
|
||||
def on_apply(self, *args):
|
||||
""" Callback for the Apply button in the top bar
|
||||
"""
|
||||
# Store all values to the JSON file
|
||||
self.prefs.store_preferences()
|
||||
|
||||
# Use the new settings
|
||||
loop = Loop()
|
||||
loop.exchange_image()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
Preferences().show()
|
||||
|
Before Width: | Height: | Size: 9.1 KiB After Width: | Height: | Size: 9.1 KiB |
|
Before Width: | Height: | Size: 5.0 KiB After Width: | Height: | Size: 5.0 KiB |
|
After Width: | Height: | Size: 42 KiB |
|
Before Width: | Height: | Size: 233 KiB After Width: | Height: | Size: 233 KiB |
|
Before Width: | Height: | Size: 228 KiB After Width: | Height: | Size: 228 KiB |
|
Before Width: | Height: | Size: 246 KiB After Width: | Height: | Size: 246 KiB |
|
Before Width: | Height: | Size: 230 KiB After Width: | Height: | Size: 230 KiB |
|
Before Width: | Height: | Size: 236 KiB After Width: | Height: | Size: 236 KiB |
|
Before Width: | Height: | Size: 244 KiB After Width: | Height: | Size: 244 KiB |
|
Before Width: | Height: | Size: 229 KiB After Width: | Height: | Size: 229 KiB |
|
Before Width: | Height: | Size: 216 KiB After Width: | Height: | Size: 216 KiB |
|
Before Width: | Height: | Size: 176 KiB After Width: | Height: | Size: 176 KiB |
|
Before Width: | Height: | Size: 217 KiB After Width: | Height: | Size: 217 KiB |
|
Before Width: | Height: | Size: 371 KiB After Width: | Height: | Size: 371 KiB |
|
Before Width: | Height: | Size: 491 KiB After Width: | Height: | Size: 491 KiB |
|
Before Width: | Height: | Size: 513 KiB After Width: | Height: | Size: 513 KiB |
|
Before Width: | Height: | Size: 527 KiB After Width: | Height: | Size: 527 KiB |
|
Before Width: | Height: | Size: 526 KiB After Width: | Height: | Size: 526 KiB |
|
Before Width: | Height: | Size: 495 KiB After Width: | Height: | Size: 495 KiB |
|
Before Width: | Height: | Size: 455 KiB After Width: | Height: | Size: 455 KiB |
|
Before Width: | Height: | Size: 450 KiB After Width: | Height: | Size: 450 KiB |
|
Before Width: | Height: | Size: 352 KiB After Width: | Height: | Size: 352 KiB |
|
Before Width: | Height: | Size: 513 KiB After Width: | Height: | Size: 513 KiB |
|
Before Width: | Height: | Size: 1012 KiB After Width: | Height: | Size: 1012 KiB |
|
Before Width: | Height: | Size: 1.2 MiB After Width: | Height: | Size: 1.2 MiB |
|
Before Width: | Height: | Size: 695 KiB After Width: | Height: | Size: 695 KiB |
|
Before Width: | Height: | Size: 710 KiB After Width: | Height: | Size: 710 KiB |
|
Before Width: | Height: | Size: 1.1 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.1 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.0 MiB After Width: | Height: | Size: 1.0 MiB |
|
Before Width: | Height: | Size: 856 KiB After Width: | Height: | Size: 856 KiB |
|
Before Width: | Height: | Size: 1.1 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.5 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.5 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.5 MiB |
|
Before Width: | Height: | Size: 1.7 MiB After Width: | Height: | Size: 1.7 MiB |
|
Before Width: | Height: | Size: 1.7 MiB After Width: | Height: | Size: 1.7 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.5 MiB |
|
Before Width: | Height: | Size: 1.4 MiB After Width: | Height: | Size: 1.4 MiB |
|
After Width: | Height: | Size: 1.0 MiB |
|
After Width: | Height: | Size: 1.3 MiB |
|
After Width: | Height: | Size: 1.6 MiB |
|
After Width: | Height: | Size: 1.5 MiB |
|
After Width: | Height: | Size: 1.7 MiB |
@@ -0,0 +1 @@
|
||||
4.jpg
|
||||
|
After Width: | Height: | Size: 1.6 MiB |
|
After Width: | Height: | Size: 1.6 MiB |
|
After Width: | Height: | Size: 1.3 MiB |
|
Before Width: | Height: | Size: 104 KiB After Width: | Height: | Size: 104 KiB |
|
Before Width: | Height: | Size: 254 KiB After Width: | Height: | Size: 254 KiB |
|
Before Width: | Height: | Size: 634 KiB After Width: | Height: | Size: 634 KiB |
|
Before Width: | Height: | Size: 1.2 MiB After Width: | Height: | Size: 1.2 MiB |
|
Before Width: | Height: | Size: 1.2 MiB After Width: | Height: | Size: 1.2 MiB |
|
Before Width: | Height: | Size: 759 KiB After Width: | Height: | Size: 759 KiB |
|
Before Width: | Height: | Size: 528 KiB After Width: | Height: | Size: 528 KiB |
|
Before Width: | Height: | Size: 335 KiB After Width: | Height: | Size: 335 KiB |
|
Before Width: | Height: | Size: 195 KiB After Width: | Height: | Size: 195 KiB |
|
Before Width: | Height: | Size: 120 KiB After Width: | Height: | Size: 120 KiB |
|
Before Width: | Height: | Size: 163 KiB After Width: | Height: | Size: 163 KiB |
|
Before Width: | Height: | Size: 238 KiB After Width: | Height: | Size: 238 KiB |
|
Before Width: | Height: | Size: 241 KiB After Width: | Height: | Size: 241 KiB |
|
Before Width: | Height: | Size: 293 KiB After Width: | Height: | Size: 293 KiB |
|
Before Width: | Height: | Size: 183 KiB After Width: | Height: | Size: 183 KiB |
|
Before Width: | Height: | Size: 211 KiB After Width: | Height: | Size: 211 KiB |
|
Before Width: | Height: | Size: 210 KiB After Width: | Height: | Size: 210 KiB |
|
Before Width: | Height: | Size: 249 KiB After Width: | Height: | Size: 249 KiB |
|
Before Width: | Height: | Size: 182 KiB After Width: | Height: | Size: 182 KiB |
|
Before Width: | Height: | Size: 109 KiB After Width: | Height: | Size: 109 KiB |
|
After Width: | Height: | Size: 816 KiB |
|
After Width: | Height: | Size: 1.1 MiB |
@@ -0,0 +1 @@
|
||||
1.jpg
|
||||
@@ -0,0 +1 @@
|
||||
1.jpg
|
||||
|
After Width: | Height: | Size: 1.2 MiB |
@@ -0,0 +1 @@
|
||||
4.jpg
|
||||
|
After Width: | Height: | Size: 1.2 MiB |
@@ -0,0 +1 @@
|
||||
6.jpg
|
||||
@@ -0,0 +1 @@
|
||||
6.jpg
|
||||
@@ -0,0 +1 @@
|
||||
0.jpg
|
||||
|
After Width: | Height: | Size: 1.2 MiB |
|
After Width: | Height: | Size: 1.4 MiB |
|
After Width: | Height: | Size: 1.6 MiB |
|
After Width: | Height: | Size: 1.7 MiB |
|
After Width: | Height: | Size: 1.8 MiB |
@@ -0,0 +1 @@
|
||||
4.jpg
|
||||
|
After Width: | Height: | Size: 1.7 MiB |
|
After Width: | Height: | Size: 1.5 MiB |
|
After Width: | Height: | Size: 1.3 MiB |
@@ -0,0 +1 @@
|
||||
0.jpg
|
||||