Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'source/blender/datatoc/datatoc_icon_split.py')
-rwxr-xr-xsource/blender/datatoc/datatoc_icon_split.py292
1 files changed, 292 insertions, 0 deletions
diff --git a/source/blender/datatoc/datatoc_icon_split.py b/source/blender/datatoc/datatoc_icon_split.py
new file mode 100755
index 00000000000..44d8e5fd0fb
--- /dev/null
+++ b/source/blender/datatoc/datatoc_icon_split.py
@@ -0,0 +1,292 @@
+# ##### BEGIN GPL LICENSE BLOCK #####
+#
+# This program is free software; you can redistribute it and/or
+# modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation; either version 2
+# of the License, or (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software Foundation,
+# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+#
+# ##### END GPL LICENSE BLOCK #####
+
+# <pep8 compliant>
+
+"""
+This script dices up PNG into small files to store in version control.
+
+Example:
+
+./blender.bin \
+ --background -noaudio \
+ --python ./release/datafiles/icon_dice.py -- \
+ --image=./release/datafiles/blender_icons16.png \
+ --output=./release/datafiles/blender_icons16
+ --output_prefix=icon16_
+ --name_style=UI_ICONS
+ --parts_x 26 --parts_y 32 \
+ --minx=10 --maxx 10 --miny 10 --maxy 10
+ --minx_icon 2 --maxx_icon 2 --miny_icon 2 --maxy_icon 2 \
+ --spacex_icon 1 --spacey_icon 1
+
+"""
+
+import os
+
+SOURCE_DIR = os.path.normpath(os.path.join(os.path.dirname(__file__), "..", "..", ".."))
+VERBOSE = False
+
+
+def image_from_file__bpy(filepath):
+ import bpy
+
+ image = bpy.data.images.load(filepath)
+ image.reload()
+
+ pixel_w, pixel_h = image.size
+ pixels = image.pixels[:]
+ return pixels, pixel_w, pixel_h
+
+
+def image_from_file(filepath):
+ """
+ Return pixels, w, h from an image.
+
+ note: bpy import is ONLY used here.
+ """
+
+ try:
+ import bpy
+ except:
+ bpy = None
+
+ if bpy is not None:
+ pixels, pixel_w, pixel_h = image_from_file__bpy(filepath)
+ #else:
+ # pixels, pixel_w, pixel_h = image_from_file__py(filepath)
+
+ return pixels, pixel_w, pixel_h
+
+
+def write_subimage(sub_x, sub_y, sub_w, sub_h,
+ filepath,
+ pixels, pixel_w, pixel_h):
+ import struct
+
+ # first check if the icon is worth writing
+ is_fill = False
+ for y in range(sub_h):
+ for x in range(sub_w):
+ i = (sub_x + x) + ((sub_y + y) * pixel_w)
+ a = pixels[(i * 4) + 3]
+ if a != 0.0:
+ is_fill = True
+ break
+
+ if not is_fill:
+ # print("skipping:", filepath)
+ return
+
+ with open(filepath, 'wb') as f:
+
+ f.write(struct.pack('<6I',
+ sub_w, sub_h,
+ sub_x, sub_y,
+ # redundant but including to maintain consistency
+ pixel_w, pixel_h,
+ ))
+
+ for y in range(sub_h):
+ for x in range(sub_w):
+ i = (sub_x + x) + ((sub_y + y) * pixel_w)
+ rgba = pixels[(i * 4):(i * 4) + 4]
+ c = sum((int(p * 255) << (8 * i)) for i, p in enumerate(rgba))
+ f.write(struct.pack("<I", c))
+
+
+_dice_icon_name_cache = {}
+
+
+def dice_icon_name(x, y, parts_x, parts_y,
+ name_style=None, prefix=""):
+ """
+ How to name icons, this is mainly for what name we get in git,
+ the actual names don't really matter, its just nice to have the
+ name match up with something recognizable for commits.
+ """
+ if name_style == 'UI_ICONS':
+
+ # Init on demand
+ if not _dice_icon_name_cache:
+ import re
+
+ # Search for eg: DEF_ICON(BRUSH_NUDGE) --> BRUSH_NUDGE
+ re_icon = re.compile('^\s*DEF_ICON\(\s*([A-Za-z0-9_]+)\s*\).*$')
+
+ ui_icons_h = os.path.join(SOURCE_DIR, "source", "blender", "editors", "include", "UI_icons.h")
+ with open(ui_icons_h, 'r', encoding="utf-8") as f:
+ for l in f:
+ match = re_icon.search(l)
+ if match:
+ icon_name = match.group(1).lower()
+ # print(l.rstrip())
+ _dice_icon_name_cache[len(_dice_icon_name_cache)] = icon_name
+ # ---- Done with icon cache
+
+ index = (y * parts_x) + x
+ icon_name = _dice_icon_name_cache[index]
+
+ # for debugging its handy to sort by number
+ #~ id_str = "%03d_%s%s.dat" % (index, prefix, icon_name)
+
+ id_str = "%s%s.dat" % (prefix, icon_name)
+
+ elif name_style == "":
+ # flip so icons are numbered from top-left
+ # because new icons will be added at the bottom
+ y_flip = parts_y - (y + 1)
+ id_str = "%s%02xx%02x.dat" % (prefix, x, y_flip)
+ else:
+ raise Exception("Invalid '--name_style' arg")
+
+ return id_str
+
+
+def dice(filepath, output, output_prefix, name_style,
+ parts_x, parts_y,
+ minx, miny, maxx, maxy,
+ minx_icon, miny_icon, maxx_icon, maxy_icon,
+ spacex_icon, spacey_icon,
+ ):
+ import struct
+
+ is_simple = (max(minx, miny, maxx, maxy,
+ minx_icon, miny_icon, maxx_icon, maxy_icon,
+ spacex_icon, spacey_icon) == 0)
+
+ pixels, pixel_w, pixel_h = image_from_file(filepath)
+
+ if not (pixel_w and pixel_h):
+ print("Image not found %r!" % filepath)
+ return
+
+ if not os.path.exists(output):
+ os.mkdir(output)
+
+ if is_simple:
+ pixels_w_clip = pixel_w
+ pixels_h_clip = pixel_h
+
+ icon_w = pixels_w_clip // parts_x
+ icon_h = pixels_h_clip // parts_y
+ icon_w_clip = icon_w
+ icon_h_clip = icon_h
+ else:
+ pixels_w_clip = pixel_w - (minx + maxx)
+ pixels_h_clip = pixel_h - (miny + maxy)
+
+ icon_w = (pixels_w_clip - ((parts_x - 1) * spacex_icon)) // parts_x
+ icon_h = (pixels_h_clip - ((parts_y - 1) * spacey_icon)) // parts_y
+ icon_w_clip = icon_w - (minx_icon + maxx_icon)
+ icon_h_clip = icon_h - (miny_icon + maxy_icon)
+
+ print(pixel_w, pixel_h, icon_w, icon_h)
+
+ for x in range(parts_x):
+ for y in range(parts_y):
+ id_str = dice_icon_name(x, y,
+ parts_x, parts_y,
+ name_style=name_style, prefix=output_prefix)
+ filepath = os.path.join(output, id_str)
+ if VERBOSE:
+ print(" writing:", filepath)
+
+ # simple, no margins
+ if is_simple:
+ sub_x = x * icon_x
+ sub_y = y * icon_h
+ else:
+ sub_x = minx + ((x * (icon_w + spacex_icon)) + minx_icon)
+ sub_y = miny + ((y * (icon_h + spacey_icon)) + miny_icon)
+
+ write_subimage(sub_x, sub_y, icon_w_clip, icon_h_clip,
+ filepath,
+ pixels, pixel_w, pixel_h)
+
+
+def main():
+ import sys
+ import argparse
+
+ epilog = "Run this after updating the SVG file"
+
+ argv = sys.argv
+
+ if "--" not in argv:
+ argv = []
+ else:
+ argv = argv[argv.index("--") + 1:]
+
+ parser = argparse.ArgumentParser(description=__doc__, epilog=epilog)
+
+ # File path options
+ parser.add_argument("--image", dest="image", metavar='FILE',
+ help="Image file")
+
+ parser.add_argument("--output", dest="output", metavar='DIR',
+ help="Output directory")
+
+ parser.add_argument("--output_prefix", dest="output_prefix", metavar='STRING',
+ help="Output prefix")
+
+ # Icon naming option
+ parser.add_argument("--name_style", dest="name_style", metavar='ENUM', type=str,
+ choices=('', 'UI_ICONS'),
+ help="The metod used for naming output data")
+
+ # Options for dicing up the image
+ parser.add_argument("--parts_x", dest="parts_x", metavar='INT', type=int,
+ help="Grid X parts")
+ parser.add_argument("--parts_y", dest="parts_y", metavar='INT', type=int,
+ help="Grid Y parts")
+
+ _help = "Inset from the outer edge (in pixels)"
+ parser.add_argument("--minx", dest="minx", metavar='INT', type=int, help=_help)
+ parser.add_argument("--miny", dest="miny", metavar='INT', type=int, help=_help)
+ parser.add_argument("--maxx", dest="maxx", metavar='INT', type=int, help=_help)
+ parser.add_argument("--maxy", dest="maxy", metavar='INT', type=int, help=_help)
+
+ _help = "Inset from each icons bounds (in pixels)"
+ parser.add_argument("--minx_icon", dest="minx_icon", metavar='INT', type=int, help=_help)
+ parser.add_argument("--miny_icon", dest="miny_icon", metavar='INT', type=int, help=_help)
+ parser.add_argument("--maxx_icon", dest="maxx_icon", metavar='INT', type=int, help=_help)
+ parser.add_argument("--maxy_icon", dest="maxy_icon", metavar='INT', type=int, help=_help)
+
+ _help = "Empty space between icons"
+ parser.add_argument("--spacex_icon", dest="spacex_icon", metavar='INT', type=int, help=_help)
+ parser.add_argument("--spacey_icon", dest="spacey_icon", metavar='INT', type=int, help=_help)
+
+ del _help
+
+ args = parser.parse_args(argv)
+
+ if not argv:
+ print("No args given!")
+ parser.print_help()
+ return
+
+ dice(args.image, args.output, args.output_prefix, args.name_style,
+ args.parts_x, args.parts_y,
+ args.minx, args.miny, args.maxx, args.maxy,
+ args.minx_icon, args.miny_icon, args.maxx_icon, args.maxy_icon,
+ args.spacex_icon, args.spacey_icon,
+ )
+
+if __name__ == "__main__":
+ main()