From 8c659c8af9e23c0b7bf941ebae9c765241f97566 Mon Sep 17 00:00:00 2001 From: Mauricio Baeza Date: Thu, 17 Dec 2020 17:32:21 -0600 Subject: [PATCH] Replace easymacro2 for easymacro --- .gitignore | 2 + README.md | 2 +- easymacro.py | 6616 +++++++++++++++++ easymacro2.py | 1 - files/ZAZLaTex2SVG_v0.1.0.oxt | Bin 58610 -> 64442 bytes source/ZAZLaTex2SVG.py | 2 +- .../{easymacro2.py => easymacro.py} | 1149 ++- zaz.py | 56 +- 8 files changed, 7758 insertions(+), 70 deletions(-) create mode 100644 easymacro.py delete mode 120000 easymacro2.py rename source/pythonpath/{easymacro2.py => easymacro.py} (76%) diff --git a/.gitignore b/.gitignore index 13d1490..fd613cb 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,6 @@ # ---> Python +*.po~ + # Byte-compiled / optimized / DLL files __pycache__/ *.py[cod] diff --git a/README.md b/README.md index 3879342..b0e72f6 100644 --- a/README.md +++ b/README.md @@ -20,7 +20,7 @@ Requirements: #### but, don't make the mistake of many of *thinking only in gratis software* that so much damage has done to **Free Software**. -This extension have a cost of maintenance of 5 euros every year. +This extension have a cost of maintenance of 1 euro every year. BCH: `qztd3l00xle5tffdqvh2snvadkuau2ml0uqm4n875d` diff --git a/easymacro.py b/easymacro.py new file mode 100644 index 0000000..d91c6af --- /dev/null +++ b/easymacro.py @@ -0,0 +1,6616 @@ +#!/usr/bin/env python3 + +# == Rapid Develop Macros in LibreOffice == + +# ~ This file is part of ZAZ. + +# ~ https://git.elmau.net/elmau/zaz + +# ~ ZAZ 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 3 of the License, or +# ~ (at your option) any later version. + +# ~ ZAZ 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 ZAZ. If not, see . + +import base64 +import csv +import datetime +import getpass +import gettext +import hashlib +import json +import logging +import os +import platform +import re +import shlex +import shutil +import socket +import ssl +import subprocess +import sys +import tempfile +import threading +import time +import traceback +import zipfile + +from collections import OrderedDict +from collections.abc import MutableMapping +from decimal import Decimal +from enum import IntEnum +from functools import wraps +from pathlib import Path +from pprint import pprint +from string import Template +from typing import Any, Union +from urllib.request import Request, urlopen +from urllib.error import URLError, HTTPError + +import imaplib +import smtplib +from smtplib import SMTPException, SMTPAuthenticationError +from email.mime.multipart import MIMEMultipart +from email.mime.base import MIMEBase +from email.mime.text import MIMEText +from email.utils import formatdate +from email import encoders +import mailbox + +import uno +import unohelper +from com.sun.star.awt import MessageBoxButtons as MSG_BUTTONS +from com.sun.star.awt.MessageBoxResults import YES +from com.sun.star.awt import Rectangle, Size, Point +from com.sun.star.awt.PosSize import POSSIZE, SIZE +from com.sun.star.awt import Key, KeyModifier, KeyEvent +from com.sun.star.container import NoSuchElementException +from com.sun.star.datatransfer import XTransferable, DataFlavor + +from com.sun.star.beans import PropertyValue, NamedValue +from com.sun.star.sheet import TableFilterField +from com.sun.star.table.CellContentType import EMPTY, VALUE, TEXT, FORMULA +from com.sun.star.util import Time, Date, DateTime + +from com.sun.star.text.ControlCharacter import PARAGRAPH_BREAK +from com.sun.star.text.TextContentAnchorType import AS_CHARACTER + +from com.sun.star.awt import XActionListener +from com.sun.star.lang import XEventListener +from com.sun.star.awt import XMenuListener +from com.sun.star.awt import XMouseListener +from com.sun.star.awt import XMouseMotionListener +from com.sun.star.awt import XFocusListener +from com.sun.star.awt import XKeyListener +from com.sun.star.awt import XItemListener +from com.sun.star.awt import XTabListener +from com.sun.star.awt import XWindowListener +from com.sun.star.awt import XTopWindowListener +from com.sun.star.awt.grid import XGridDataListener +from com.sun.star.awt.grid import XGridSelectionListener +from com.sun.star.script import ScriptEventDescriptor + +# ~ https://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1awt_1_1FontUnderline.html +from com.sun.star.awt import FontUnderline +from com.sun.star.style.VerticalAlignment import TOP, MIDDLE, BOTTOM + +from com.sun.star.view.SelectionType import SINGLE, MULTI, RANGE + +from com.sun.star.sdb.CommandType import TABLE, QUERY, COMMAND + +try: + from peewee import Database, DateTimeField, DateField, TimeField, \ + __exception_wrapper__ +except ImportError as e: + Database = DateField = TimeField = DateTimeField = object + print('You need install peewee, only if you will develop with Base') + + +LOG_FORMAT = '%(asctime)s - %(levelname)s - %(message)s' +LOG_DATE = '%d/%m/%Y %H:%M:%S' +logging.addLevelName(logging.ERROR, '\033[1;41mERROR\033[1;0m') +logging.addLevelName(logging.DEBUG, '\x1b[33mDEBUG\033[1;0m') +logging.addLevelName(logging.INFO, '\x1b[32mINFO\033[1;0m') +logging.basicConfig(level=logging.DEBUG, format=LOG_FORMAT, datefmt=LOG_DATE) +log = logging.getLogger(__name__) + + +# ~ You can get custom salt +# ~ codecs.encode(os.urandom(16), 'hex') +# ~ but, not modify this file, modify in import file +SALT = b'c9548699d4e432dfd2b46adddafbb06d' + +TIMEOUT = 10 +LOG_NAME = 'ZAZ' +FILE_NAME_CONFIG = 'zaz-{}.json' + +LEFT = 0 +CENTER = 1 +RIGHT = 2 + +CALC = 'calc' +WRITER = 'writer' +DRAW = 'draw' +IMPRESS = 'impress' +BASE = 'base' +MATH = 'math' +BASIC = 'basic' +MAIN = 'main' +TYPE_DOC = { + CALC: 'com.sun.star.sheet.SpreadsheetDocument', + WRITER: 'com.sun.star.text.TextDocument', + DRAW: 'com.sun.star.drawing.DrawingDocument', + IMPRESS: 'com.sun.star.presentation.PresentationDocument', + BASE: 'com.sun.star.sdb.DocumentDataSource', + MATH: 'com.sun.star.formula.FormulaProperties', + BASIC: 'com.sun.star.script.BasicIDE', + MAIN: 'com.sun.star.frame.StartModule', +} + +OBJ_CELL = 'ScCellObj' +OBJ_RANGE = 'ScCellRangeObj' +OBJ_RANGES = 'ScCellRangesObj' +TYPE_RANGES = (OBJ_CELL, OBJ_RANGE, OBJ_RANGES) + +OBJ_SHAPES = 'com.sun.star.drawing.SvxShapeCollection' +OBJ_SHAPE = 'com.sun.star.comp.sc.ScShapeObj' +OBJ_GRAPHIC = 'SwXTextGraphicObject' + +OBJ_TEXTS = 'SwXTextRanges' +OBJ_TEXT = 'SwXTextRange' + + +# ~ from com.sun.star.sheet.FilterOperator import EMPTY, NO_EMPTY, EQUAL, NOT_EQUAL +class FilterOperator(IntEnum): + EMPTY = 0 + NO_EMPTY = 1 + EQUAL = 2 + NOT_EQUAL = 3 + +# ~ https://api.libreoffice.org/docs/idl/ref/servicecom_1_1sun_1_1star_1_1awt_1_1UnoControlEditModel.html#a54d3ff280d892218d71e667f81ce99d4 +class Border(IntEnum): + NO_BORDER = 0 + BORDER = 1 + SIMPLE = 2 + + +# ~ https://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1sheet.html#aa5aa6dbecaeb5e18a476b0a58279c57a +class ValidationType(): + from com.sun.star.sheet.ValidationType \ + import ANY, WHOLE, DECIMAL, DATE, TIME, TEXT_LEN, LIST, CUSTOM +VT = ValidationType + + +# ~ https://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1sheet.html#aecf58149730f4c8c5c18c70f3c7c5db7 +class ValidationAlertStyle(): + from com.sun.star.sheet.ValidationAlertStyle \ + import STOP, WARNING, INFO, MACRO +VAS = ValidationAlertStyle + + +# ~ https://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1sheet_1_1ConditionOperator2.html +class ConditionOperator(): + from com.sun.star.sheet.ConditionOperator2 \ + import NONE, EQUAL, NOT_EQUAL, GREATER, GREATER_EQUAL, LESS, \ + LESS_EQUAL, BETWEEN, NOT_BETWEEN, FORMULA, DUPLICATE, NOT_DUPLICATE +CO = ConditionOperator + + +OS = platform.system() +IS_WIN = OS == 'Windows' +IS_MAC = OS == 'Darwin' +USER = getpass.getuser() +PC = platform.node() +DESKTOP = os.environ.get('DESKTOP_SESSION', '') +INFO_DEBUG = f"{sys.version}\n\n{platform.platform()}\n\n" + '\n'.join(sys.path) + +PYTHON = 'python' +if IS_WIN: + PYTHON = 'python.exe' + +_MACROS = {} +_start = 0 + +SECONDS_DAY = 60 * 60 * 24 +DIR = { + 'images': 'images', + 'locales': 'locales', +} + +KEY = { + 'enter': 1280, +} + +MODIFIERS = { + 'shift': KeyModifier.SHIFT, + 'ctrl': KeyModifier.MOD1, + 'alt': KeyModifier.MOD2, + 'ctrlmac': KeyModifier.MOD3, +} + +# ~ Menus +NODE_MENUBAR = 'private:resource/menubar/menubar' +MENUS = { + 'file': '.uno:PickList', + 'tools': '.uno:ToolsMenu', + 'help': '.uno:HelpMenu', + 'windows': '.uno:WindowList', + 'edit': '.uno:EditMenu', + 'view': '.uno:ViewMenu', + 'insert': '.uno:InsertMenu', + 'format': '.uno:FormatMenu', + 'styles': '.uno:FormatStylesMenu', + 'sheet': '.uno:SheetMenu', + 'data': '.uno:DataMenu', + 'table': '.uno:TableMenu', + 'form': '.uno:FormatFormMenu', + 'page': '.uno:PageMenu', + 'shape': '.uno:ShapeMenu', + 'slide': '.uno:SlideMenu', + 'show': '.uno:SlideShowMenu', +} + +DEFAULT_MIME_TYPE = 'png' +MIME_TYPE = { + 'png': 'image/png', + 'jpg': 'image/jpeg', +} + +MESSAGES = { + 'es': { + 'OK': 'Aceptar', + 'Cancel': 'Cancelar', + 'Select path': 'Seleccionar ruta', + 'Select directory': 'Seleccionar directorio', + 'Select file': 'Seleccionar archivo', + 'Incorrect user or password': 'Nombre de usuario o contraseña inválidos', + 'Allow less secure apps in GMail': 'Activa: Permitir aplicaciones menos segura en GMail', + } +} + + +CTX = uno.getComponentContext() +SM = CTX.getServiceManager() + + +def create_instance(name: str, with_context: bool=False, args: Any=None) -> Any: + if with_context: + instance = SM.createInstanceWithContext(name, CTX) + elif args: + instance = SM.createInstanceWithArguments(name, (args,)) + else: + instance = SM.createInstance(name) + return instance + + +def get_app_config(node_name, key=''): + name = 'com.sun.star.configuration.ConfigurationProvider' + service = 'com.sun.star.configuration.ConfigurationAccess' + cp = create_instance(name, True) + node = PropertyValue(Name='nodepath', Value=node_name) + try: + ca = cp.createInstanceWithArguments(service, (node,)) + if ca and not key: + return ca + if ca and ca.hasByName(key): + return ca.getPropertyValue(key) + except Exception as e: + error(e) + return '' + + +LANGUAGE = get_app_config('org.openoffice.Setup/L10N/', 'ooLocale') +LANG = LANGUAGE.split('-')[0] +NAME = TITLE = get_app_config('org.openoffice.Setup/Product', 'ooName') +VERSION = get_app_config('org.openoffice.Setup/Product','ooSetupVersion') + +INFO_DEBUG = f"{NAME} v{VERSION} {LANGUAGE}\n\n{INFO_DEBUG}" + +node = '/org.openoffice.Office.Calc/Calculate/Other/Date' +y = get_app_config(node, 'YY') +m = get_app_config(node, 'MM') +d = get_app_config(node, 'DD') +DATE_OFFSET = datetime.date(y, m, d).toordinal() + + +def error(info): + log.error(info) + return + + +def debug(*args): + data = [str(a) for a in args] + log.debug('\t'.join(data)) + return + + +def info(*args): + data = [str(a) for a in args] + log.info('\t'.join(data)) + return + + +def save_log(path, data): + with open(path, 'a') as f: + f.write(f'{str(now())[:19]} -{LOG_NAME}- ') + pprint(data, stream=f) + return + + +def catch_exception(f): + @wraps(f) + def func(*args, **kwargs): + try: + return f(*args, **kwargs) + except Exception as e: + name = f.__name__ + if IS_WIN: + msgbox(traceback.format_exc()) + log.error(name, exc_info=True) + return func + + +def inspect(obj: Any) -> None: + zaz = create_instance('net.elmau.zaz.inspect') + if hasattr(obj, 'obj'): + obj = obj.obj + zaz.inspect(obj) + return + + +def mri(obj): + m = create_instance('mytools.Mri') + if m is None: + msg = 'Extension MRI not found' + error(msg) + return + + m.inspect(obj) + return + + +def run_in_thread(fn): + def run(*k, **kw): + t = threading.Thread(target=fn, args=k, kwargs=kw) + t.start() + return t + return run + + +def now(only_time=False): + now = datetime.datetime.now() + if only_time: + now = now.time() + return now + + +def today(): + return datetime.date.today() + + +def _(msg): + if LANG == 'en': + return msg + + if not LANG in MESSAGES: + return msg + + return MESSAGES[LANG][msg] + + +def msgbox(message, title=TITLE, buttons=MSG_BUTTONS.BUTTONS_OK, type_msg='infobox'): + """ Create message box + type_msg: infobox, warningbox, errorbox, querybox, messbox + http://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1awt_1_1XMessageBoxFactory.html + """ + toolkit = create_instance('com.sun.star.awt.Toolkit') + parent = toolkit.getDesktopWindow() + box = toolkit.createMessageBox(parent, type_msg, buttons, title, str(message)) + return box.execute() + + +def question(message, title=TITLE): + result = msgbox(message, title, MSG_BUTTONS.BUTTONS_YES_NO, 'querybox') + return result == YES + + +def warning(message, title=TITLE): + return msgbox(message, title, type_msg='warningbox') + + +def errorbox(message, title=TITLE): + return msgbox(message, title, type_msg='errorbox') + + +def get_type_doc(obj: Any) -> str: + for k, v in TYPE_DOC.items(): + if obj.supportsService(v): + return k + return '' + + +def _get_class_doc(obj: Any) -> Any: + classes = { + CALC: LOCalc, + WRITER: LOWriter, + DRAW: LODraw, + IMPRESS: LOImpress, + BASE: LOBase, + MATH: LOMath, + BASIC: LOBasic, + } + type_doc = get_type_doc(obj) + return classes[type_doc](obj) + + +def dict_to_property(values: dict, uno_any: bool=False): + ps = tuple([PropertyValue(Name=n, Value=v) for n, v in values.items()]) + if uno_any: + ps = uno.Any('[]com.sun.star.beans.PropertyValue', ps) + return ps + + +def _array_to_dict(values): + d = {v[0]: v[1] for v in values} + return d + + +def _property_to_dict(values): + d = {v.Name: v.Value for v in values} + return d + + +def json_dumps(data): + return json.dumps(data, indent=4, sort_keys=True) + + +def json_loads(data): + return json.loads(data) + + +def data_to_dict(data): + if isinstance(data, tuple) and isinstance(data[0], tuple): + return _array_to_dict(data) + + if isinstance(data, tuple) and isinstance(data[0], (PropertyValue, NamedValue)): + return _property_to_dict(data) + return {} + + +def _get_dispatch() -> Any: + return create_instance('com.sun.star.frame.DispatchHelper') + + +# ~ https://wiki.documentfoundation.org/Development/DispatchCommands +# ~ Used only if not exists in API +def call_dispatch(frame: Any, url: str, args: dict={}) -> None: + dispatch = _get_dispatch() + opt = dict_to_property(args) + dispatch.executeDispatch(frame, url, '', 0, opt) + return + + +def get_desktop(): + return create_instance('com.sun.star.frame.Desktop', True) + + +def _date_to_struct(value): + if isinstance(value, datetime.datetime): + d = DateTime() + d.Year = value.year + d.Month = value.month + d.Day = value.day + d.Hours = value.hour + d.Minutes = value.minute + d.Seconds = value.second + elif isinstance(value, datetime.date): + d = Date() + d.Day = value.day + d.Month = value.month + d.Year = value.year + elif isinstance(value, datetime.time): + d = Time() + d.Hours = value.hour + d.Minutes = value.minute + d.Seconds = value.second + return d + + +def _struct_to_date(value): + d = None + if isinstance(value, Time): + d = datetime.time(value.Hours, value.Minutes, value.Seconds) + elif isinstance(value, Date): + if value != Date(): + d = datetime.date(value.Year, value.Month, value.Day) + elif isinstance(value, DateTime): + if value.Year > 0: + d = datetime.datetime( + value.Year, value.Month, value.Day, + value.Hours, value.Minutes, value.Seconds) + return d + + +def _get_url_script(args): + library = args['library'] + module = '.' + name = args['name'] + language = args.get('language', 'Python') + location = args.get('location', 'user') + + if language == 'Python': + module = '.py$' + elif language == 'Basic': + module = f".{module}." + if location == 'user': + location = 'application' + + url = 'vnd.sun.star.script' + url = f'{url}:{library}{module}{name}?language={language}&location={location}' + return url + + +def _call_macro(args): + #~ https://wiki.openoffice.org/wiki/Documentation/DevGuide/Scripting/Scripting_Framework_URI_Specification + + url = _get_url_script(args) + args = args.get('args', ()) + + service = 'com.sun.star.script.provider.MasterScriptProviderFactory' + factory = create_instance(service) + script = factory.createScriptProvider('').getScript(url) + result = script.invoke(args, None, None)[0] + + return result + + +def call_macro(args, in_thread=False): + result = None + if in_thread: + t = threading.Thread(target=_call_macro, args=(args,)) + t.start() + else: + result = _call_macro(args) + return result + + +def run(command, capture=False, split=True): + if not split: + return subprocess.check_output(command, shell=True).decode() + + cmd = shlex.split(command) + result = subprocess.run(cmd, capture_output=capture, text=True, shell=IS_WIN) + if capture: + result = result.stdout + else: + result = result.returncode + return result + + +def popen(command): + try: + proc = subprocess.Popen(shlex.split(command), shell=IS_WIN, + stdout=subprocess.PIPE, stderr=subprocess.STDOUT) + for line in proc.stdout: + yield line.decode().rstrip() + except Exception as e: + error(e) + yield (e.errno, e.strerror) + + +def sleep(seconds): + time.sleep(seconds) + return + + +class TimerThread(threading.Thread): + + def __init__(self, event, seconds, macro): + threading.Thread.__init__(self) + self.stopped = event + self.seconds = seconds + self.macro = macro + + def run(self): + info('Timer started... {}'.format(self.macro['name'])) + while not self.stopped.wait(self.seconds): + _call_macro(self.macro) + info('Timer stopped... {}'.format(self.macro['name'])) + return + + +def start_timer(name, seconds, macro): + global _MACROS + _MACROS[name] = threading.Event() + thread = TimerThread(_MACROS[name], seconds, macro) + thread.start() + return + + +def stop_timer(name): + global _MACROS + _MACROS[name].set() + del _MACROS[name] + return + + +def install_locales(path, domain='base', dir_locales=DIR['locales']): + path_locales = _P.join(_P(path).path, dir_locales) + try: + lang = gettext.translation(domain, path_locales, languages=[LANG]) + lang.install() + _ = lang.gettext + except Exception as e: + from gettext import gettext as _ + error(e) + return _ + + +def _export_image(obj, args): + name = 'com.sun.star.drawing.GraphicExportFilter' + exporter = create_instance(name) + path = _P.to_system(args['URL']) + args = dict_to_property(args) + exporter.setSourceDocument(obj) + exporter.filter(args) + return _P.exists(path) + + +def sha256(data): + result = hashlib.sha256(data.encode()).hexdigest() + return result + +def sha512(data): + result = hashlib.sha512(data.encode()).hexdigest() + return result + + +def get_config(key='', default={}, prefix='conf'): + name_file = FILE_NAME_CONFIG.format(prefix) + values = None + path = _P.join(_P.config('UserConfig'), name_file) + if not _P.exists(path): + return default + + values = _P.from_json(path) + if key: + values = values.get(key, default) + + return values + + +def set_config(key, value, prefix='conf'): + name_file = FILE_NAME_CONFIG.format(prefix) + path = _P.join(_P.config('UserConfig'), name_file) + values = get_config(default={}, prefix=prefix) + values[key] = value + result = _P.to_json(path, values) + return result + + +def start(): + global _start + _start = now() + info(_start) + return + + +def end(get_seconds=False): + global _start + e = now() + td = e - _start + result = str(td) + if get_seconds: + result = td.total_seconds() + return result + + +def get_epoch(): + n = now() + return int(time.mktime(n.timetuple())) + + +def render(template, data): + s = Template(template) + return s.safe_substitute(**data) + + +def get_size_screen(): + if IS_WIN: + user32 = ctypes.windll.user32 + res = f'{user32.GetSystemMetrics(0)}x{user32.GetSystemMetrics(1)}' + else: + args = 'xrandr | grep "*" | cut -d " " -f4' + res = run(args, split=False) + return res.strip() + + +def url_open(url, data=None, headers={}, verify=True, get_json=False): + err = '' + req = Request(url) + for k, v in headers.items(): + req.add_header(k, v) + try: + # ~ debug(url) + if verify: + if not data is None and isinstance(data, str): + data = data.encode() + response = urlopen(req, data=data) + else: + context = ssl._create_unverified_context() + response = urlopen(req, context=context) + except HTTPError as e: + error(e) + err = str(e) + except URLError as e: + error(e.reason) + err = str(e.reason) + else: + headers = dict(response.info()) + result = response.read() + if get_json: + result = json.loads(result) + + return result, headers, err + + +def _get_key(password): + from cryptography.hazmat.primitives import hashes + from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC + + kdf = PBKDF2HMAC(algorithm=hashes.SHA256(), length=32, salt=SALT, + iterations=100000) + key = base64.urlsafe_b64encode(kdf.derive(password.encode())) + return key + + +def encrypt(data, password): + from cryptography.fernet import Fernet + + f = Fernet(_get_key(password)) + if isinstance(data, str): + data = data.encode() + token = f.encrypt(data).decode() + return token + + +def decrypt(token, password): + from cryptography.fernet import Fernet, InvalidToken + + data = '' + f = Fernet(_get_key(password)) + try: + data = f.decrypt(token.encode()).decode() + except InvalidToken as e: + error('Invalid Token') + return data + + +def switch_design_mode(doc): + call_dispatch(doc.frame, '.uno:SwitchControlDesignMode') + return + + +class SmtpServer(object): + + def __init__(self, config): + self._server = None + self._error = '' + self._sender = '' + self._is_connect = self._login(config) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + self.close() + + @property + def is_connect(self): + return self._is_connect + + @property + def error(self): + return self._error + + def _login(self, config): + name = config['server'] + port = config['port'] + is_ssl = config['ssl'] + self._sender = config['user'] + hosts = ('gmail' in name or 'outlook' in name) + try: + if is_ssl and hosts: + self._server = smtplib.SMTP(name, port, timeout=TIMEOUT) + self._server.ehlo() + self._server.starttls() + self._server.ehlo() + elif is_ssl: + self._server = smtplib.SMTP_SSL(name, port, timeout=TIMEOUT) + self._server.ehlo() + else: + self._server = smtplib.SMTP(name, port, timeout=TIMEOUT) + + self._server.login(self._sender, config['password']) + msg = 'Connect to: {}'.format(name) + debug(msg) + return True + except smtplib.SMTPAuthenticationError as e: + if '535' in str(e): + self._error = _('Incorrect user or password') + return False + if '534' in str(e) and 'gmail' in name: + self._error = _('Allow less secure apps in GMail') + return False + except smtplib.SMTPException as e: + self._error = str(e) + return False + except Exception as e: + self._error = str(e) + return False + return False + + def _body(self, msg): + body = msg.replace('\\n', '
') + return body + + def send(self, message): + file_name = 'attachment; filename={}' + email = MIMEMultipart() + email['From'] = self._sender + email['To'] = message['to'] + email['Cc'] = message.get('cc', '') + email['Subject'] = message['subject'] + email['Date'] = formatdate(localtime=True) + if message.get('confirm', False): + email['Disposition-Notification-To'] = email['From'] + email.attach(MIMEText(self._body(message['body']), 'html')) + + for path in message.get('files', ()): + fn = _P(path).file_name + part = MIMEBase('application', 'octet-stream') + part.set_payload(_P.read_bin(path)) + encoders.encode_base64(part) + part.add_header('Content-Disposition', f'attachment; filename={fn}') + email.attach(part) + + receivers = ( + email['To'].split(',') + + email['CC'].split(',') + + message.get('bcc', '').split(',')) + try: + self._server.sendmail(self._sender, receivers, email.as_string()) + msg = 'Email sent...' + debug(msg) + if message.get('path', ''): + self.save_message(email, message['path']) + return True + except Exception as e: + self._error = str(e) + return False + return False + + def save_message(self, email, path): + mbox = mailbox.mbox(path, create=True) + mbox.lock() + try: + msg = mailbox.mboxMessage(email) + mbox.add(msg) + mbox.flush() + finally: + mbox.unlock() + return + + def close(self): + try: + self._server.quit() + msg = 'Close connection...' + debug(msg) + except: + pass + return + + +def _send_email(server, messages): + with SmtpServer(server) as server: + if server.is_connect: + for msg in messages: + server.send(msg) + else: + error(server.error) + return server.error + + +def send_email(server, message): + messages = message + if isinstance(message, dict): + messages = (message,) + t = threading.Thread(target=_send_email, args=(server, messages)) + t.start() + return + + +class ImapServer(object): + + def __init__(self, config): + self._server = None + self._error = '' + self._is_connect = self._login(config) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + self.close() + + @property + def is_connect(self): + return self._is_connect + + @property + def error(self): + return self._error + + def _login(self, config): + try: + # ~ hosts = 'gmail' in config['server'] + if config['ssl']: + self._server = imaplib.IMAP4_SSL(config['server'], config['port']) + else: + self._server = imaplib.IMAP4(config['server'], config['port']) + self._server.login(config['user'], config['password']) + self._server.select() + return True + except imaplib.IMAP4.error as e: + self._error = str(e) + return False + except Exception as e: + self._error = str(e) + return False + return False + + def get_folders(self, exclude=()): + folders = {} + result, subdir = self._server.list() + for s in subdir: + print(s.decode('utf-8')) + return folders + + def close(self): + try: + self._server.close() + self._server.logout() + msg = 'Close connection...' + debug(msg) + except: + pass + return + + +# ~ Classes + +class LOBaseObject(object): + + def __init__(self, obj): + self._obj = obj + + def __setattr__(self, name, value): + exists = hasattr(self, name) + if not exists and not name in ('_obj', '_index', '_view'): + setattr(self._obj, name, value) + else: + super().__setattr__(name, value) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + @property + def obj(self): + return self._obj + + +class LODocument(object): + FILTERS = { + 'doc': 'MS Word 97', + 'docx': 'MS Word 2007 XML', + } + + def __init__(self, obj): + self._obj = obj + self._cc = self.obj.getCurrentController() + self._undo = True + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + self.close() + + @property + def obj(self): + return self._obj + + @property + def title(self): + return self.obj.getTitle() + @title.setter + def title(self, value): + self.obj.setTitle(value) + + @property + def type(self): + return self._type + + @property + def uid(self): + return self.obj.RuntimeUID + + @property + def frame(self): + return self._cc.getFrame() + + @property + def is_saved(self): + return self.obj.hasLocation() + + @property + def is_modified(self): + return self.obj.isModified() + + @property + def is_read_only(self): + return self.obj.isReadOnly() + + @property + def path(self): + return _P.to_system(self.obj.URL) + + @property + def dir(self): + return _P(self.path).path + + @property + def file_name(self): + return _P(self.path).file_name + + @property + def name(self): + return _P(self.path).name + + @property + def status_bar(self): + return self._cc.getStatusIndicator() + + @property + def visible(self): + w = self.frame.ContainerWindow + return w.isVisible() + @visible.setter + def visible(self, value): + w = self.frame.ContainerWindow + w.setVisible(value) + + @property + def zoom(self): + return self._cc.ZoomValue + @zoom.setter + def zoom(self, value): + self._cc.ZoomValue = value + + @property + def undo(self): + return self._undo + @undo.setter + def undo(self, value): + self._undo = value + um = self.obj.UndoManager + if value: + try: + um.leaveUndoContext() + except: + pass + else: + um.enterHiddenUndoContext() + + def clear_undo(self): + self.obj.getUndoManager().clear() + return + + @property + def selection(self): + sel = self.obj.CurrentSelection + # ~ return _get_class_uno(sel) + return sel + + @property + def table_auto_formats(self): + taf = create_instance('com.sun.star.sheet.TableAutoFormats') + return taf.ElementNames + + def create_instance(self, name): + obj = self.obj.createInstance(name) + return obj + + def set_focus(self): + w = self.frame.ComponentWindow + w.setFocus() + return + + def copy(self): + call_dispatch(self.frame, '.uno:Copy') + return + + def insert_contents(self, args={}): + call_dispatch(self.frame, '.uno:InsertContents', args) + return + + def paste(self): + sc = create_instance('com.sun.star.datatransfer.clipboard.SystemClipboard') + transferable = sc.getContents() + self._cc.insertTransferable(transferable) + # ~ return self.obj.getCurrentSelection() + return + + def select(self, obj): + self._cc.select(obj) + return + + def to_pdf(self, path: str='', args: dict={}): + path_pdf = path + filter_name = '{}_pdf_Export'.format(self.type) + filter_data = dict_to_property(args, True) + args = { + 'FilterName': filter_name, + 'FilterData': filter_data, + } + opt = dict_to_property(args) + try: + self.obj.storeToURL(_P.to_url(path), opt) + except Exception as e: + error(e) + path_pdf = '' + + return _P.exists(path_pdf) + + def export(self, path: str, ext: str='', args: dict={}): + if not ext: + ext = _P(path).ext + filter_name = self.FILTERS[ext] + filter_data = dict_to_property(args, True) + args = { + 'FilterName': filter_name, + 'FilterData': filter_data, + } + opt = dict_to_property(args) + try: + self.obj.storeToURL(_P.to_url(path), opt) + except Exception as e: + error(e) + path = '' + return _P.exists(path) + + def save(self, path: str='', args: dict={}) -> bool: + result = True + opt = dict_to_property(args) + if path: + try: + self.obj.storeAsURL(_P.to_url(path), opt) + except Exception as e: + error(e) + result = False + else: + self.obj.store() + return result + + def close(self): + self.obj.close(True) + return + + +class LOCellStyle(LOBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def name(self): + return self.obj.Name + + @property + def properties(self): + properties = self.obj.PropertySetInfo.Properties + data = {p.Name: getattr(self.obj, p.Name) for p in properties} + return data + @properties.setter + def properties(self, values): + _set_properties(self.obj, values) + + +class LOCellStyles(object): + + def __init__(self, obj, doc): + self._obj = obj + self._doc = doc + + def __len__(self): + return len(self.obj) + + def __getitem__(self, index): + return LOCellStyle(self.obj[index]) + + def __setitem__(self, key, value): + self.obj[key] = value + + def __delitem__(self, key): + if not isinstance(key, str): + key = key.Name + del self.obj[key] + + def __contains__(self, item): + return item in self.obj + + @property + def obj(self): + return self._obj + + @property + def names(self): + return self.obj.ElementNames + + def new(self, name: str=''): + obj = self._doc.create_instance('com.sun.star.style.CellStyle') + if name: + self.obj[name] = obj + obj = LOCellStyle(obj) + return obj + + +class LOCalc(LODocument): + + def __init__(self, obj): + super().__init__(obj) + self._type = CALC + self._sheets = obj.Sheets + + def __getitem__(self, index): + return LOCalcSheet(self._sheets[index]) + + def __setitem__(self, key, value): + self._sheets[key] = value + + def __len__(self): + return self._sheets.Count + + def __contains__(self, item): + return item in self._sheets + + @property + def names(self): + names = self.obj.Sheets.ElementNames + return names + + @property + def selection(self): + sel = self.obj.CurrentSelection + if sel.ImplementationName in TYPE_RANGES: + sel = LOCalcRange(sel) + elif sel.ImplementationName == OBJ_SHAPES: + if len(sel) == 1: + sel = sel[0] + sel = LODrawPage(sel.Parent)[sel.Name] + else: + debug(sel.ImplementationName) + return sel + + @property + def active(self): + return LOCalcSheet(self._cc.ActiveSheet) + + @property + def headers(self): + return self._cc.ColumnRowHeaders + @headers.setter + def headers(self, value): + self._cc.ColumnRowHeaders = value + + @property + def tabs(self): + return self._cc.SheetTabs + @tabs.setter + def tabs(self, value): + self._cc.SheetTabs = value + + @property + def cs(self): + return self.cell_styles + @property + def cell_styles(self): + obj = self.obj.StyleFamilies['CellStyles'] + return LOCellStyles(obj, self) + + @property + def db_ranges(self): + # ~ return LOCalcDataBaseRanges(self.obj.DataBaseRanges) + return self.obj.DatabaseRanges + + def activate(self, sheet): + obj = sheet + if isinstance(sheet, LOCalcSheet): + obj = sheet.obj + elif isinstance(sheet, str): + obj = self._sheets[sheet] + self._cc.setActiveSheet(obj) + return + + def new_sheet(self): + s = self.create_instance('com.sun.star.sheet.Spreadsheet') + return s + + def insert(self, name): + names = name + if isinstance(name, str): + names = (name,) + for n in names: + self._sheets[n] = self.new_sheet() + return LOCalcSheet(self._sheets[n]) + + def move(self, name, pos=-1): + index = pos + if pos < 0: + index = len(self) + if isinstance(name, LOCalcSheet): + name = name.name + self._sheets.moveByName(name, index) + return + + def remove(self, name): + if isinstance(name, LOCalcSheet): + name = name.name + self._sheets.removeByName(name) + return + + def copy(self, name, new_name='', pos=-1): + if isinstance(name, LOCalcSheet): + name = name.name + index = pos + if pos < 0: + index = len(self) + self._sheets.copyByName(name, new_name, index) + return LOCalcSheet(self._sheets[new_name]) + + def copy_from(self, doc, source='', target='', pos=-1): + index = pos + if pos < 0: + index = len(self) + + names = source + if not source: + names = doc.names + elif isinstance(source, str): + names = (source,) + + new_names = target + if not target: + new_names = names + elif isinstance(target, str): + new_names = (target,) + + for i, name in enumerate(names): + self._sheets.importSheet(doc.obj, name, index + i) + self[index + i].name = new_names[i] + + return LOCalcSheet(self._sheets[index]) + + def sort(self, reverse=False): + names = sorted(self.names, reverse=reverse) + for i, n in enumerate(names): + self.move(n, i) + return + + def render(self, data, sheet=None, clean=True): + if sheet is None: + sheet = self.active + return sheet.render(data, clean=clean) + + +class LOChart(object): + + def __init__(self, name, obj, draw_page): + self._name = name + self._obj = obj + self._eobj = self._obj.EmbeddedObject + self._type = 'Column' + self._cell = None + self._shape = self._get_shape(draw_page) + self._pos = self._shape.Position + + def __getitem__(self, index): + return LOBaseObject(self.diagram.getDataRowProperties(index)) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + @property + def obj(self): + return self._obj + + @property + def name(self): + return self._name + + @property + def diagram(self): + return self._eobj.Diagram + + @property + def type(self): + return self._type + @type.setter + def type(self, value): + self._type = value + if value == 'Bar': + self.diagram.Vertical = True + return + type_chart = f'com.sun.star.chart.{value}Diagram' + self._eobj.setDiagram(self._eobj.createInstance(type_chart)) + + @property + def cell(self): + return self._cell + @cell.setter + def cell(self, value): + self._cell = value + self._shape.Anchor = value.obj + + @property + def position(self): + return self._pos + @position.setter + def position(self, value): + self._pos = value + self._shape.Position = value + + def _get_shape(self, draw_page): + for shape in draw_page: + if shape.PersistName == self.name: + break + return shape + + +class LOSheetCharts(object): + + def __init__(self, obj, sheet): + self._obj = obj + self._sheet = sheet + + def __getitem__(self, index): + return LOChart(index, self.obj[index], self._sheet.draw_page) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def __contains__(self, item): + return item in self.obj + + def __len__(self): + return len(self.obj) + + @property + def obj(self): + return self._obj + + def new(self, name, pos_size, data): + self.obj.addNewByName(name, pos_size, data, True, True) + return LOChart(name, self.obj[name], self._sheet.draw_page) + + +class LOFormControl(LOBaseObject): + EVENTS = { + 'action': 'actionPerformed', + 'click': 'mousePressed', + } + TYPES = { + 'actionPerformed': 'XActionListener', + 'mousePressed': 'XMouseListener', + } + + def __init__(self, obj, view, form): + super().__init__(obj) + self._view = view + self._form = form + self._m = view.Model + self._index = -1 + + def __setattr__(self, name, value): + if name in ('_form', '_view', '_m', '_index'): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + def __str__(self): + return f'{self.name} ({self.type}) {[self.index]}' + + @property + def form(self): + return self._form + + @property + def doc(self): + return self.obj.Parent.Forms.Parent + + @property + def name(self): + return self._m.Name + @name.setter + def name(self, value): + self._m.Name = value + + @property + def tag(self): + return self._m.Tag + @tag.setter + def tag(self, value): + self._m.Tag = value + + @property + def index(self): + return self._index + @index.setter + def index(self, value): + self._index = value + + @property + def enabled(self): + return self._m.Enabled + @enabled.setter + def enabled(self, value): + self._m.Enabled = value + + @property + def events(self): + return self.form.getScriptEvents(self.index) + def add_event(self, name, macro): + if not 'name' in macro: + macro['name'] = '{}_{}'.format(self.name, name) + + event = ScriptEventDescriptor() + event.AddListenerParam = '' + event.EventMethod = self.EVENTS[name] + event.ListenerType = self.TYPES[event.EventMethod] + event.ScriptCode = _get_url_script(macro) + event.ScriptType = 'Script' + + for ev in self.events: + if ev.EventMethod == event.EventMethod and \ + ev.ListenerType == event.ListenerType: + self.form.revokeScriptEvent(self.index, + event.ListenerType, event.EventMethod, event.AddListenerParam) + break + + self.form.registerScriptEvent(self.index, event) + return + + def set_focus(self): + self._view.setFocus() + return + + +class LOFormControlLabel(LOFormControl): + + def __init__(self, obj, view, form): + super().__init__(obj, view, form) + + @property + def type(self): + return 'label' + + @property + def value(self): + return self._m.Label + @value.setter + def value(self, value): + self._m.Label = value + + +class LOFormControlText(LOFormControl): + + def __init__(self, obj, view, form): + super().__init__(obj, view, form) + + @property + def type(self): + return 'text' + + @property + def value(self): + return self._m.Text + @value.setter + def value(self, value): + self._m.Text = value + + +class LOFormControlButton(LOFormControl): + + def __init__(self, obj, view, form): + super().__init__(obj, view, form) + + @property + def type(self): + return 'button' + + @property + def value(self): + return self._m.Label + @value.setter + def value(self, value): + self._m.Text = Label + + +FORM_CONTROL_CLASS = { + 'label': LOFormControlLabel, + 'text': LOFormControlText, + 'button': LOFormControlButton, +} + + +class LOForm(object): + MODELS = { + 'label': 'com.sun.star.form.component.FixedText', + 'text': 'com.sun.star.form.component.TextField', + 'button': 'com.sun.star.form.component.CommandButton', + } + + def __init__(self, obj, draw_page): + self._obj = obj + self._dp = draw_page + self._controls = {} + self._init_controls() + + def __getitem__(self, index): + control = self.obj[index] + return self._controls[control.Name] + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def __contains__(self, item): + return item in self.obj + + def __len__(self): + return len(self.obj) + + def __str__(self): + return f'Form: {self.name}' + + def _init_controls(self): + types = { + 'com.sun.star.form.OFixedTextModel': 'label', + 'com.sun.star.form.OEditModel': 'text', + 'com.sun.star.form.OButtonModel': 'button', + } + for i, control in enumerate(self.obj): + name = control.Name + tipo = types[control.ImplementationName] + view = self.doc.CurrentController.getControl(control) + control = FORM_CONTROL_CLASS[tipo](control, view) + control.index = i + setattr(self, name, control) + self._controls[name] = control + return + + @property + def obj(self): + return self._obj + + @property + def name(self): + return self.obj.Name + @name.setter + def name(self, value): + self.obj.Name = value + + @property + def source(self): + return self.obj.DataSourceName + @source.setter + def source(self, value): + self.obj.DataSourceName = value + + @property + def type(self): + return self.obj.CommandType + @type.setter + def type(self, value): + self.obj.CommandType = value + + @property + def command(self): + return self.obj.Command + @command.setter + def command(self, value): + self.obj.Command = value + + @property + def doc(self): + return self.obj.Parent.Parent + + def _special_properties(self, tipo, args): + if tipo == 'button': + # ~ if 'ImageURL' in args: + # ~ args['ImageURL'] = self._set_image_url(args['ImageURL']) + args['FocusOnClick'] = args.get('FocusOnClick', False) + return args + return args + + def add(self, args): + name = args['Name'] + tipo = args.pop('Type').lower() + w = args.pop('Width') + h = args.pop('Height') + x = args.pop('X', 0) + y = args.pop('Y', 0) + control = self.doc.createInstance('com.sun.star.drawing.ControlShape') + control.setSize(Size(w, h)) + control.setPosition(Point(x, y)) + model = self.doc.createInstance(self.MODELS[tipo]) + args = self._special_properties(tipo, args) + _set_properties(model, args) + control.Control = model + index = len(self) + self.obj.insertByIndex(index, model) + self._dp.add(control) + view = self.doc.CurrentController.getControl(self.obj.getByName(name)) + control = FORM_CONTROL_CLASS[tipo](control, view, self.obj) + control.index = index + setattr(self, name, control) + self._controls[name] = control + return control + + +class LOSheetForms(object): + + def __init__(self, draw_page): + self._dp = draw_page + self._obj = draw_page.Forms + + def __getitem__(self, index): + return LOForm(self.obj[index], self._dp) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def __contains__(self, item): + return item in self.obj + + def __len__(self): + return len(self.obj) + + @property + def obj(self): + return self._obj + + @property + def doc(self): + return self.obj.Parent + + @property + def count(self): + return len(self) + + @property + def names(self): + return self.obj.ElementNames + + def insert(self, name): + form = self.doc.createInstance('com.sun.star.form.component.Form') + self.obj.insertByName(name, form) + return LOForm(form, self._dp) + + def remove(self, index): + if isinstance(index, int): + self.obj.removeByIndex(index) + else: + self.obj.removeByName(index) + return + + +# ~ IsFiltered, +# ~ IsManualPageBreak, +# ~ IsStartOfNewPage +class LOSheetRows(object): + + def __init__(self, sheet, obj): + self._sheet = sheet + self._obj = obj + + def __getitem__(self, index): + if isinstance(index, int): + rows = LOSheetRows(self._sheet, self.obj[index]) + else: + rango = self._sheet[index.start:index.stop,0:] + rows = LOSheetRows(self._sheet, rango.obj.Rows) + return rows + + def __len__(self): + return self.obj.Count + + @property + def obj(self): + return self._obj + + @property + def visible(self): + return self._obj.IsVisible + @visible.setter + def visible(self, value): + self._obj.IsVisible = value + + @property + def color(self): + return self.obj.CellBackColor + @color.setter + def color(self, value): + self.obj.CellBackColor = value + + @property + def is_transparent(self): + return self.obj.IsCellBackgroundTransparent + @is_transparent.setter + def is_transparent(self, value): + self.obj.IsCellBackgroundTransparent = value + + @property + def height(self): + return self.obj.Height + @height.setter + def height(self, value): + self.obj.Height = value + + def optimal(self): + self.obj.OptimalHeight = True + return + + def insert(self, index, count): + self.obj.insertByIndex(index, count) + return + + def remove(self, index, count): + self.obj.removeByIndex(index, count) + return + + +# ~ IsManualPageBreak, +# ~ IsStartOfNewPage +class LOSheetColumns(object): + + def __init__(self, sheet, obj): + self._sheet = sheet + self._obj = obj + + def __getitem__(self, index): + if isinstance(index, (int, str)): + rows = LOSheetColumns(self._sheet, self.obj[index]) + else: + rango = self._sheet[0,index.start:index.stop] + rows = LOSheetColumns(self._sheet, rango.obj.Columns) + return rows + + def __len__(self): + return self.obj.Count + + @property + def obj(self): + return self._obj + + @property + def visible(self): + return self._obj.IsVisible + @visible.setter + def visible(self, value): + self._obj.IsVisible = value + + @property + def width(self): + return self.obj.Width + @width.setter + def width(self, value): + self.obj.Width = value + + def optimal(self): + self.obj.OptimalWidth = True + return + + def insert(self, index, count): + self.obj.insertByIndex(index, count) + return + + def remove(self, index, count): + self.obj.removeByIndex(index, count) + return + + +class LOCalcSheet(object): + + def __init__(self, obj): + self._obj = obj + + def __getitem__(self, index): + return LOCalcRange(self.obj[index]) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def __str__(self): + return f'easymacro.LOCalcSheet: {self.name}' + + @property + def obj(self): + return self._obj + + @property + def name(self): + return self._obj.Name + @name.setter + def name(self, value): + self._obj.Name = value + + @property + def code_name(self): + return self._obj.CodeName + @code_name.setter + def code_name(self, value): + self._obj.CodeName = value + + @property + def visible(self): + return self._obj.IsVisible + @visible.setter + def visible(self, value): + self._obj.IsVisible = value + + @property + def is_protected(self): + return self._obj.isProtected() + + @property + def password(self): + return '' + @visible.setter + def password(self, value): + self.obj.protect(value) + + def unprotect(self, value): + try: + self.obj.unprotect(value) + return True + except: + pass + return False + + @property + def color(self): + return self._obj.TabColor + @color.setter + def color(self, value): + self._obj.TabColor = get_color(value) + + @property + def used_area(self): + cursor = self.get_cursor() + cursor.gotoEndOfUsedArea(True) + return LOCalcRange(self[cursor.AbsoluteName].obj) + + @property + def draw_page(self): + return LODrawPage(self.obj.DrawPage) + + @property + def dp(self): + return self.draw_page + + @property + def shapes(self): + return self.draw_page + + @property + def doc(self): + return LOCalc(self.obj.DrawPage.Forms.Parent) + + @property + def charts(self): + return LOSheetCharts(self.obj.Charts, self) + + @property + def rows(self): + return LOSheetRows(self, self.obj.Rows) + + @property + def columns(self): + return LOSheetColumns(self, self.obj.Columns) + + @property + def forms(self): + return LOSheetForms(self.obj.DrawPage) + + @property + def events(self): + names = ('OnFocus', 'OnUnfocus', 'OnSelect', 'OnDoubleClick', + 'OnRightClick', 'OnChange', 'OnCalculate') + evs = self.obj.Events + events = {n: _property_to_dict(evs.getByName(n)) for n in names + if evs.getByName(n)} + return events + @events.setter + def events(self, values): + pv = '[]com.sun.star.beans.PropertyValue' + ev = self.obj.Events + for name, v in values.items(): + url = _get_url_script(v) + args = dict_to_property(dict(EventType='Script', Script=url)) + # ~ e.replaceByName(k, args) + uno.invoke(ev, 'replaceByName', (name, uno.Any(pv, args))) + + @property + def search_descriptor(self): + return self.obj.createSearchDescriptor() + + @property + def replace_descriptor(self): + return self.obj.createReplaceDescriptor() + + def activate(self): + self.doc.activate(self.obj) + return + + def clean(self): + doc = self.doc + sheet = doc.create_instance('com.sun.star.sheet.Spreadsheet') + doc._sheets.replaceByName(self.name, sheet) + return + + def move(self, pos=-1): + index = pos + if pos < 0: + index = len(self.doc) + self.doc._sheets.moveByName(self.name, index) + return + + def remove(self): + self.doc._sheets.removeByName(self.name) + return + + def copy(self, new_name='', pos=-1): + index = pos + if pos < 0: + index = len(self.doc) + self.doc._sheets.copyByName(self.name, new_name, index) + return LOCalcSheet(self.doc._sheets[new_name]) + + def copy_to(self, doc, target='', pos=-1): + index = pos + if pos < 0: + index = len(doc) + name = self.name + if not target: + new_name = name + + doc._sheets.importSheet(self.doc.obj, name, index) + sheet = doc[name] + sheet.name = new_name + return sheet + + def get_cursor(self, cell=None): + if cell is None: + cursor = self.obj.createCursor() + else: + cursor = self.obj.createCursorByRange(cell) + return cursor + + def render(self, data, rango=None, clean=True): + if rango is None: + rango = self.used_area + return rango.render(data, clean) + + def find(self, search_string, rango=None): + if rango is None: + rango = self.used_area + return rango.find(search_string) + + +class LOCalcRange(object): + + def __init__(self, obj): + self._obj = obj + self._sd = None + self._is_cell = obj.ImplementationName == OBJ_CELL + + def __getitem__(self, index): + return LOCalcRange(self.obj[index]) + + def __iter__(self): + self._r = 0 + self._c = 0 + return self + + def __next__(self): + try: + rango = self[self._r, self._c] + except Exception as e: + raise StopIteration + self._c += 1 + if self._c == self.columns: + self._c = 0 + self._r +=1 + return rango + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def __contains__(self, item): + return item.in_range(self) + + def __str__(self): + if self.is_none: + s = 'Range: None' + else: + s = f'Range: {self.name}' + return s + + @property + def obj(self): + return self._obj + + @property + def is_none(self): + return self.obj is None + + @property + def is_cell(self): + return self._is_cell + + @property + def back_color(self): + return self._obj.CellBackColor + @back_color.setter + def back_color(self, value): + self._obj.CellBackColor = get_color(value) + + @property + def dp(self): + return self.sheet.dp + + @property + def sheet(self): + return LOCalcSheet(self.obj.Spreadsheet) + + @property + def doc(self): + doc = self.obj.Spreadsheet.DrawPage.Forms.Parent + return LODocument(doc) + + @property + def name(self): + return self.obj.AbsoluteName + + @property + def code_name(self): + name = self.name.replace('$', '').replace('.', '_').replace(':', '') + return name + + @property + def columns(self): + return self.obj.Columns.Count + + @property + def column(self): + c1 = self.address.Column + c2 = c1 + 1 + ra = self.current_region.range_address + r1 = ra.StartRow + r2 = ra.EndRow + 1 + return LOCalcRange(self.sheet[r1:r2, c1:c2].obj) + + @property + def rows(self): + return LOSheetRows(self.sheet, self.obj.Rows) + + @property + def row(self): + r1 = self.address.Row + r2 = r1 + 1 + ra = self.current_region.range_address + c1 = ra.StartColumn + c2 = ra.EndColumn + 1 + return LOCalcRange(self.sheet[r1:r2, c1:c2].obj) + + @property + def type(self): + return self.obj.Type + + @property + def value(self): + v = None + if self.type == VALUE: + v = self.obj.getValue() + elif self.type == TEXT: + v = self.obj.getString() + elif self.type == FORMULA: + v = self.obj.getFormula() + return v + @value.setter + def value(self, data): + if isinstance(data, str): + # ~ print(isinstance(data, str), data[0]) + if data[0] in '=': + self.obj.setFormula(data) + # ~ print('Set Formula') + else: + self.obj.setString(data) + elif isinstance(data, Decimal): + self.obj.setValue(float(data)) + elif isinstance(data, (int, float, bool)): + self.obj.setValue(data) + elif isinstance(data, datetime.datetime): + d = data.toordinal() + t = (data - datetime.datetime.fromordinal(d)).seconds / SECONDS_DAY + self.obj.setValue(d - DATE_OFFSET + t) + elif isinstance(data, datetime.date): + d = data.toordinal() + self.obj.setValue(d - DATE_OFFSET) + elif isinstance(data, datetime.time): + d = (data.hour * 3600 + data.minute * 60 + data.second) / SECONDS_DAY + self.obj.setValue(d) + + @property + def date(self): + value = int(self.obj.Value) + date = datetime.date.fromordinal(value + DATE_OFFSET) + return date + + @property + def time(self): + seconds = self.obj.Value * SECONDS_DAY + time_delta = datetime.timedelta(seconds=seconds) + time = (datetime.datetime.min + time_delta).time() + return time + + @property + def datetime(self): + return datetime.datetime.combine(self.date, self.time) + + @property + def data(self): + return self.obj.getDataArray() + @data.setter + def data(self, values): + if self._is_cell: + self.to_size(len(values), len(values[0])).data = values + else: + self.obj.setDataArray(values) + + @property + def dict(self): + rows = self.data + k = rows[0] + data = [dict(zip(k, r)) for r in rows[1:]] + return data + @dict.setter + def dict(self, values): + data = [tuple(values[0].keys())] + data += [tuple(d.values()) for d in values] + self.data = data + + @property + def formula(self): + return self.obj.getFormulaArray() + @formula.setter + def formula(self, values): + self.obj.setFormulaArray(values) + + @property + def array_formula(self): + return self.obj.ArrayFormula + @array_formula.setter + def array_formula(self, value): + self.obj.ArrayFormula = value + + @property + def address(self): + return self.obj.CellAddress + + @property + def range_address(self): + return self.obj.RangeAddress + + @property + def cursor(self): + cursor = self.obj.Spreadsheet.createCursorByRange(self.obj) + return cursor + + @property + def current_region(self): + cursor = self.cursor + cursor.collapseToCurrentRegion() + return LOCalcRange(self.sheet[cursor.AbsoluteName].obj) + + @property + def next_cell(self): + a = self.current_region.range_address + col = a.StartColumn + row = a.EndRow + 1 + return LOCalcRange(self.sheet[row, col].obj) + + @property + def position(self): + return self.obj.Position + + @property + def size(self): + return self.obj.Size + + @property + def possize(self): + data = { + 'Width': self.size.Width, + 'Height': self.size.Height, + 'X': self.position.X, + 'Y': self.position.Y, + } + return data + + @property + def visible(self): + cursor = self.cursor + rangos = cursor.queryVisibleCells() + rangos = [LOCalcRange(self.sheet[r.AbsoluteName].obj) for r in rangos] + return tuple(rangos) + + @property + def merged_area(self): + cursor = self.cursor + cursor.collapseToMergedArea() + rango = LOCalcRange(self.sheet[cursor.AbsoluteName].obj) + return rango + + @property + def empty(self): + cursor = self.sheet.get_cursor(self.obj) + cursor = self.cursor + rangos = cursor.queryEmptyCells() + rangos = [LOCalcRange(self.sheet[r.AbsoluteName].obj) for r in rangos] + return tuple(rangos) + + @property + def merge(self): + return self.obj.IsMerged + @merge.setter + def merge(self, value): + self.obj.merge(value) + + @property + def style(self): + return self.obj.CellStyle + @style.setter + def style(self, value): + self.obj.CellStyle = value + + @property + def auto_format(self): + return '' + @auto_format.setter + def auto_format(self, value): + self.obj.autoFormat(value) + + @property + def validation(self): + return self.obj.Validation + @validation.setter + def validation(self, values): + current = self.validation + if not values: + current.Type = ValidationType.ANY + current.ShowInputMessage = False + else: + is_list = False + for k, v in values.items(): + if k == 'Type' and v == VT.LIST: + is_list = True + if k == 'Formula1' and is_list: + if isinstance(v, (tuple, list)): + v = ';'.join(['"{}"'.format(i) for i in v]) + setattr(current, k, v) + self.obj.Validation = current + + def select(self): + self.doc.select(self.obj) + return + + def search(self, options, find_all=True): + rangos = None + + descriptor = self.sheet.search_descriptor + descriptor.setSearchString(options['Search']) + descriptor.SearchCaseSensitive = options.get('CaseSensitive', False) + descriptor.SearchWords = options.get('Words', False) + if hasattr(descriptor, 'SearchRegularExpression'): + descriptor.SearchRegularExpression = options.get('RegularExpression', False) + if hasattr(descriptor, 'SearchType') and 'Type' in options: + descriptor.SearchType = options['Type'] + + if find_all: + found = self.obj.findAll(descriptor) + else: + found = self.obj.findFirst(descriptor) + + if found: + if found.ImplementationName == OBJ_CELL: + rangos = LOCalcRange(found) + else: + rangos = [LOCalcRange(f) for f in found] + + return rangos + + def replace(self, options): + descriptor = self.sheet.replace_descriptor + descriptor.setSearchString(options['Search']) + descriptor.setReplaceString(options['Replace']) + descriptor.SearchCaseSensitive = options.get('CaseSensitive', False) + descriptor.SearchWords = options.get('Words', False) + if hasattr(descriptor, 'SearchRegularExpression'): + descriptor.SearchRegularExpression = options.get('RegularExpression', False) + if hasattr(descriptor, 'SearchType') and 'Type' in options: + descriptor.SearchType = options['Type'] + count = self.obj.replaceAll(descriptor) + return count + + def in_range(self, rango): + if isinstance(rango, LOCalcRange): + address = rango.range_address + else: + address = rango.RangeAddress + result = self.cursor.queryIntersection(address) + return bool(result.Count) + + def offset(self, rows=0, cols=1): + ra = self.range_address + col = ra.EndColumn + cols + row = ra.EndRow + rows + return LOCalcRange(self.sheet[row, col].obj) + + def to_size(self, rows, cols): + cursor = self.cursor + cursor.collapseToSize(cols, rows) + return LOCalcRange(self.sheet[cursor.AbsoluteName].obj) + + def copy(self, source): + self.sheet.obj.copyRange(self.address, source.range_address) + return + + def copy_to(self, cell, formula=False): + rango = cell.to_size(self.rows, self.columns) + if formula: + rango.formula = self.data + else: + rango.data = self.data + return + + def copy_from(self, rango, formula=False): + data = rango + if isinstance(rango, LOCalcRange): + if formula: + data = rango.formula + else: + data = rango.data + rows = len(data) + cols = len(data[0]) + if formula: + self.to_size(rows, cols).formula = data + else: + self.to_size(rows, cols).data = data + return + + def optimal_width(self): + self.obj.Columns.OptimalWidth = True + return + + def clean_render(self, template='\{(\w.+)\}'): + self._sd.SearchRegularExpression = True + self._sd.setSearchString(template) + self.obj.replaceAll(self._sd) + return + + def render(self, data, clean=True): + self._sd = self.sheet.obj.createSearchDescriptor() + self._sd.SearchCaseSensitive = False + for k, v in data.items(): + cell = self._render_value(k, v) + return cell + + def _render_value(self, key, value, parent=''): + cell = None + if isinstance(value, dict): + for k, v in value.items(): + cell = self._render_value(k, v, key) + return cell + elif isinstance(value, (list, tuple)): + self._render_list(key, value) + return + + search = f'{{{key}}}' + if parent: + search = f'{{{parent}.{key}}}' + ranges = self.find_all(search) + + for cell in ranges or range(0): + self._set_new_value(cell, search, value) + return LOCalcRange(cell) + + def _set_new_value(self, cell, search, value): + if not cell.ImplementationName == 'ScCellObj': + return + + if isinstance(value, str): + pattern = re.compile(search, re.IGNORECASE) + new_value = pattern.sub(value, cell.String) + cell.String = new_value + else: + LOCalcRange(cell).value = value + return + + def _render_list(self, key, rows): + for row in rows: + for k, v in row.items(): + self._render_value(k, v) + return + + def find(self, search_string): + if self._sd is None: + self._sd = self.sheet.obj.createSearchDescriptor() + self._sd.SearchCaseSensitive = False + + self._sd.setSearchString(search_string) + cell = self.obj.findFirst(self._sd) + if cell: + cell = LOCalcRange(cell) + return cell + + def find_all(self, search_string): + if self._sd is None: + self._sd = self.sheet.obj.createSearchDescriptor() + self._sd.SearchCaseSensitive = False + + self._sd.setSearchString(search_string) + ranges = self.obj.findAll(self._sd) + return ranges + + def filter(self, args, with_headers=True): + ff = TableFilterField() + ff.Field = args['Field'] + ff.Operator = args['Operator'] + if isinstance(args['Value'], str): + ff.IsNumeric = False + ff.StringValue = args['Value'] + else: + ff.IsNumeric = True + ff.NumericValue = args['Value'] + + fd = self.obj.createFilterDescriptor(True) + fd.ContainsHeader = with_headers + fd.FilterFields = ((ff,)) + # ~ self.obj.AutoFilter = True + self.obj.filter(fd) + return + + def copy_format_from(self, rango): + rango.select() + self.doc.copy() + self.select() + args = { + 'Flags': 'T', + 'MoveMode': 4, + } + url = '.uno:InsertContents' + call_dispatch(self.doc.frame, url, args) + return + + def to_image(self): + self.select() + self.doc.copy() + args = {'SelectedFormat': 141} + url = '.uno:ClipboardFormatItems' + call_dispatch(self.doc.frame, url, args) + return self.sheet.shapes[-1] + + def insert_image(self, path, args={}): + ps = self.possize + args['Width'] = args.get('Width', ps['Width']) + args['Height'] = args.get('Height', ps['Height']) + args['X'] = args.get('X', ps['X']) + args['Y'] = args.get('Y', ps['Y']) + # ~ img.ResizeWithCell = True + img = self.sheet.dp.insert_image(path, args) + img.anchor = self.obj + args.clear() + return img + + def insert_shape(self, tipo, args={}): + ps = self.possize + args['Width'] = args.get('Width', ps['Width']) + args['Height'] = args.get('Height', ps['Height']) + args['X'] = args.get('X', ps['X']) + args['Y'] = args.get('Y', ps['Y']) + + shape = self.sheet.dp.add(tipo, args) + shape.anchor = self.obj + args.clear() + return + + def filter_by_color(self, cell): + rangos = cell.column[1:,:].visible + for r in rangos: + for c in r: + if c.back_color != cell.back_color: + c.rows.visible = False + return + + def clear(self, what=1023): + # ~ http://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1sheet_1_1CellFlags.html + self.obj.clearContents(what) + return + + def transpose(self): + # ~ 'Flags': 'A', + # ~ 'FormulaCommand': 0, + # ~ 'SkipEmptyCells': False, + # ~ 'AsLink': False, + # ~ 'MoveMode': 4, + self.select() + self.doc.copy() + self.clear(1023) + self[0,0].select() + self.doc.insert_contents({'Transpose': True}) + _CB.set('') + return + + def transpose_data(self, formula=False): + data = self.data + if formula: + data = self.formula + data = tuple(zip(*data)) + self.clear(1023) + self[0,0].copy_from(data, formula=formula) + return + + def merge_by_row(self): + for r in range(len(self.rows)): + self[r].merge = True + return + + def fill(self, source=1): + self.obj.fillAuto(0, source) + return + + +class LOWriterPageStyle(LOBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + def __str__(self): + return f'Page Style: {self.name}' + + @property + def name(self): + return self._obj.Name + + +class LOWriterPageStyles(object): + + def __init__(self, styles): + self._styles = styles + + def __getitem__(self, index): + return LOWriterPageStyle(self._styles[index]) + + @property + def names(self): + return self._styles.ElementNames + + def __str__(self): + return '\n'.join(self.names) + + +class LOWriterTextRange(object): + + def __init__(self, obj, doc): + self._obj = obj + self._doc = doc + self._is_paragraph = self.obj.ImplementationName == 'SwXParagraph' + self._is_table = self.obj.ImplementationName == 'SwXTextTable' + + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + for i, p in enumerate(self.obj): + if i == self._index: + obj = LOWriterTextRange(p, self._doc) + self._index += 1 + return obj + raise StopIteration + + @property + def obj(self): + return self._obj + + @property + def string(self): + s = '' + if self._is_paragraph: + s = self.obj.String + return s + @string.setter + def string(self, value): + self.obj.String = value + + @property + def value(self): + return self.string + + @property + def is_table(self): + return self._is_table + + @property + def text(self): + return self.obj.Text + + @property + def cursor(self): + return self.text.createTextCursorByRange(self.obj) + + @property + def dp(self): + return self._doc.dp + + def offset(self): + cursor = self.cursor.getEnd() + return LOWriterTextRange(cursor, self._doc) + + def insert_content(self, data, cursor=None, replace=False): + if cursor is None: + cursor = self.cursor + self.text.insertTextContent(cursor, data, replace) + return + + def new_line(self, count=1): + cursor = self.cursor + for i in range(count): + self.text.insertControlCharacter(cursor, PARAGRAPH_BREAK, False) + return self._doc.selection + + def insert_table(self, data): + table = self._doc.create_instance('com.sun.star.text.TextTable') + rows = len(data) + cols = len(data[0]) + table.initialize(rows, cols) + self.insert_content(table) + table.DataArray = data + name = table.Name + table = LOWriterTextTable(self._doc.tables[name], self._doc) + return table + + def insert_image(self, path, args={}): + w = args.get('Width', 1000) + h = args.get('Height', 1000) + image = self._doc.create_instance('com.sun.star.text.GraphicObject') + image.GraphicURL = _P.to_url(path) + image.AnchorType = AS_CHARACTER + image.Width = w + image.Height = h + self.insert_content(image) + return self._doc.dp.last + + +class LOWriterTextRanges(object): + + def __init__(self, obj, doc): + self._obj = obj + self._doc = doc + + def __getitem__(self, index): + for i, p in enumerate(self.obj): + if i == index: + obj = LOWriterTextRange(p, self._doc) + break + return obj + + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + for i, p in enumerate(self.obj): + if i == self._index: + obj = LOWriterTextRange(p, self._doc) + self._index += 1 + return obj + raise StopIteration + + @property + def obj(self): + return self._obj + + +class LOWriterTextTable(object): + + def __init__(self, obj, doc): + self._obj = obj + self._doc = doc + + @property + def obj(self): + return self._obj + + @property + def name(self): + return self._obj.Name + + @property + def data(self): + return self._obj.DataArray + @data.setter + def data(self, values): + self._obj.DataArray = values + + +class LOWriterTextTables(object): + + def __init__(self, doc): + self._doc = doc + self._obj = doc.obj.TextTables + + def __getitem__(self, key): + return LOWriterTextTable(self._obj[key], self._doc) + + def __len__(self): + return self._obj.Count + + def insert(self, data, text_range=None): + if text_range is None: + text_range = self._doc.selection + text_range.insert_table(data) + return + + +class LOWriter(LODocument): + + def __init__(self, obj): + super().__init__(obj) + self._type = WRITER + + @property + def text(self): + return LOWriterTextRange(self.obj.Text, self) + + @property + def paragraphs(self): + return LOWriterTextRanges(self.obj.Text, self) + + @property + def tables(self): + return LOWriterTextTables(self) + + @property + def selection(self): + sel = self.obj.CurrentSelection + if sel.ImplementationName == OBJ_TEXTS: + if len(sel) == 1: + sel = LOWriterTextRanges(sel, self)[0] + else: + sel = LOWriterTextRanges(sel, self) + return sel + + if sel.ImplementationName == OBJ_SHAPES: + if len(sel) == 1: + sel = sel[0] + sel = LODrawPage(sel.Parent)[sel.Name] + return sel + + if sel.ImplementationName == OBJ_GRAPHIC: + sel = self.dp[sel.Name] + else: + debug(sel.ImplementationName) + + return sel + + @property + def dp(self): + return self.draw_page + @property + def draw_page(self): + return LODrawPage(self.obj.DrawPage) + + @property + def view_cursor(self): + return self._cc.ViewCursor + + @property + def cursor(self): + return self.obj.Text.createTextCursor() + + @property + def page_styles(self): + ps = self.obj.StyleFamilies['PageStyles'] + return LOWriterPageStyles(ps) + + @property + def search_descriptor(self): + return self.obj.createSearchDescriptor() + + @property + def replace_descriptor(self): + return self.obj.createReplaceDescriptor() + + def goto_start(self): + self.view_cursor.gotoStart(False) + return self.selection + + def goto_end(self): + self.view_cursor.gotoEnd(False) + return self.selection + + def search(self, options, find_all=True): + descriptor = self.search_descriptor + descriptor.setSearchString(options.get('Search', '')) + descriptor.SearchCaseSensitive = options.get('CaseSensitive', False) + descriptor.SearchWords = options.get('Words', False) + if 'Attributes' in options: + attr = dict_to_property(options['Attributes']) + descriptor.setSearchAttributes(attr) + if hasattr(descriptor, 'SearchRegularExpression'): + descriptor.SearchRegularExpression = options.get('RegularExpression', False) + if hasattr(descriptor, 'SearchType') and 'Type' in options: + descriptor.SearchType = options['Type'] + + result = False + if find_all: + found = self.obj.findAll(descriptor) + if len(found): + result = [LOWriterTextRange(f, self) for f in found] + else: + found = self.obj.findFirst(descriptor) + if found: + result = LOWriterTextRange(found, self) + + return result + + def replace(self, options): + descriptor = self.replace_descriptor + descriptor.setSearchString(options['Search']) + descriptor.setReplaceString(options['Replace']) + descriptor.SearchCaseSensitive = options.get('CaseSensitive', False) + descriptor.SearchWords = options.get('Words', False) + if 'Attributes' in options: + attr = dict_to_property(options['Attributes']) + descriptor.setSearchAttributes(attr) + if hasattr(descriptor, 'SearchRegularExpression'): + descriptor.SearchRegularExpression = options.get('RegularExpression', False) + if hasattr(descriptor, 'SearchType') and 'Type' in options: + descriptor.SearchType = options['Type'] + found = self.obj.replaceAll(descriptor) + return found + + def select(self, text): + if hasattr(text, 'obj'): + text = text.obj + self._cc.select(text) + return + + +class LOShape(LOBaseObject): + IMAGE = 'com.sun.star.drawing.GraphicObjectShape' + + def __init__(self, obj, index): + self._index = index + super().__init__(obj) + + @property + def type(self): + t = self.shape_type[21:] + if self.is_image: + t = 'image' + return t + + @property + def shape_type(self): + return self.obj.ShapeType + + @property + def is_image(self): + return self.shape_type == self.IMAGE + + @property + def name(self): + return self.obj.Name or f'{self.type}{self.index}' + @name.setter + def name(self, value): + self.obj.Name = value + + @property + def index(self): + return self._index + + @property + def size(self): + s = self.obj.Size + a = dict(Width=s.Width, Height=s.Height) + return a + + @property + def string(self): + return self.obj.String + @string.setter + def string(self, value): + self.obj.String = value + + @property + def description(self): + return self.obj.Description + @description.setter + def description(self, value): + self.obj.Description = value + + @property + def cell(self): + return self.anchor + + @property + def anchor(self): + obj = self.obj.Anchor + if obj.ImplementationName == OBJ_CELL: + obj = LOCalcRange(obj) + elif obj.ImplementationName == OBJ_TEXT: + obj = LOWriterTextRange(obj, LODocs().active) + else: + debug('Anchor', obj.ImplementationName) + return obj + @anchor.setter + def anchor(self, value): + if hasattr(value, 'obj'): + value = value.obj + self.obj.Anchor = value + + @property + def visible(self): + return self.obj.Visible + @visible.setter + def visible(self, value): + self.obj.Visible = value + + @property + def path(self): + return self.url + @property + def url(self): + url = '' + if self.is_image: + url = _P.to_system(self.obj.GraphicURL.OriginURL) + return url + + @property + def mimetype(self): + mt = '' + if self.is_image: + mt = self.obj.GraphicURL.MimeType + return mt + + @property + def linked(self): + l = False + if self.is_image: + l = self.obj.GraphicURL.Linked + return l + + def delete(self): + self.remove() + return + def remove(self): + self.obj.Parent.remove(self.obj) + return + + def save(self, path: str, mimetype=DEFAULT_MIME_TYPE): + if _P.is_dir(path): + name = self.name + ext = mimetype.lower() + else: + p = _P(path) + path = p.path + name = p.name + ext = p.ext.lower() + + path = _P.join(path, f'{name}.{ext}') + args = dict( + URL = _P.to_url(path), + MimeType = MIME_TYPE[ext], + ) + if not _export_image(self.obj, args): + path = '' + return path + + # ~ def save2(self, path: str): + # ~ size = len(self.obj.Bitmap.DIB) + # ~ data = self.obj.GraphicStream.readBytes((), size) + # ~ data = data[-1].value + # ~ path = _P.join(path, f'{self.name}.png') + # ~ _P.save_bin(path, b'') + # ~ return + + +class LODrawPage(LOBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + def __getitem__(self, index): + if isinstance(index, int): + shape = LOShape(self.obj[index], index) + else: + for i, o in enumerate(self.obj): + shape = self.obj[i] + name = shape.Name or f'shape{i}' + if name == index: + shape = LOShape(shape, i) + break + return shape + + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + if self._index == self.count: + raise StopIteration + shape = self[self._index] + self._index += 1 + return shape + + + @property + def name(self): + return self.obj.Name + + @property + def doc(self): + return self.obj.Forms.Parent + + @property + def width(self): + return self.obj.Width + + @property + def height(self): + return self.obj.Height + + @property + def count(self): + return self.obj.Count + + @property + def last(self): + return self[self.count - 1] + + def create_instance(self, name): + return self.doc.createInstance(name) + + def add(self, type_shape, args={}): + """Insert a shape in page, type shapes: + Line + Rectangle + Ellipse + Text + """ + index = self.count + w = args.get('Width', 3000) + h = args.get('Height', 3000) + x = args.get('X', 1000) + y = args.get('Y', 1000) + name = args.get('Name', f'{type_shape.lower()}{index}') + + service = f'com.sun.star.drawing.{type_shape}Shape' + shape = self.create_instance(service) + shape.Size = Size(w, h) + shape.Position = Point(x, y) + shape.Name = name + self.obj.add(shape) + return LOShape(self.obj[index], index) + + def remove(self, shape): + if hasattr(shape, 'obj'): + shape = shape.obj + return self.obj.remove(shape) + + def remove_all(self): + while self.count: + self.obj.remove(self.obj[0]) + return + + def insert_image(self, path, args={}): + index = self.count + w = args.get('Width', 3000) + h = args.get('Height', 3000) + x = args.get('X', 1000) + y = args.get('Y', 1000) + name = args.get('Name', f'image{index}') + + image = self.create_instance('com.sun.star.drawing.GraphicObjectShape') + image.GraphicURL = _P.to_url(path) + image.Size = Size(w, h) + image.Position = Point(x, y) + image.Name = name + self.obj.add(image) + return LOShape(self.obj[index], index) + + +class LODrawImpress(LODocument): + + def __init__(self, obj): + super().__init__(obj) + + def __getitem__(self, index): + if isinstance(index, int): + page = self.obj.DrawPages[index] + else: + page = self.obj.DrawPages.getByName(index) + return LODrawPage(page) + + @property + def selection(self): + sel = self.obj.CurrentSelection[0] + # ~ return _get_class_uno(sel) + return sel + + @property + def current_page(self): + return LODrawPage(self._cc.getCurrentPage()) + + def paste(self): + call_dispatch(self.frame, '.uno:Paste') + return self.current_page[-1] + + def add(self, type_shape, args={}): + return self.current_page.add(type_shape, args) + + def insert_image(self, path, args={}): + self.current_page.insert_image(path, args) + return + + # ~ def export(self, path, mimetype='png'): + # ~ args = dict( + # ~ URL = _P.to_url(path), + # ~ MimeType = MIME_TYPE[mimetype], + # ~ ) + # ~ result = _export_image(self.obj, args) + # ~ return result + + +class LODraw(LODrawImpress): + + def __init__(self, obj): + super().__init__(obj) + self._type = DRAW + + +class LOImpress(LODrawImpress): + + def __init__(self, obj): + super().__init__(obj) + self._type = IMPRESS + + +class BaseDateField(DateField): + + def db_value(self, value): + return _date_to_struct(value) + + def python_value(self, value): + return _struct_to_date(value) + + +class BaseTimeField(TimeField): + + def db_value(self, value): + return _date_to_struct(value) + + def python_value(self, value): + return _struct_to_date(value) + + +class BaseDateTimeField(DateTimeField): + + def db_value(self, value): + return _date_to_struct(value) + + def python_value(self, value): + return _struct_to_date(value) + + +class FirebirdDatabase(Database): + field_types = {'BOOL': 'BOOLEAN', 'DATETIME': 'TIMESTAMP'} + + def __init__(self, database, **kwargs): + super().__init__(database, **kwargs) + self._db = database + + def _connect(self): + return self._db + + def create_tables(self, models, **options): + options['safe'] = False + tables = self._db.tables + models = [m for m in models if not m.__name__.lower() in tables] + super().create_tables(models, **options) + + def execute_sql(self, sql, params=None, commit=True): + with __exception_wrapper__: + cursor = self._db.execute(sql, params) + return cursor + + def last_insert_id(self, cursor, query_type=None): + # ~ debug('LAST_ID', cursor) + return 0 + + def rows_affected(self, cursor): + return self._db.rows_affected + + @property + def path(self): + return self._db.path + + +class BaseRow: + pass + + +class BaseQuery(object): + PY_TYPES = { + 'SQL_LONG': 'getLong', + 'SQL_VARYING': 'getString', + 'SQL_FLOAT': 'getFloat', + 'SQL_BOOLEAN': 'getBoolean', + 'SQL_TYPE_DATE': 'getDate', + 'SQL_TYPE_TIME': 'getTime', + 'SQL_TIMESTAMP': 'getTimestamp', + } + TYPES_DATE = ('SQL_TYPE_DATE', 'SQL_TYPE_TIME', 'SQL_TIMESTAMP') + + def __init__(self, query): + self._query = query + self._meta = query.MetaData + self._cols = self._meta.ColumnCount + self._names = query.Columns.ElementNames + self._data = self._get_data() + + def __getitem__(self, index): + return self._data[index] + + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + try: + row = self._data[self._index] + except IndexError: + raise StopIteration + self._index += 1 + return row + + def _to_python(self, index): + type_field = self._meta.getColumnTypeName(index) + value = getattr(self._query, self.PY_TYPES[type_field])(index) + if type_field in self.TYPES_DATE: + value = _struct_to_date(value) + return value + + def _get_row(self): + row = BaseRow() + for i in range(1, self._cols + 1): + column_name = self._meta.getColumnName(i) + value = self._to_python(i) + setattr(row, column_name, value) + return row + + def _get_data(self): + data = [] + while self._query.next(): + row = self._get_row() + data.append(row) + return data + + @property + def tuples(self): + data = [tuple(r.__dict__.values()) for r in self._data] + return tuple(data) + + @property + def dicts(self): + data = [r.__dict__ for r in self._data] + return tuple(data) + + +class LOBase(object): + DB_TYPES = { + str: 'setString', + int: 'setInt', + float: 'setFloat', + bool: 'setBoolean', + Date: 'setDate', + Time: 'setTime', + DateTime: 'setTimestamp', + } + # ~ setArray + # ~ setBinaryStream + # ~ setBlob + # ~ setByte + # ~ setBytes + # ~ setCharacterStream + # ~ setClob + # ~ setNull + # ~ setObject + # ~ setObjectNull + # ~ setObjectWithInfo + # ~ setPropertyValue + # ~ setRef + + def __init__(self, obj, args={}): + self._obj = obj + self._type = BASE + self._dbc = create_instance('com.sun.star.sdb.DatabaseContext') + self._rows_affected = 0 + path = args.get('path', '') + self._path = _P(path) + self._name = self._path.name + if _P.exists(path): + if not self.is_registered: + self.register() + db = self._dbc.getByName(self.name) + else: + db = self._dbc.createInstance() + db.URL = 'sdbc:embedded:firebird' + db.DatabaseDocument.storeAsURL(self._path.url, ()) + self.register() + self._obj = db + self._con = db.getConnection('', '') + + def __contains__(self, item): + return item in self.tables + + @property + def obj(self): + return self._obj + + @property + def name(self): + return self._name + + @property + def path(self): + return str(self._path) + + @property + def is_registered(self): + return self._dbc.hasRegisteredDatabase(self.name) + + @property + def tables(self): + tables = [t.Name.lower() for t in self._con.getTables()] + return tables + + @property + def rows_affected(self): + return self._rows_affected + + def register(self): + if not self.is_registered: + self._dbc.registerDatabaseLocation(self.name, self._path.url) + return + + def revoke(self, name): + self._dbc.revokeDatabaseLocation(name) + return True + + def save(self): + self.obj.DatabaseDocument.store() + self.refresh() + return + + def close(self): + self._con.close() + return + + def refresh(self): + self._con.getTables().refresh() + return + + def initialize(self, database_proxy, tables): + db = FirebirdDatabase(self) + database_proxy.initialize(db) + db.create_tables(tables) + return + + def _validate_sql(self, sql, params): + limit = ' LIMIT ' + for p in params: + sql = sql.replace('?', f"'{p}'", 1) + if limit in sql: + sql = sql.split(limit)[0] + sql = sql.replace('SELECT', f'SELECT FIRST {params[-1]}') + return sql + + def cursor(self, sql, params): + if sql.startswith('SELECT'): + sql = self._validate_sql(sql, params) + cursor = self._con.prepareStatement(sql) + return cursor + + if not params: + cursor = self._con.createStatement() + return cursor + + cursor = self._con.prepareStatement(sql) + for i, v in enumerate(params, 1): + t = type(v) + if not t in self.DB_TYPES: + error('Type not support') + debug((i, t, v, self.DB_TYPES[t])) + getattr(cursor, self.DB_TYPES[t])(i, v) + return cursor + + def execute(self, sql, params): + debug(sql, params) + cursor = self.cursor(sql, params) + + if sql.startswith('SELECT'): + result = cursor.executeQuery() + elif params: + result = cursor.executeUpdate() + self._rows_affected = result + self.save() + else: + result = cursor.execute(sql) + self.save() + + return result + + def select(self, sql): + debug('SELECT', sql) + if not sql.startswith('SELECT'): + return () + + cursor = self._con.prepareStatement(sql) + query = cursor.executeQuery() + return BaseQuery(query) + + def get_query(self, query): + sql, args = query.sql() + sql = self._validate_sql(sql, args) + return self.select(sql) + + +class LOMath(LODocument): + + def __init__(self, obj): + super().__init__(obj) + self._type = MATH + + +class LOBasic(LODocument): + + def __init__(self, obj): + super().__init__(obj) + self._type = BASIC + + +class LODocs(object): + _desktop = None + + def __init__(self): + self._desktop = get_desktop() + LODocs._desktop = self._desktop + + def __getitem__(self, index): + document = None + for i, doc in enumerate(self._desktop.Components): + if isinstance(index, int) and i == index: + document = _get_class_doc(doc) + break + elif isinstance(index, str) and doc.Title == index: + document = _get_class_doc(doc) + break + return document + + def __contains__(self, item): + doc = self[item] + return not doc is None + + def __iter__(self): + self._i = -1 + return self + + def __next__(self): + self._i += 1 + doc = self[self._i] + if doc is None: + raise StopIteration + else: + return doc + + def __len__(self): + for i, _ in enumerate(self._desktop.Components): + pass + return i + 1 + + @property + def active(self): + return _get_class_doc(self._desktop.getCurrentComponent()) + + @classmethod + def new(cls, type_doc=CALC, args={}): + if type_doc == BASE: + return LOBase(None, args) + + path = f'private:factory/s{type_doc}' + opt = dict_to_property(args) + doc = cls._desktop.loadComponentFromURL(path, '_default', 0, opt) + return _get_class_doc(doc) + + @classmethod + def open(cls, path, args={}): + """ Open document in path + Usually options: + Hidden: True or False + AsTemplate: True or False + ReadOnly: True or False + Password: super_secret + MacroExecutionMode: 4 = Activate macros + Preview: True or False + + http://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1frame_1_1XComponentLoader.html + http://api.libreoffice.org/docs/idl/ref/servicecom_1_1sun_1_1star_1_1document_1_1MediaDescriptor.html + """ + path = _P.to_url(path) + opt = dict_to_property(args) + doc = cls._desktop.loadComponentFromURL(path, '_default', 0, opt) + if doc is None: + return + + return _get_class_doc(doc) + + def connect(self, path): + return LOBase(None, {'path': path}) + + +def _add_listeners(events, control, name=''): + listeners = { + 'addActionListener': EventsButton, + 'addMouseListener': EventsMouse, + 'addFocusListener': EventsFocus, + 'addItemListener': EventsItem, + 'addKeyListener': EventsKey, + 'addTabListener': EventsTab, + } + if hasattr(control, 'obj'): + control = control.obj + # ~ debug(control.ImplementationName) + is_grid = control.ImplementationName == 'stardiv.Toolkit.GridControl' + is_link = control.ImplementationName == 'stardiv.Toolkit.UnoFixedHyperlinkControl' + is_roadmap = control.ImplementationName == 'stardiv.Toolkit.UnoRoadmapControl' + is_pages = control.ImplementationName == 'stardiv.Toolkit.UnoMultiPageControl' + + for key, value in listeners.items(): + if hasattr(control, key): + if is_grid and key == 'addMouseListener': + control.addMouseListener(EventsMouseGrid(events, name)) + continue + if is_link and key == 'addMouseListener': + control.addMouseListener(EventsMouseLink(events, name)) + continue + if is_roadmap and key == 'addItemListener': + control.addItemListener(EventsItemRoadmap(events, name)) + continue + + getattr(control, key)(listeners[key](events, name)) + + if is_grid: + controllers = EventsGrid(events, name) + control.addSelectionListener(controllers) + control.Model.GridDataModel.addGridDataListener(controllers) + return + + +def _set_properties(model, properties): + if 'X' in properties: + properties['PositionX'] = properties.pop('X') + if 'Y' in properties: + properties['PositionY'] = properties.pop('Y') + keys = tuple(properties.keys()) + values = tuple(properties.values()) + model.setPropertyValues(keys, values) + return + + +class EventsListenerBase(unohelper.Base, XEventListener): + + def __init__(self, controller, name, window=None): + self._controller = controller + self._name = name + self._window = window + + @property + def name(self): + return self._name + + def disposing(self, event): + self._controller = None + if not self._window is None: + self._window.setMenuBar(None) + + +class EventsMouse(EventsListenerBase, XMouseListener, XMouseMotionListener): + + def __init__(self, controller, name): + super().__init__(controller, name) + + def mousePressed(self, event): + event_name = '{}_click'.format(self._name) + if event.ClickCount == 2: + event_name = '{}_double_click'.format(self._name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + def mouseReleased(self, event): + pass + + def mouseEntered(self, event): + pass + + def mouseExited(self, event): + pass + + # ~ XMouseMotionListener + def mouseMoved(self, event): + pass + + def mouseDragged(self, event): + pass + + +class EventsMouseLink(EventsMouse): + + def __init__(self, controller, name): + super().__init__(controller, name) + self._text_color = 0 + + def mouseEntered(self, event): + model = event.Source.Model + self._text_color = model.TextColor or 0 + model.TextColor = get_color('blue') + return + + def mouseExited(self, event): + model = event.Source.Model + model.TextColor = self._text_color + return + + +class EventsButton(EventsListenerBase, XActionListener): + + def __init__(self, controller, name): + super().__init__(controller, name) + + def actionPerformed(self, event): + event_name = f'{self.name}_action' + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + +class EventsFocus(EventsListenerBase, XFocusListener): + CONTROLS = ( + 'stardiv.Toolkit.UnoControlEditModel', + ) + + def __init__(self, controller, name): + super().__init__(controller, name) + + def focusGained(self, event): + service = event.Source.Model.ImplementationName + # ~ print('Focus enter', service) + if service in self.CONTROLS: + obj = event.Source.Model + obj.BackgroundColor = COLOR_ON_FOCUS + return + + def focusLost(self, event): + service = event.Source.Model.ImplementationName + if service in self.CONTROLS: + obj = event.Source.Model + obj.BackgroundColor = -1 + return + + +class EventsKey(EventsListenerBase, XKeyListener): + """ + event.KeyChar + event.KeyCode + event.KeyFunc + event.Modifiers + """ + + def __init__(self, controller, name): + super().__init__(controller, name) + + def keyPressed(self, event): + pass + + def keyReleased(self, event): + event_name = '{}_key_released'.format(self._name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + # ~ else: + # ~ if event.KeyFunc == QUIT and hasattr(self._cls, 'close'): + # ~ self._cls.close() + return + + +class EventsItem(EventsListenerBase, XItemListener): + + def __init__(self, controller, name): + super().__init__(controller, name) + + def disposing(self, event): + pass + + def itemStateChanged(self, event): + event_name = '{}_item_changed'.format(self.name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + +class EventsItemRoadmap(EventsItem): + + def itemStateChanged(self, event): + dialog = event.Source.Context.Model + dialog.Step = event.ItemId + 1 + return + + +class EventsGrid(EventsListenerBase, XGridDataListener, XGridSelectionListener): + + def __init__(self, controller, name): + super().__init__(controller, name) + + def dataChanged(self, event): + event_name = '{}_data_changed'.format(self.name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + def rowHeadingChanged(self, event): + pass + + def rowsInserted(self, event): + pass + + def rowsRemoved(self, evemt): + pass + + def selectionChanged(self, event): + event_name = '{}_selection_changed'.format(self.name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + +class EventsMouseGrid(EventsMouse): + selected = False + + def mousePressed(self, event): + super().mousePressed(event) + # ~ obj = event.Source + # ~ col = obj.getColumnAtPoint(event.X, event.Y) + # ~ row = obj.getRowAtPoint(event.X, event.Y) + # ~ print(col, row) + # ~ if col == -1 and row == -1: + # ~ if self.selected: + # ~ obj.deselectAllRows() + # ~ else: + # ~ obj.selectAllRows() + # ~ self.selected = not self.selected + return + + def mouseReleased(self, event): + # ~ obj = event.Source + # ~ col = obj.getColumnAtPoint(event.X, event.Y) + # ~ row = obj.getRowAtPoint(event.X, event.Y) + # ~ if row == -1 and col > -1: + # ~ gdm = obj.Model.GridDataModel + # ~ for i in range(gdm.RowCount): + # ~ gdm.updateRowHeading(i, i + 1) + return + + +class EventsTab(EventsListenerBase, XTabListener): + + def __init__(self, controller, name): + super().__init__(controller, name) + + def activated(self, id): + event_name = '{}_activated'.format(self.name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(id) + return + + +class EventsMenu(EventsListenerBase, XMenuListener): + + def __init__(self, controller): + super().__init__(controller, '') + + def itemHighlighted(self, event): + pass + + def itemSelected(self, event): + name = event.Source.getCommand(event.MenuId) + if name.startswith('menu'): + event_name = '{}_selected'.format(name) + else: + event_name = 'menu_{}_selected'.format(name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + def itemActivated(self, event): + return + + def itemDeactivated(self, event): + return + + +class EventsWindow(EventsListenerBase, XTopWindowListener, XWindowListener): + + def __init__(self, cls): + self._cls = cls + super().__init__(cls.events, cls.name, cls._window) + + def windowOpened(self, event): + event_name = '{}_opened'.format(self._name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + def windowActivated(self, event): + control_name = '{}_activated'.format(event.Source.Model.Name) + if hasattr(self._controller, control_name): + getattr(self._controller, control_name)(event) + return + + def windowDeactivated(self, event): + control_name = '{}_deactivated'.format(event.Source.Model.Name) + if hasattr(self._controller, control_name): + getattr(self._controller, control_name)(event) + return + + def windowMinimized(self, event): + pass + + def windowNormalized(self, event): + pass + + def windowClosing(self, event): + if self._window: + control_name = 'window_closing' + else: + control_name = '{}_closing'.format(event.Source.Model.Name) + + if hasattr(self._controller, control_name): + getattr(self._controller, control_name)(event) + # ~ else: + # ~ if not self._modal and not self._block: + # ~ event.Source.Visible = False + return + + def windowClosed(self, event): + control_name = '{}_closed'.format(event.Source.Model.Name) + if hasattr(self._controller, control_name): + getattr(self._controller, control_name)(event) + return + + # ~ XWindowListener + def windowResized(self, event): + sb = self._cls._subcont + sb.setPosSize(0, 0, event.Width, event.Height, SIZE) + event_name = '{}_resized'.format(self._name) + if hasattr(self._controller, event_name): + getattr(self._controller, event_name)(event) + return + + def windowMoved(self, event): + pass + + def windowShown(self, event): + pass + + def windowHidden(self, event): + pass + + +# ~ BorderColor = ? +# ~ FontStyleName = ? +# ~ HelpURL = ? +class UnoBaseObject(object): + + def __init__(self, obj, path=''): + self._obj = obj + self._model = obj.Model + + def __setattr__(self, name, value): + exists = hasattr(self, name) + if not exists and not name in ('_obj', '_model'): + setattr(self._model, name, value) + else: + super().__setattr__(name, value) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + @property + def obj(self): + return self._obj + + @property + def model(self): + return self._model + @property + def m(self): + return self._model + + @property + def properties(self): + return {} + @properties.setter + def properties(self, values): + _set_properties(self.model, values) + + @property + def name(self): + return self.model.Name + + @property + def parent(self): + return self.obj.Context + + @property + def tag(self): + return self.model.Tag + @tag.setter + def tag(self, value): + self.model.Tag = value + + @property + def visible(self): + return self.obj.Visible + @visible.setter + def visible(self, value): + self.obj.setVisible(value) + + @property + def enabled(self): + return self.model.Enabled + @enabled.setter + def enabled(self, value): + self.model.Enabled = value + + @property + def step(self): + return self.model.Step + @step.setter + def step(self, value): + self.model.Step = value + + @property + def align(self): + return self.model.Align + @align.setter + def align(self, value): + self.model.Align = value + + @property + def valign(self): + return self.model.VerticalAlign + @valign.setter + def valign(self, value): + self.model.VerticalAlign = value + + @property + def font_weight(self): + return self.model.FontWeight + @font_weight.setter + def font_weight(self, value): + self.model.FontWeight = value + + @property + def font_height(self): + return self.model.FontHeight + @font_height.setter + def font_height(self, value): + self.model.FontHeight = value + + @property + def font_name(self): + return self.model.FontName + @font_name.setter + def font_name(self, value): + self.model.FontName = value + + @property + def font_underline(self): + return self.model.FontUnderline + @font_underline.setter + def font_underline(self, value): + self.model.FontUnderline = value + + @property + def text_color(self): + return self.model.TextColor + @text_color.setter + def text_color(self, value): + self.model.TextColor = value + + @property + def back_color(self): + return self.model.BackgroundColor + @back_color.setter + def back_color(self, value): + self.model.BackgroundColor = value + + @property + def multi_line(self): + return self.model.MultiLine + @multi_line.setter + def multi_line(self, value): + self.model.MultiLine = value + + @property + def help_text(self): + return self.model.HelpText + @help_text.setter + def help_text(self, value): + self.model.HelpText = value + + @property + def border(self): + return self.model.Border + @border.setter + def border(self, value): + # ~ Bug for report + self.model.Border = value + + @property + def width(self): + return self._model.Width + @width.setter + def width(self, value): + self.model.Width = value + + @property + def height(self): + return self.model.Height + @height.setter + def height(self, value): + self.model.Height = value + + def _get_possize(self, name): + ps = self.obj.getPosSize() + return getattr(ps, name) + + def _set_possize(self, name, value): + ps = self.obj.getPosSize() + setattr(ps, name, value) + self.obj.setPosSize(ps.X, ps.Y, ps.Width, ps.Height, POSSIZE) + return + + @property + def x(self): + if hasattr(self.model, 'PositionX'): + return self.model.PositionX + return self._get_possize('X') + @x.setter + def x(self, value): + if hasattr(self.model, 'PositionX'): + self.model.PositionX = value + else: + self._set_possize('X', value) + + @property + def y(self): + if hasattr(self.model, 'PositionY'): + return self.model.PositionY + return self._get_possize('Y') + @y.setter + def y(self, value): + if hasattr(self.model, 'PositionY'): + self.model.PositionY = value + else: + self._set_possize('Y', value) + + @property + def tab_index(self): + return self._model.TabIndex + @tab_index.setter + def tab_index(self, value): + self.model.TabIndex = value + + @property + def tab_stop(self): + return self._model.Tabstop + @tab_stop.setter + def tab_stop(self, value): + self.model.Tabstop = value + + @property + def ps(self): + ps = self.obj.getPosSize() + return ps + @ps.setter + def ps(self, ps): + self.obj.setPosSize(ps.X, ps.Y, ps.Width, ps.Height, POSSIZE) + + def set_focus(self): + self.obj.setFocus() + return + + def ps_from(self, source): + self.ps = source.ps + return + + def center(self, horizontal=True, vertical=False): + p = self.parent.Model + w = p.Width + h = p.Height + if horizontal: + x = w / 2 - self.width / 2 + self.x = x + if vertical: + y = h / 2 - self.height / 2 + self.y = y + return + + def move(self, origin, x=0, y=5, center=False): + if x: + self.x = origin.x + origin.width + x + else: + self.x = origin.x + if y: + self.y = origin.y + origin.height + y + else: + self.y = origin.y + + if center: + self.center() + return + + +class UnoLabel(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'label' + + @property + def value(self): + return self.model.Label + @value.setter + def value(self, value): + self.model.Label = value + + +class UnoLabelLink(UnoLabel): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'link' + + +class UnoButton(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'button' + + @property + def value(self): + return self.model.Label + @value.setter + def value(self, value): + self.model.Label = value + + +class UnoRadio(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'radio' + + @property + def value(self): + return self.model.Label + @value.setter + def value(self, value): + self.model.Label = value + + +class UnoCheckBox(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'checkbox' + + @property + def value(self): + return self.model.State + @value.setter + def value(self, value): + self.model.State = value + + @property + def label(self): + return self.model.Label + @label.setter + def label(self, value): + self.model.Label = value + + @property + def tri_state(self): + return self.model.TriState + @tri_state.setter + def tri_state(self, value): + self.model.TriState = value + + +# ~ https://api.libreoffice.org/docs/idl/ref/servicecom_1_1sun_1_1star_1_1awt_1_1UnoControlEditModel.html +class UnoText(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'text' + + @property + def value(self): + return self.model.Text + @value.setter + def value(self, value): + self.model.Text = value + + def validate(self): + return + + +class UnoImage(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def type(self): + return 'image' + + @property + def value(self): + return self.url + @value.setter + def value(self, value): + self.url = value + + @property + def url(self): + return self.m.ImageURL + @url.setter + def url(self, value): + self.m.ImageURL = None + self.m.ImageURL = _P.to_url(value) + + +class UnoListBox(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + self._path = '' + + def __setattr__(self, name, value): + if name in ('_path',): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + @property + def type(self): + return 'listbox' + + @property + def value(self): + return self.obj.getSelectedItem() + + @property + def count(self): + return len(self.data) + + @property + def data(self): + return self.model.StringItemList + @data.setter + def data(self, values): + self.model.StringItemList = list(sorted(values)) + + @property + def path(self): + return self._path + @path.setter + def path(self, value): + self._path = value + + def unselect(self): + self.obj.selectItem(self.value, False) + return + + def select(self, pos=0): + if isinstance(pos, str): + self.obj.selectItem(pos, True) + else: + self.obj.selectItemPos(pos, True) + return + + def clear(self): + self.model.removeAllItems() + return + + def _set_image_url(self, image): + if _P.exists(image): + return _P.to_url(image) + + path = _P.join(self._path, DIR['images'], image) + return _P.to_url(path) + + def insert(self, value, path='', pos=-1, show=True): + if pos < 0: + pos = self.count + if path: + self.model.insertItem(pos, value, self._set_image_url(path)) + else: + self.model.insertItemText(pos, value) + if show: + self.select(pos) + return + + +class UnoRoadmap(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + self._options = () + + def __setattr__(self, name, value): + if name in ('_options',): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + @property + def options(self): + return self._options + @options.setter + def options(self, values): + self._options = values + for i, v in enumerate(values): + opt = self.model.createInstance() + opt.ID = i + opt.Label = v + self.model.insertByIndex(i, opt) + return + + @property + def enabled(self): + return True + @enabled.setter + def enabled(self, value): + for m in self.model: + m.Enabled = value + return + + def set_enabled(self, index, value): + self.model.getByIndex(index).Enabled = value + return + + +class UnoTree(UnoBaseObject): + + def __init__(self, obj, ): + super().__init__(obj) + self._tdm = None + self._data = [] + + def __setattr__(self, name, value): + if name in ('_tdm', '_data'): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + @property + def selection(self): + sel = self.obj.Selection + return sel.DataValue, sel.DisplayValue + + @property + def parent(self): + parent = self.obj.Selection.Parent + if parent is None: + return () + return parent.DataValue, parent.DisplayValue + + def _get_parents(self, node): + value = (node.DisplayValue,) + parent = node.Parent + if parent is None: + return value + return self._get_parents(parent) + value + + @property + def parents(self): + values = self._get_parents(self.obj.Selection) + return values + + @property + def root(self): + if self._tdm is None: + return '' + return self._tdm.Root.DisplayValue + @root.setter + def root(self, value): + self._add_data_model(value) + + def _add_data_model(self, name): + tdm = create_instance('com.sun.star.awt.tree.MutableTreeDataModel') + root = tdm.createNode(name, True) + root.DataValue = 0 + tdm.setRoot(root) + self.model.DataModel = tdm + self._tdm = self.model.DataModel + return + + @property + def path(self): + return self.root + @path.setter + def path(self, value): + self.data = _P.walk_dir(value, True) + + @property + def data(self): + return self._data + @data.setter + def data(self, values): + self._data = list(values) + self._add_data() + + def _add_data(self): + if not self.data: + return + + parents = {} + for node in self.data: + parent = parents.get(node[1], self._tdm.Root) + child = self._tdm.createNode(node[2], False) + child.DataValue = node[0] + parent.appendChild(child) + parents[node[0]] = child + self.obj.expandNode(self._tdm.Root) + return + + +# ~ https://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1awt_1_1grid.html +class UnoGrid(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + self._gdm = self.model.GridDataModel + self._columns = [] + self._data = [] + # ~ self._format_columns = () + + def __setattr__(self, name, value): + if name in ('_gdm', '_columns', '_data'): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + def __getitem__(self, key): + value = self._gdm.getCellData(key[0], key[1]) + return value + + def __setitem__(self, key, value): + self._gdm.updateCellData(key[0], key[1], value) + return + + @property + def type(self): + return 'grid' + + @property + def columns(self): + return self._columns + @columns.setter + def columns(self, values): + self._columns = values + #~ https://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1awt_1_1grid_1_1XGridColumn.html + model = create_instance('com.sun.star.awt.grid.DefaultGridColumnModel', True) + for properties in values: + column = create_instance('com.sun.star.awt.grid.GridColumn', True) + for k, v in properties.items(): + setattr(column, k, v) + model.addColumn(column) + self.model.ColumnModel = model + return + + @property + def data(self): + return self._data + @data.setter + def data(self, values): + self._data = values + self.clear() + headings = tuple(range(1, len(values) + 1)) + self._gdm.addRows(headings, values) + # ~ rows = range(grid_dm.RowCount) + # ~ colors = [COLORS['GRAY'] if r % 2 else COLORS['WHITE'] for r in rows] + # ~ grid.Model.RowBackgroundColors = tuple(colors) + return + + @property + def value(self): + if self.column == -1 or self.row == -1: + return '' + return self[self.column, self.row] + @value.setter + def value(self, value): + if self.column > -1 and self.row > -1: + self[self.column, self.row] = value + + @property + def row(self): + return self.obj.CurrentRow + + @property + def column(self): + return self.obj.CurrentColumn + + def clear(self): + self._gdm.removeAllRows() + return + + # UP + def _format_cols(self): + rows = tuple(tuple( + self._format_columns[i].format(r) for i, r in enumerate(row)) for row in self._data + ) + return rows + + # ~ @property + # ~ def format_columns(self): + # ~ return self._format_columns + # ~ @format_columns.setter + # ~ def format_columns(self, value): + # ~ self._format_columns = value + + # ~ @property + # ~ def rows(self): + # ~ return self._gdm.RowCount + + # ~ @property + # ~ def columns(self): + # ~ return self._gdm.ColumnCount + + def set_cell_tooltip(self, col, row, value): + self._gdm.updateCellToolTip(col, row, value) + return + + def get_cell_tooltip(self, col, row): + value = self._gdm.getCellToolTip(col, row) + return value + + def _validate_column(self, data): + row = [] + for i, d in enumerate(data): + if i in self._columns: + if 'image' in self._columns[i]: + row.append(self._columns[i]['image']) + else: + row.append(d) + return tuple(row) + + def add_row(self, data): + # ~ self._data.append(data) + data = self._validate_column(data) + self._gdm.addRow(self.rows + 1, data) + return + + def remove_row(self, row): + self._gdm.removeRow(row) + # ~ del self._data[row] + self.update_row_heading() + return + + def update_row_heading(self): + for i in range(self.rows): + self._gdm.updateRowHeading(i, i + 1) + return + + def sort(self, column, asc=True): + self._gdm.sortByColumn(column, asc) + self.update_row_heading() + return + + def set_column_image(self, column, path): + gp = create_instance('com.sun.star.graphic.GraphicProvider') + data = dict_to_property({'URL': _path_url(path)}) + image = gp.queryGraphic(data) + if not column in self._columns: + self._columns[column] = {} + self._columns[column]['image'] = image + return + + +class UnoPage(object): + + def __init__(self, obj): + self._obj = obj + self._events = None + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + @property + def obj(self): + return self._obj + + @property + def model(self): + return self._obj.Model + + # ~ @property + # ~ def id(self): + # ~ return self.m.TabPageID + + @property + def parent(self): + return self.obj.Context + + def _set_image_url(self, image): + if _P.exists(image): + return _P.to_url(image) + + path = _P.join(self._path, DIR['images'], image) + return _P.to_url(path) + + def _special_properties(self, tipo, args): + if tipo == 'link' and not 'Label' in args: + args['Label'] = args['URL'] + return args + + if tipo == 'button': + if 'ImageURL' in args: + args['ImageURL'] = self._set_image_url(args['ImageURL']) + args['FocusOnClick'] = args.get('FocusOnClick', False) + return args + + if tipo == 'roadmap': + args['Height'] = args.get('Height', self.height) + if 'Title' in args: + args['Text'] = args.pop('Title') + return args + + if tipo == 'tree': + args['SelectionType'] = args.get('SelectionType', SINGLE) + return args + + if tipo == 'grid': + args['ShowRowHeader'] = args.get('ShowRowHeader', True) + return args + + if tipo == 'pages': + args['Width'] = args.get('Width', self.width) + args['Height'] = args.get('Height', self.height) + + return args + + def add_control(self, args): + tipo = args.pop('Type').lower() + root = args.pop('Root', '') + sheets = args.pop('Sheets', ()) + columns = args.pop('Columns', ()) + + args = self._special_properties(tipo, args) + model = self.model.createInstance(UNO_MODELS[tipo]) + _set_properties(model, args) + name = args['Name'] + self.model.insertByName(name, model) + control = self.obj.getControl(name) + _add_listeners(self._events, control, name) + control = UNO_CLASSES[tipo](control) + + if tipo in ('listbox',): + control.path = self.path + + if tipo == 'tree' and root: + control.root = root + elif tipo == 'grid' and columns: + control.columns = columns + elif tipo == 'pages' and sheets: + control.sheets = sheets + control.events = self.events + + setattr(self, name, control) + return control + + +class UnoPages(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + self._sheets = [] + self._events = None + + def __setattr__(self, name, value): + if name in ('_sheets', '_events'): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + def __getitem__(self, index): + name = index + if isinstance(index, int): + name = f'sheet{index}' + sheet = self.obj.getControl(name) + page = UnoPage(sheet) + page._events = self._events + return page + + @property + def type(self): + return 'pages' + + @property + def current(self): + return self.obj.ActiveTabID + @property + def active(self): + return self.current + + @property + def sheets(self): + return self._sheets + @sheets.setter + def sheets(self, values): + self._sheets = values + for i, title in enumerate(values): + sheet = self.m.createInstance('com.sun.star.awt.UnoPageModel') + sheet.Title = title + self.m.insertByName(f'sheet{i + 1}', sheet) + return + + @property + def events(self): + return self._events + @events.setter + def events(self, controllers): + self._events = controllers + + @property + def visible(self): + return self.obj.Visible + @visible.setter + def visible(self, value): + self.obj.Visible = value + + def insert(self, title): + self._sheets.append(title) + id = len(self._sheets) + sheet = self.m.createInstance('com.sun.star.awt.UnoPageModel') + sheet.Title = title + self.m.insertByName(f'sheet{id}', sheet) + return self[id] + + def remove(self, id): + self.obj.removeTab(id) + return + + def activate(self, id): + self.obj.activateTab(id) + return + + +UNO_CLASSES = { + 'label': UnoLabel, + 'link': UnoLabelLink, + 'button': UnoButton, + 'radio': UnoRadio, + 'checkbox': UnoCheckBox, + 'text': UnoText, + 'image': UnoImage, + 'listbox': UnoListBox, + 'roadmap': UnoRoadmap, + 'tree': UnoTree, + 'grid': UnoGrid, + 'pages': UnoPages, +} + +UNO_MODELS = { + 'label': 'com.sun.star.awt.UnoControlFixedTextModel', + 'link': 'com.sun.star.awt.UnoControlFixedHyperlinkModel', + 'button': 'com.sun.star.awt.UnoControlButtonModel', + 'radio': 'com.sun.star.awt.UnoControlRadioButtonModel', + 'checkbox': 'com.sun.star.awt.UnoControlCheckBoxModel', + 'text': 'com.sun.star.awt.UnoControlEditModel', + 'image': 'com.sun.star.awt.UnoControlImageControlModel', + 'listbox': 'com.sun.star.awt.UnoControlListBoxModel', + 'roadmap': 'com.sun.star.awt.UnoControlRoadmapModel', + 'tree': 'com.sun.star.awt.tree.TreeControlModel', + 'grid': 'com.sun.star.awt.grid.UnoControlGridModel', + 'pages': 'com.sun.star.awt.UnoMultiPageModel', + 'groupbox': 'com.sun.star.awt.UnoControlGroupBoxModel', + 'combobox': 'com.sun.star.awt.UnoControlComboBoxModel', +} +# ~ 'CurrencyField': 'com.sun.star.awt.UnoControlCurrencyFieldModel', +# ~ 'DateField': 'com.sun.star.awt.UnoControlDateFieldModel', +# ~ 'FileControl': 'com.sun.star.awt.UnoControlFileControlModel', +# ~ 'FormattedField': 'com.sun.star.awt.UnoControlFormattedFieldModel', +# ~ 'NumericField': 'com.sun.star.awt.UnoControlNumericFieldModel', +# ~ 'PatternField': 'com.sun.star.awt.UnoControlPatternFieldModel', +# ~ 'ProgressBar': 'com.sun.star.awt.UnoControlProgressBarModel', +# ~ 'ScrollBar': 'com.sun.star.awt.UnoControlScrollBarModel', +# ~ 'SimpleAnimation': 'com.sun.star.awt.UnoControlSimpleAnimationModel', +# ~ 'SpinButton': 'com.sun.star.awt.UnoControlSpinButtonModel', +# ~ 'Throbber': 'com.sun.star.awt.UnoControlThrobberModel', +# ~ 'TimeField': 'com.sun.star.awt.UnoControlTimeFieldModel', + + +class LODialog(object): + SEPARATION = 5 + MODELS = { + 'label': 'com.sun.star.awt.UnoControlFixedTextModel', + 'link': 'com.sun.star.awt.UnoControlFixedHyperlinkModel', + 'button': 'com.sun.star.awt.UnoControlButtonModel', + 'radio': 'com.sun.star.awt.UnoControlRadioButtonModel', + 'checkbox': 'com.sun.star.awt.UnoControlCheckBoxModel', + 'text': 'com.sun.star.awt.UnoControlEditModel', + 'image': 'com.sun.star.awt.UnoControlImageControlModel', + 'listbox': 'com.sun.star.awt.UnoControlListBoxModel', + 'roadmap': 'com.sun.star.awt.UnoControlRoadmapModel', + 'tree': 'com.sun.star.awt.tree.TreeControlModel', + 'grid': 'com.sun.star.awt.grid.UnoControlGridModel', + 'pages': 'com.sun.star.awt.UnoMultiPageModel', + 'groupbox': 'com.sun.star.awt.UnoControlGroupBoxModel', + 'combobox': 'com.sun.star.awt.UnoControlComboBoxModel', + } + + def __init__(self, args): + self._obj = self._create(args) + self._model = self.obj.Model + self._events = None + self._modal = True + self._controls = {} + self._color_on_focus = COLOR_ON_FOCUS + self._id = '' + self._path = '' + self._init_controls() + + def _create(self, args): + service = 'com.sun.star.awt.DialogProvider' + path = args.pop('Path', '') + if path: + dp = create_instance(service, True) + dlg = dp.createDialog(_P.to_url(path)) + return dlg + + if 'Location' in args: + name = args['Name'] + library = args.get('Library', 'Standard') + location = args.get('Location', 'application').lower() + if location == 'user': + location = 'application' + url = f'vnd.sun.star.script:{library}.{name}?location={location}' + if location == 'document': + dp = create_instance(service, args=docs.active.obj) + else: + dp = create_instance(service, True) + # ~ uid = docs.active.uid + # ~ url = f'vnd.sun.star.tdoc:/{uid}/Dialogs/{library}/{name}.xml' + dlg = dp.createDialog(url) + return dlg + + dlg = create_instance('com.sun.star.awt.UnoControlDialog', True) + model = create_instance('com.sun.star.awt.UnoControlDialogModel', True) + toolkit = create_instance('com.sun.star.awt.Toolkit', True) + _set_properties(model, args) + dlg.setModel(model) + dlg.setVisible(False) + dlg.createPeer(toolkit, None) + return dlg + + def _get_type_control(self, name): + name = name.split('.')[2] + types = { + 'UnoFixedTextControl': 'label', + 'UnoEditControl': 'text', + 'UnoButtonControl': 'button', + } + return types[name] + + def _init_controls(self): + for control in self.obj.getControls(): + tipo = self._get_type_control(control.ImplementationName) + name = control.Model.Name + control = UNO_CLASSES[tipo](control) + setattr(self, name, control) + return + + @property + def obj(self): + return self._obj + + @property + def model(self): + return self._model + + @property + def controls(self): + return self._controls + + @property + def path(self): + return self._path + @property + def id(self): + return self._id + @id.setter + def id(self, value): + self._id = value + self._path = _P.from_id(value) + + @property + def height(self): + return self.model.Height + @height.setter + def height(self, value): + self.model.Height = value + + @property + def width(self): + return self.model.Width + @width.setter + def width(self, value): + self.model.Width = value + + @property + def visible(self): + return self.obj.Visible + @visible.setter + def visible(self, value): + self.obj.Visible = value + + @property + def step(self): + return self.model.Step + @step.setter + def step(self, value): + self.model.Step = value + + @property + def events(self): + return self._events + @events.setter + def events(self, controllers): + self._events = controllers(self) + self._connect_listeners() + + @property + def color_on_focus(self): + return self._color_on_focus + @color_on_focus.setter + def color_on_focus(self, value): + self._color_on_focus = get_color(value) + + def _connect_listeners(self): + for control in self.obj.Controls: + _add_listeners(self.events, control, control.Model.Name) + return + + def _set_image_url(self, image): + if _P.exists(image): + return _P.to_url(image) + + path = _P.join(self._path, DIR['images'], image) + return _P.to_url(path) + + def _special_properties(self, tipo, args): + if tipo == 'link' and not 'Label' in args: + args['Label'] = args['URL'] + return args + + if tipo == 'button': + if 'ImageURL' in args: + args['ImageURL'] = self._set_image_url(args['ImageURL']) + args['FocusOnClick'] = args.get('FocusOnClick', False) + return args + + if tipo == 'roadmap': + args['Height'] = args.get('Height', self.height) + if 'Title' in args: + args['Text'] = args.pop('Title') + return args + + if tipo == 'tree': + args['SelectionType'] = args.get('SelectionType', SINGLE) + return args + + if tipo == 'grid': + args['ShowRowHeader'] = args.get('ShowRowHeader', True) + return args + + if tipo == 'pages': + args['Width'] = args.get('Width', self.width) + args['Height'] = args.get('Height', self.height) + + return args + + def add_control(self, args): + tipo = args.pop('Type').lower() + root = args.pop('Root', '') + sheets = args.pop('Sheets', ()) + columns = args.pop('Columns', ()) + + args = self._special_properties(tipo, args) + model = self.model.createInstance(self.MODELS[tipo]) + _set_properties(model, args) + name = args['Name'] + self.model.insertByName(name, model) + control = self.obj.getControl(name) + _add_listeners(self.events, control, name) + control = UNO_CLASSES[tipo](control) + + if tipo in ('listbox',): + control.path = self.path + + if tipo == 'tree' and root: + control.root = root + elif tipo == 'grid' and columns: + control.columns = columns + elif tipo == 'pages' and sheets: + control.sheets = sheets + control.events = self.events + + setattr(self, name, control) + self._controls[name] = control + return control + + def center(self, control, x=0, y=0): + w = self.width + h = self.height + + if isinstance(control, tuple): + wt = self.SEPARATION * -1 + for c in control: + wt += c.width + self.SEPARATION + x = w / 2 - wt / 2 + for c in control: + c.x = x + x = c.x + c.width + self.SEPARATION + return + + if x < 0: + x = w + x - control.width + elif x == 0: + x = w / 2 - control.width / 2 + if y < 0: + y = h + y - control.height + elif y == 0: + y = h / 2 - control.height / 2 + control.x = x + control.y = y + return + + def open(self, modal=True): + self._modal = modal + if modal: + return self.obj.execute() + else: + self.visible = True + return + + def close(self, value=0): + if self._modal: + value = self.obj.endDialog(value) + else: + self.visible = False + self.obj.dispose() + return value + + +class LOSheets(object): + + def __getitem__(self, index): + return LODocs().active[index] + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + +class LOCells(object): + + def __getitem__(self, index): + return LODocs().active.active[index] + + +class LOShortCut(object): +# ~ getKeyEventsByCommand + + def __init__(self, app): + self._app = app + self._scm = None + self._init_values() + + def _init_values(self): + name = 'com.sun.star.ui.GlobalAcceleratorConfiguration' + instance = 'com.sun.star.ui.ModuleUIConfigurationManagerSupplier' + service = TYPE_DOC[self._app] + manager = create_instance(instance, True) + uicm = manager.getUIConfigurationManager(service) + self._scm = uicm.ShortCutManager + return + + def __contains__(self, item): + cmd = self._get_command(item) + return bool(cmd) + + def _get_key_event(self, command): + events = self._scm.AllKeyEvents + for event in events: + cmd = self._scm.getCommandByKeyEvent(event) + if cmd == command: + break + return event + + def _to_key_event(self, shortcut): + key_event = KeyEvent() + keys = shortcut.split('+') + for v in keys[:-1]: + key_event.Modifiers += MODIFIERS[v.lower()] + key_event.KeyCode = getattr(Key, keys[-1].upper()) + return key_event + + def _get_command(self, shortcut): + command = '' + key_event = self._to_key_event(shortcut) + try: + command = self._scm.getCommandByKeyEvent(key_event) + except NoSuchElementException: + debug(f'No exists: {shortcut}') + return command + + def add(self, shortcut, command): + if isinstance(command, dict): + command = _get_url_script(command) + key_event = self._to_key_event(shortcut) + self._scm.setKeyEvent(key_event, command) + self._scm.store() + return + + def reset(self): + self._scm.reset() + self._scm.store() + return + + def remove(self, shortcut): + key_event = self._to_key_event(shortcut) + try: + self._scm.removeKeyEvent(key_event) + self._scm.store() + except NoSuchElementException: + debug(f'No exists: {shortcut}') + return + + def remove_by_command(self, command): + if isinstance(command, dict): + command = _get_url_script(command) + try: + self._scm.removeCommandFromAllKeyEvents(command) + self._scm.store() + except NoSuchElementException: + debug(f'No exists: {command}') + return + + +class LOShortCuts(object): + + def __getitem__(self, index): + return LOShortCut(index) + + +class LOMenu(object): + + def __init__(self, app): + self._app = app + self._ui = None + self._pymenus = None + self._menu = None + self._menus = self._get_menus() + + def __getitem__(self, index): + if isinstance(index, int): + self._menu = self._menus[index] + else: + for menu in self._menus: + cmd = menu.get('CommandURL', '') + if MENUS[index.lower()] == cmd: + self._menu = menu + break + line = self._menu.get('CommandURL', '') + line += self._get_submenus(self._menu['ItemDescriptorContainer']) + return line + + def _get_menus(self): + instance = 'com.sun.star.ui.ModuleUIConfigurationManagerSupplier' + service = TYPE_DOC[self._app] + manager = create_instance(instance, True) + self._ui = manager.getUIConfigurationManager(service) + self._pymenus = self._ui.getSettings(NODE_MENUBAR, True) + data = [] + for menu in self._pymenus: + data.append(data_to_dict(menu)) + return data + + def _get_info(self, menu): + line = menu.get('CommandURL', '') + line += self._get_submenus(menu['ItemDescriptorContainer']) + return line + + def _get_submenus(self, menu, level=1): + line = '' + for i, v in enumerate(menu): + data = data_to_dict(v) + cmd = data.get('CommandURL', '----------') + line += f'\n{" " * level}├─ ({i}) {cmd}' + submenu = data.get('ItemDescriptorContainer', None) + if not submenu is None: + line += self._get_submenus(submenu, level + 1) + return line + + def __str__(self): + info = '\n'.join([self._get_info(m) for m in self._menus]) + return info + + def _get_index_menu(self, menu, command): + index = -1 + for i, v in enumerate(menu): + data = data_to_dict(v) + cmd = data.get('CommandURL', '') + if cmd == command: + index = i + break + return index + + def insert(self, name, args): + idc = None + replace = False + command = args['CommandURL'] + label = args['Label'] + + self[name] + menu = self._menu['ItemDescriptorContainer'] + submenu = args.get('Submenu', False) + if submenu: + idc = self._ui.createSettings() + + index = self._get_index_menu(menu, command) + if index == -1: + if 'Index' in args: + index = args['Index'] + else: + index = self._get_index_menu(menu, args['After']) + 1 + else: + replace = True + + data = dict ( + CommandURL = command, + Label = label, + Style = 0, + Type = 0, + ItemDescriptorContainer = idc, + ) + self._save(menu, data, index, replace) + self._insert_submenu(idc, submenu) + return + + def _get_command(self, args): + shortcut = args.get('ShortCut', '') + cmd = args['CommandURL'] + if isinstance(cmd, dict): + cmd = _get_url_script(cmd) + if shortcut: + LOShortCut(self._app).add(shortcut, cmd) + return cmd + + def _insert_submenu(self, parent, menus): + for i, v in enumerate(menus): + submenu = v.pop('Submenu', False) + if submenu: + idc = self._ui.createSettings() + v['ItemDescriptorContainer'] = idc + v['Type'] = 0 + if v['Label'] == '-': + v['Type'] = 1 + else: + v['CommandURL'] = self._get_command(v) + self._save(parent, v, i) + if submenu: + self._insert_submenu(idc, submenu) + return + + def remove(self, name, command): + self[name] + menu = self._menu['ItemDescriptorContainer'] + index = self._get_index_menu(menu, command) + if index > -1: + uno.invoke(menu, 'removeByIndex', (index,)) + self._ui.replaceSettings(NODE_MENUBAR, self._pymenus) + self._ui.store() + return + + def _save(self, menu, properties, index, replace=False): + properties = dict_to_property(properties, True) + if replace: + uno.invoke(menu, 'replaceByIndex', (index, properties)) + else: + uno.invoke(menu, 'insertByIndex', (index, properties)) + self._ui.replaceSettings(NODE_MENUBAR, self._pymenus) + self._ui.store() + return + + +class LOMenus(object): + + def __getitem__(self, index): + return LOMenu(index) + + +class LOWindow(object): + EMPTY = """ + +""" + MODELS = { + 'label': 'com.sun.star.awt.UnoControlFixedTextModel', + 'link': 'com.sun.star.awt.UnoControlFixedHyperlinkModel', + 'button': 'com.sun.star.awt.UnoControlButtonModel', + 'radio': 'com.sun.star.awt.UnoControlRadioButtonModel', + 'checkbox': 'com.sun.star.awt.UnoControlCheckBoxModel', + 'text': 'com.sun.star.awt.UnoControlEditModel', + 'image': 'com.sun.star.awt.UnoControlImageControlModel', + 'listbox': 'com.sun.star.awt.UnoControlListBoxModel', + 'roadmap': 'com.sun.star.awt.UnoControlRoadmapModel', + 'tree': 'com.sun.star.awt.tree.TreeControlModel', + 'grid': 'com.sun.star.awt.grid.UnoControlGridModel', + 'pages': 'com.sun.star.awt.UnoMultiPageModel', + 'groupbox': 'com.sun.star.awt.UnoControlGroupBoxModel', + 'combobox': 'com.sun.star.awt.UnoControlComboBoxModel', + } + + def __init__(self, args): + self._events = None + self._menu = None + self._container = None + self._model = None + self._id = '' + self._path = '' + self._obj = self._create(args) + + def _create(self, properties): + ps = ( + properties.get('X', 0), + properties.get('Y', 0), + properties.get('Width', 500), + properties.get('Height', 500), + ) + self._title = properties.get('Title', TITLE) + self._create_frame(ps) + self._create_container(ps) + self._create_subcontainer(ps) + # ~ self._create_splitter(ps) + return + + def _create_frame(self, ps): + service = 'com.sun.star.frame.TaskCreator' + tc = create_instance(service, True) + self._frame = tc.createInstanceWithArguments(( + NamedValue('FrameName', 'EasyMacroWin'), + NamedValue('PosSize', Rectangle(*ps)), + )) + self._window = self._frame.getContainerWindow() + self._toolkit = self._window.getToolkit() + desktop = get_desktop() + self._frame.setCreator(desktop) + desktop.getFrames().append(self._frame) + self._frame.Title = self._title + return + + def _create_container(self, ps): + service = 'com.sun.star.awt.UnoControlContainer' + self._container = create_instance(service, True) + service = 'com.sun.star.awt.UnoControlContainerModel' + model = create_instance(service, True) + model.BackgroundColor = get_color((225, 225, 225)) + self._container.setModel(model) + self._container.createPeer(self._toolkit, self._window) + self._container.setPosSize(*ps, POSSIZE) + self._frame.setComponent(self._container, None) + return + + def _create_subcontainer(self, ps): + service = 'com.sun.star.awt.ContainerWindowProvider' + cwp = create_instance(service, True) + + path_tmp = _P.save_tmp(self.EMPTY) + subcont = cwp.createContainerWindow( + _P.to_url(path_tmp), '', self._container.getPeer(), None) + _P.kill(path_tmp) + + subcont.setPosSize(0, 0, 500, 500, POSSIZE) + subcont.setVisible(True) + self._container.addControl('subcont', subcont) + self._subcont = subcont + self._model = subcont.Model + return + + def _create_popupmenu(self, menus): + menu = create_instance('com.sun.star.awt.PopupMenu', True) + for i, m in enumerate(menus): + label = m['label'] + cmd = m.get('event', '') + if not cmd: + cmd = label.lower().replace(' ', '_') + if label == '-': + menu.insertSeparator(i) + else: + menu.insertItem(i, label, m.get('style', 0), i) + menu.setCommand(i, cmd) + # ~ menu.setItemImage(i, path?, True) + menu.addMenuListener(EventsMenu(self.events)) + return menu + + def _create_menu(self, menus): + #~ https://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1awt_1_1XMenu.html + #~ nItemId specifies the ID of the menu item to be inserted. + #~ aText specifies the label of the menu item. + #~ nItemStyle 0 = Standard, CHECKABLE = 1, RADIOCHECK = 2, AUTOCHECK = 4 + #~ nItemPos specifies the position where the menu item will be inserted. + self._menu = create_instance('com.sun.star.awt.MenuBar', True) + for i, m in enumerate(menus): + self._menu.insertItem(i, m['label'], m.get('style', 0), i) + cmd = m['label'].lower().replace(' ', '_') + self._menu.setCommand(i, cmd) + submenu = self._create_popupmenu(m['submenu']) + self._menu.setPopupMenu(i, submenu) + + self._window.setMenuBar(self._menu) + return + + def _add_listeners(self, control=None): + if self.events is None: + return + controller = EventsWindow(self) + self._window.addTopWindowListener(controller) + self._window.addWindowListener(controller) + # ~ self._container.addKeyListener(EventsKeyWindow(self)) + return + + def _set_image_url(self, image): + if _P.exists(image): + return _P.to_url(image) + + path = _P.join(self._path, DIR['images'], image) + return _P.to_url(path) + + def _special_properties(self, tipo, args): + if tipo == 'link' and not 'Label' in args: + args['Label'] = args['URL'] + return args + + if tipo == 'button': + if 'ImageURL' in args: + args['ImageURL'] = self._set_image_url(args['ImageURL']) + args['FocusOnClick'] = args.get('FocusOnClick', False) + return args + + if tipo == 'roadmap': + args['Height'] = args.get('Height', self.height) + if 'Title' in args: + args['Text'] = args.pop('Title') + return args + + if tipo == 'tree': + args['SelectionType'] = args.get('SelectionType', SINGLE) + return args + + if tipo == 'grid': + args['ShowRowHeader'] = args.get('ShowRowHeader', True) + return args + + if tipo == 'pages': + args['Width'] = args.get('Width', self.width) + args['Height'] = args.get('Height', self.height) + + return args + + def add_control(self, args): + tipo = args.pop('Type').lower() + root = args.pop('Root', '') + sheets = args.pop('Sheets', ()) + columns = args.pop('Columns', ()) + + args = self._special_properties(tipo, args) + model = self.model.createInstance(self.MODELS[tipo]) + _set_properties(model, args) + name = args['Name'] + self.model.insertByName(name, model) + control = self._subcont.getControl(name) + _add_listeners(self.events, control, name) + control = UNO_CLASSES[tipo](control) + + # ~ if tipo in ('listbox',): + # ~ control.path = self.path + + if tipo == 'tree' and root: + control.root = root + elif tipo == 'grid' and columns: + control.columns = columns + elif tipo == 'pages' and sheets: + control.sheets = sheets + control.events = self.events + + setattr(self, name, control) + return control + + @property + def events(self): + return self._events + @events.setter + def events(self, controllers): + self._events = controllers(self) + self._add_listeners() + + @property + def model(self): + return self._model + + @property + def width(self): + return self._container.Size.Width + + @property + def height(self): + return self._container.Size.Height + + @property + def name(self): + return self._title.lower().replace(' ', '_') + + def add_menu(self, menus): + self._create_menu(menus) + return + + def open(self): + self._window.setVisible(True) + return + + def close(self): + self._window.setMenuBar(None) + self._window.dispose() + self._frame.close(True) + return + + +def create_window(args): + return LOWindow(args) + + +class classproperty: + def __init__(self, method=None): + self.fget = method + + def __get__(self, instance, cls=None): + return self.fget(cls) + + def getter(self, method): + self.fget = method + return self + + +class ClipBoard(object): + SERVICE = 'com.sun.star.datatransfer.clipboard.SystemClipboard' + CLIPBOARD_FORMAT_TEXT = 'text/plain;charset=utf-16' + + class TextTransferable(unohelper.Base, XTransferable): + + def __init__(self, text): + df = DataFlavor() + df.MimeType = ClipBoard.CLIPBOARD_FORMAT_TEXT + df.HumanPresentableName = "encoded text utf-16" + self.flavors = (df,) + self._data = text + + def getTransferData(self, flavor): + return self._data + + def getTransferDataFlavors(self): + return self.flavors + + + @classmethod + def set(cls, value): + ts = cls.TextTransferable(value) + sc = create_instance(cls.SERVICE) + sc.setContents(ts, None) + return + + @classproperty + def contents(cls): + df = None + text = '' + sc = create_instance(cls.SERVICE) + transferable = sc.getContents() + data = transferable.getTransferDataFlavors() + for df in data: + if df.MimeType == cls.CLIPBOARD_FORMAT_TEXT: + break + if df: + text = transferable.getTransferData(df) + return text +_CB = ClipBoard + + +class Paths(object): + FILE_PICKER = 'com.sun.star.ui.dialogs.FilePicker' + + def __init__(self, path=''): + if path.startswith('file://'): + path = str(Path(uno.fileUrlToSystemPath(path)).resolve()) + self._path = Path(path) + + @property + def path(self): + return str(self._path.parent) + + @property + def file_name(self): + return self._path.name + + @property + def name(self): + return self._path.stem + + @property + def ext(self): + return self._path.suffix[1:] + + @property + def info(self): + return self.path, self.file_name, self.name, self.ext + + @property + def url(self): + return self._path.as_uri() + + @property + def size(self): + return self._path.stat().st_size + + @classproperty + def home(self): + return str(Path.home()) + + @classproperty + def documents(self): + return self.config() + + @classproperty + def temp_dir(self): + return tempfile.gettempdir() + + @classproperty + def python(self): + if IS_WIN: + path = self.join(self.config('Module'), PYTHON) + elif IS_MAC: + path = self.join(self.config('Module'), '..', 'Resources', PYTHON) + else: + path = sys.executable + return path + + @classmethod + def dir_tmp(self, only_name=False): + dt = tempfile.TemporaryDirectory() + if only_name: + dt = dt.name + return dt + + @classmethod + def tmp(cls, ext=''): + tmp = tempfile.NamedTemporaryFile(suffix=ext) + return tmp.name + + @classmethod + def save_tmp(cls, data): + path_tmp = cls.tmp() + cls.save(path_tmp, data) + return path_tmp + + @classmethod + def config(cls, name='Work'): + """ + Return de path name in config + http://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1util_1_1XPathSettings.html + """ + path = create_instance('com.sun.star.util.PathSettings') + return cls.to_system(getattr(path, name)) + + @classmethod + def get(cls, init_dir='', filters: str=''): + """ + Options: http://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1ui_1_1dialogs_1_1TemplateDescription.html + filters: 'xml' or 'txt,xml' + """ + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + file_picker = create_instance(cls.FILE_PICKER) + file_picker.setTitle(_('Select path')) + file_picker.setDisplayDirectory(init_dir) + file_picker.initialize((2,)) + if filters: + filters = [(f.upper(), f'*.{f.lower()}') for f in filters.split(',')] + file_picker.setCurrentFilter(filters[0][0]) + for f in filters: + file_picker.appendFilter(f[0], f[1]) + + path = '' + if file_picker.execute(): + path = cls.to_system(file_picker.getSelectedFiles()[0]) + return path + + @classmethod + def get_dir(cls, init_dir=''): + folder_picker = create_instance(cls.FILE_PICKER) + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + folder_picker.setTitle(_('Select directory')) + folder_picker.setDisplayDirectory(init_dir) + + path = '' + if folder_picker.execute(): + path = cls.to_system(folder_picker.getDisplayDirectory()) + return path + + @classmethod + def get_file(cls, init_dir: str='', filters: str='', multiple: bool=False): + """ + init_folder: folder default open + multiple: True for multiple selected + filters: 'xml' or 'xml,txt' + """ + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + + file_picker = create_instance(cls.FILE_PICKER) + file_picker.setTitle(_('Select file')) + file_picker.setDisplayDirectory(init_dir) + file_picker.setMultiSelectionMode(multiple) + + if filters: + filters = [(f.upper(), f'*.{f.lower()}') for f in filters.split(',')] + file_picker.setCurrentFilter(filters[0][0]) + for f in filters: + file_picker.appendFilter(f[0], f[1]) + + path = '' + if file_picker.execute(): + files = file_picker.getSelectedFiles() + path = [cls.to_system(f) for f in files] + if not multiple: + path = path[0] + return path + + @classmethod + def replace_ext(cls, path, new_ext): + p = Paths(path) + name = f'{p.name}.{new_ext}' + path = cls.join(p.path, name) + return path + + @classmethod + def exists(cls, path): + result = False + if path: + path = cls.to_system(path) + result = Path(path).exists() + return result + + @classmethod + def exists_app(cls, name_app): + return bool(shutil.which(name_app)) + + @classmethod + def open(cls, path): + if IS_WIN: + os.startfile(path) + else: + pid = subprocess.Popen(['xdg-open', path]).pid + return + + @classmethod + def is_dir(cls, path): + return Path(path).is_dir() + + @classmethod + def is_file(cls, path): + return Path(path).is_file() + + @classmethod + def join(cls, *paths): + return str(Path(paths[0]).joinpath(*paths[1:])) + + @classmethod + def save(cls, path, data, encoding='utf-8'): + result = bool(Path(path).write_text(data, encoding=encoding)) + return result + + @classmethod + def save_bin(cls, path, data): + result = bool(Path(path).write_bytes(data)) + return result + + @classmethod + def read(cls, path, encoding='utf-8'): + data = Path(path).read_text(encoding=encoding) + return data + + @classmethod + def read_bin(cls, path): + data = Path(path).read_bytes() + return data + + @classmethod + def to_url(cls, path): + if not path.startswith('file://'): + path = Path(path).as_uri() + return path + + @classmethod + def to_system(cls, path): + if path.startswith('file://'): + path = str(Path(uno.fileUrlToSystemPath(path)).resolve()) + return path + + @classmethod + def kill(cls, path): + result = True + p = Path(path) + + try: + if p.is_file(): + p.unlink() + elif p.is_dir(): + shutil.rmtree(path) + except OSError as e: + log.error(e) + result = False + + return result + + @classmethod + def files(cls, path, pattern='*'): + files = [str(p) for p in Path(path).glob(pattern) if p.is_file()] + return files + + @classmethod + def dirs(cls, path): + dirs = [str(p) for p in Path(path).iterdir() if p.is_dir()] + return dirs + + @classmethod + def walk(cls, path, filters=''): + paths = [] + if filters in ('*', '*.*'): + filters = '' + for folder, _, files in os.walk(path): + if filters: + pattern = re.compile(r'\.(?:{})$'.format(filters), re.IGNORECASE) + paths += [cls.join(folder, f) for f in files if pattern.search(f)] + else: + paths += [cls.join(folder, f) for f in files] + return paths + + @classmethod + def walk_dir(cls, path, tree=False): + folders = [] + if tree: + i = 0 + p = 0 + parents = {path: 0} + for root, dirs, _ in os.walk(path): + for name in dirs: + i += 1 + rn = cls.join(root, name) + if not rn in parents: + parents[rn] = i + folders.append((i, parents[root], name)) + else: + for root, dirs, _ in os.walk(path): + folders += [cls.join(root, name) for name in dirs] + return folders + + @classmethod + def from_id(cls, id_ext): + pip = CTX.getValueByName('/singletons/com.sun.star.deployment.PackageInformationProvider') + path = _P.to_system(pip.getPackageLocation(id_ext)) + return path + + @classmethod + def from_json(cls, path): + data = json.loads(cls.read(path)) + return data + + @classmethod + def to_json(cls, path, data): + data = json.dumps(data, indent=4, ensure_ascii=False, sort_keys=True) + return cls.save(path, data) + + @classmethod + def from_csv(cls, path, args={}): + # ~ See https://docs.python.org/3.7/library/csv.html#csv.reader + with open(path) as f: + rows = tuple(csv.reader(f, **args)) + return rows + + @classmethod + def to_csv(cls, path, data, args={}): + with open(path, 'w') as f: + writer = csv.writer(f, **args) + writer.writerows(data) + return + + @classmethod + def zip(cls, source, target='', pwd=''): + path_zip = target + if not isinstance(source, (tuple, list)): + path, _, name, _ = _P(source).info + start = len(path) + 1 + if not target: + path_zip = f'{path}/{name}.zip' + + if isinstance(source, (tuple, list)): + files = [(f, f[len(_P(f).path)+1:]) for f in source] + elif _P.is_file(source): + files = ((source, source[start:]),) + else: + files = [(f, f[start:]) for f in _P.walk(source)] + + compression = zipfile.ZIP_DEFLATED + with zipfile.ZipFile(path_zip, 'w', compression=compression) as z: + for f in files: + z.write(f[0], f[1]) + return + + @classmethod + def zip_content(cls, path): + with zipfile.ZipFile(path) as z: + names = z.namelist() + return names + + @classmethod + def unzip(cls, source, target='', members=None, pwd=None): + path = target + if not target: + path = _P(source).path + with zipfile.ZipFile(source) as z: + if not pwd is None: + pwd = pwd.encode() + if isinstance(members, str): + members = (members,) + z.extractall(path, members=members, pwd=pwd) + return True + + @classmethod + def merge_zip(cls, target, zips): + try: + with zipfile.ZipFile(target, 'w', compression=zipfile.ZIP_DEFLATED) as t: + for path in zips: + with zipfile.ZipFile(path, compression=zipfile.ZIP_DEFLATED) as s: + for name in s.namelist(): + t.writestr(name, s.open(name).read()) + except Exception as e: + error(e) + return False + + return True + + @classmethod + def copy(cls, source, target='', name=''): + p, f, n, e = _P(source).info + if target: + p = target + if name: + e = '' + n = name + path_new = cls.join(p, f'{n}{e}') + shutil.copy(source, path_new) + return path_new +_P = Paths + + +def __getattr__(name): + if name == 'active': + return LODocs().active + if name == 'active_sheet': + return LODocs().active.active + if name == 'selection': + return LODocs().active.selection + if name == 'current_region': + return LODocs().active.selection.current_region + if name in ('rectangle', 'pos_size'): + return Rectangle() + if name == 'paths': + return Paths + if name == 'docs': + return LODocs() + if name == 'sheets': + return LOSheets() + if name == 'cells': + return LOCells() + if name == 'menus': + return LOMenus() + if name == 'shortcuts': + return LOShortCuts() + if name == 'clipboard': + return ClipBoard + raise AttributeError(f"module '{__name__}' has no attribute '{name}'") + + +def create_dialog(args): + return LODialog(args) + + +def inputbox(message, default='', title=TITLE, echochar=''): + + class ControllersInput(object): + + def __init__(self, dlg): + self.d = dlg + + def cmd_ok_action(self, event): + self.d.close(1) + return + + args = { + 'Title': title, + 'Width': 200, + 'Height': 80, + } + dlg = LODialog(args) + dlg.events = ControllersInput + + args = { + 'Type': 'Label', + 'Name': 'lbl_msg', + 'Label': message, + 'Width': 140, + 'Height': 50, + 'X': 5, + 'Y': 5, + 'MultiLine': True, + 'Border': 1, + } + dlg.add_control(args) + + args = { + 'Type': 'Text', + 'Name': 'txt_value', + 'Text': default, + 'Width': 190, + 'Height': 15, + } + if echochar: + args['EchoChar'] = ord(echochar[0]) + dlg.add_control(args) + dlg.txt_value.move(dlg.lbl_msg) + + args = { + 'Type': 'button', + 'Name': 'cmd_ok', + 'Label': _('OK'), + 'Width': 40, + 'Height': 15, + 'DefaultButton': True, + 'PushButtonType': 1, + } + dlg.add_control(args) + dlg.cmd_ok.move(dlg.lbl_msg, 10, 0) + + args = { + 'Type': 'button', + 'Name': 'cmd_cancel', + 'Label': _('Cancel'), + 'Width': 40, + 'Height': 15, + 'PushButtonType': 2, + } + dlg.add_control(args) + dlg.cmd_cancel.move(dlg.cmd_ok) + + if dlg.open(): + return dlg.txt_value.value + + return '' + + +def get_fonts(): + toolkit = create_instance('com.sun.star.awt.Toolkit') + device = toolkit.createScreenCompatibleDevice(0, 0) + return device.FontDescriptors + + +# ~ From request +# ~ https://github.com/psf/requests/blob/master/requests/structures.py#L15 +class CaseInsensitiveDict(MutableMapping): + + def __init__(self, data=None, **kwargs): + self._store = OrderedDict() + if data is None: + data = {} + self.update(data, **kwargs) + + def __setitem__(self, key, value): + # Use the lowercased key for lookups, but store the actual + # key alongside the value. + self._store[key.lower()] = (key, value) + + def __getitem__(self, key): + return self._store[key.lower()][1] + + def __delitem__(self, key): + del self._store[key.lower()] + + def __iter__(self): + return (casedkey for casedkey, mappedvalue in self._store.values()) + + def __len__(self): + return len(self._store) + + def lower_items(self): + """Like iteritems(), but with all lowercase keys.""" + values = ( + (lowerkey, keyval[1]) for (lowerkey, keyval) in self._store.items() + ) + return values + + # Copy is required + def copy(self): + return CaseInsensitiveDict(self._store.values()) + + def __repr__(self): + return str(dict(self.items())) + + +# ~ https://en.wikipedia.org/wiki/Web_colors +def get_color(value): + COLORS = { + 'aliceblue': 15792383, + 'antiquewhite': 16444375, + 'aqua': 65535, + 'aquamarine': 8388564, + 'azure': 15794175, + 'beige': 16119260, + 'bisque': 16770244, + 'black': 0, + 'blanchedalmond': 16772045, + 'blue': 255, + 'blueviolet': 9055202, + 'brown': 10824234, + 'burlywood': 14596231, + 'cadetblue': 6266528, + 'chartreuse': 8388352, + 'chocolate': 13789470, + 'coral': 16744272, + 'cornflowerblue': 6591981, + 'cornsilk': 16775388, + 'crimson': 14423100, + 'cyan': 65535, + 'darkblue': 139, + 'darkcyan': 35723, + 'darkgoldenrod': 12092939, + 'darkgray': 11119017, + 'darkgreen': 25600, + 'darkgrey': 11119017, + 'darkkhaki': 12433259, + 'darkmagenta': 9109643, + 'darkolivegreen': 5597999, + 'darkorange': 16747520, + 'darkorchid': 10040012, + 'darkred': 9109504, + 'darksalmon': 15308410, + 'darkseagreen': 9419919, + 'darkslateblue': 4734347, + 'darkslategray': 3100495, + 'darkslategrey': 3100495, + 'darkturquoise': 52945, + 'darkviolet': 9699539, + 'deeppink': 16716947, + 'deepskyblue': 49151, + 'dimgray': 6908265, + 'dimgrey': 6908265, + 'dodgerblue': 2003199, + 'firebrick': 11674146, + 'floralwhite': 16775920, + 'forestgreen': 2263842, + 'fuchsia': 16711935, + 'gainsboro': 14474460, + 'ghostwhite': 16316671, + 'gold': 16766720, + 'goldenrod': 14329120, + 'gray': 8421504, + 'grey': 8421504, + 'green': 32768, + 'greenyellow': 11403055, + 'honeydew': 15794160, + 'hotpink': 16738740, + 'indianred': 13458524, + 'indigo': 4915330, + 'ivory': 16777200, + 'khaki': 15787660, + 'lavender': 15132410, + 'lavenderblush': 16773365, + 'lawngreen': 8190976, + 'lemonchiffon': 16775885, + 'lightblue': 11393254, + 'lightcoral': 15761536, + 'lightcyan': 14745599, + 'lightgoldenrodyellow': 16448210, + 'lightgray': 13882323, + 'lightgreen': 9498256, + 'lightgrey': 13882323, + 'lightpink': 16758465, + 'lightsalmon': 16752762, + 'lightseagreen': 2142890, + 'lightskyblue': 8900346, + 'lightslategray': 7833753, + 'lightslategrey': 7833753, + 'lightsteelblue': 11584734, + 'lightyellow': 16777184, + 'lime': 65280, + 'limegreen': 3329330, + 'linen': 16445670, + 'magenta': 16711935, + 'maroon': 8388608, + 'mediumaquamarine': 6737322, + 'mediumblue': 205, + 'mediumorchid': 12211667, + 'mediumpurple': 9662683, + 'mediumseagreen': 3978097, + 'mediumslateblue': 8087790, + 'mediumspringgreen': 64154, + 'mediumturquoise': 4772300, + 'mediumvioletred': 13047173, + 'midnightblue': 1644912, + 'mintcream': 16121850, + 'mistyrose': 16770273, + 'moccasin': 16770229, + 'navajowhite': 16768685, + 'navy': 128, + 'oldlace': 16643558, + 'olive': 8421376, + 'olivedrab': 7048739, + 'orange': 16753920, + 'orangered': 16729344, + 'orchid': 14315734, + 'palegoldenrod': 15657130, + 'palegreen': 10025880, + 'paleturquoise': 11529966, + 'palevioletred': 14381203, + 'papayawhip': 16773077, + 'peachpuff': 16767673, + 'peru': 13468991, + 'pink': 16761035, + 'plum': 14524637, + 'powderblue': 11591910, + 'purple': 8388736, + 'red': 16711680, + 'rosybrown': 12357519, + 'royalblue': 4286945, + 'saddlebrown': 9127187, + 'salmon': 16416882, + 'sandybrown': 16032864, + 'seagreen': 3050327, + 'seashell': 16774638, + 'sienna': 10506797, + 'silver': 12632256, + 'skyblue': 8900331, + 'slateblue': 6970061, + 'slategray': 7372944, + 'slategrey': 7372944, + 'snow': 16775930, + 'springgreen': 65407, + 'steelblue': 4620980, + 'tan': 13808780, + 'teal': 32896, + 'thistle': 14204888, + 'tomato': 16737095, + 'turquoise': 4251856, + 'violet': 15631086, + 'wheat': 16113331, + 'white': 16777215, + 'whitesmoke': 16119285, + 'yellow': 16776960, + 'yellowgreen': 10145074, + } + + if isinstance(value, tuple): + color = (value[0] << 16) + (value[1] << 8) + value[2] + else: + if value[0] == '#': + r, g, b = bytes.fromhex(value[1:]) + color = (r << 16) + (g << 8) + b + else: + color = COLORS.get(value.lower(), -1) + return color + + +COLOR_ON_FOCUS = get_color('LightYellow') + + +class LOServer(object): + HOST = 'localhost' + PORT = '8100' + ARG = f'socket,host={HOST},port={PORT};urp;StarOffice.ComponentContext' + CMD = ['soffice', + '-env:SingleAppInstance=false', + '-env:UserInstallation=file:///tmp/LO_Process8100', + '--headless', '--norestore', '--invisible', + f'--accept={ARG}'] + + def __init__(self): + self._server = None + self._ctx = None + self._sm = None + self._start_server() + self._init_values() + + def _init_values(self): + global CTX + global SM + + if not self.is_running: + return + + ctx = uno.getComponentContext() + service = 'com.sun.star.bridge.UnoUrlResolver' + resolver = ctx.ServiceManager.createInstanceWithContext(service, ctx) + self._ctx = resolver.resolve('uno:{}'.format(self.ARG)) + self._sm = self._ctx.getServiceManager() + CTX = self._ctx + SM = self._sm + return + + @property + def is_running(self): + try: + s = socket.create_connection((self.HOST, self.PORT), 5.0) + s.close() + debug('LibreOffice is running...') + return True + except ConnectionRefusedError: + return False + + def _start_server(self): + if self.is_running: + return + + for i in range(3): + self._server = subprocess.Popen(self.CMD, + stdout=subprocess.PIPE, stderr=subprocess.PIPE) + time.sleep(3) + if self.is_running: + break + return + + def stop(self): + if self._server is None: + print('Search pgrep soffice') + else: + self._server.terminate() + debug('LibreOffice is stop...') + return + + def create_instance(self, name, with_context=True): + if with_context: + instance = self._sm.createInstanceWithContext(name, self._ctx) + else: + instance = self._sm.createInstance(name) + return instance diff --git a/easymacro2.py b/easymacro2.py deleted file mode 120000 index d2de111..0000000 --- a/easymacro2.py +++ /dev/null @@ -1 +0,0 @@ -/home/mau/Projects/libre_office/zaz/source/easymacro2.py \ No newline at end of file diff --git a/files/ZAZLaTex2SVG_v0.1.0.oxt b/files/ZAZLaTex2SVG_v0.1.0.oxt index caf2286ed379fb0311ae891fb8dc7eaf70f01cb5..5f91c1df9159bb8a1d5cb5c9be7cc8be226ebf07 100644 GIT binary patch delta 29917 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

E-lEoq<a#OTTeJ7zt|-cJB4FZqOMJK3EH)-T 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 L`0DrZpW6Qcl({X_ delta 24004 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 CBTb+H diff --git a/source/ZAZLaTex2SVG.py b/source/ZAZLaTex2SVG.py index a787c96..4072fe5 100644 --- a/source/ZAZLaTex2SVG.py +++ b/source/ZAZLaTex2SVG.py @@ -1,7 +1,7 @@ import uno import unohelper from com.sun.star.task import XJobExecutor -import easymacro2 as app +import easymacro as app ID_EXTENSION = 'net.elmau.zaz.latex2svg' diff --git a/source/pythonpath/easymacro2.py b/source/pythonpath/easymacro.py similarity index 76% rename from source/pythonpath/easymacro2.py rename to source/pythonpath/easymacro.py index d11c7e2..30f0e40 100644 --- a/source/pythonpath/easymacro2.py +++ b/source/pythonpath/easymacro.py @@ -4,7 +4,7 @@ # ~ This file is part of ZAZ. -# ~ https://gitlab.com/mauriciobaeza/zaz +# ~ https://git.elmau.net/elmau/zaz # ~ ZAZ is free software: you can redistribute it and/or modify # ~ it under the terms of the GNU General Public License as published by @@ -19,11 +19,13 @@ # ~ You should have received a copy of the GNU General Public License # ~ along with ZAZ. If not, see . - +import base64 +import csv import datetime import getpass import gettext import hashlib +import json import logging import os import platform @@ -31,11 +33,14 @@ import re import shlex import shutil import socket +import ssl import subprocess import sys import tempfile import threading import time +import traceback +import zipfile from collections import OrderedDict from collections.abc import MutableMapping @@ -44,15 +49,29 @@ from enum import IntEnum from functools import wraps from pathlib import Path from pprint import pprint +from string import Template from typing import Any +from urllib.request import Request, urlopen +from urllib.error import URLError, HTTPError + +import smtplib +from smtplib import SMTPException, SMTPAuthenticationError +from email.mime.multipart import MIMEMultipart +from email.mime.base import MIMEBase +from email.mime.text import MIMEText +from email.utils import formatdate +from email import encoders +import mailbox import uno import unohelper from com.sun.star.awt import MessageBoxButtons as MSG_BUTTONS from com.sun.star.awt.MessageBoxResults import YES from com.sun.star.awt import Rectangle, Size, Point +from com.sun.star.awt.PosSize import POSSIZE from com.sun.star.awt import Key, KeyModifier, KeyEvent from com.sun.star.container import NoSuchElementException +from com.sun.star.datatransfer import XTransferable, DataFlavor from com.sun.star.beans import PropertyValue, NamedValue from com.sun.star.sheet import TableFilterField @@ -87,7 +106,14 @@ logging.addLevelName(logging.INFO, '\x1b[32mINFO\033[1;0m') logging.basicConfig(level=logging.DEBUG, format=LOG_FORMAT, datefmt=LOG_DATE) log = logging.getLogger(__name__) + +# ~ You can get custom salt +# ~ codecs.encode(os.urandom(16), 'hex') +SALT = b'c9548699d4e432dfd2b46adddafbb06d' + +TIMEOUT = 10 LOG_NAME = 'ZAZ' +FILE_NAME_CONFIG = 'zaz-{}.json' LEFT = 0 CENTER = 1 @@ -145,7 +171,12 @@ PC = platform.node() DESKTOP = os.environ.get('DESKTOP_SESSION', '') INFO_DEBUG = f"{sys.version}\n\n{platform.platform()}\n\n" + '\n'.join(sys.path) +PYTHON = 'python' +if IS_WIN: + PYTHON = 'python.exe' + _MACROS = {} +_start = 0 SECONDS_DAY = 60 * 60 * 24 DIR = { @@ -191,6 +222,8 @@ MESSAGES = { 'es': { 'OK': 'Aceptar', 'Cancel': 'Cancelar', + 'Select path': 'Seleccionar ruta', + 'Select directory': 'Seleccionar directorio', 'Select file': 'Seleccionar archivo', 'Incorrect user or password': 'Nombre de usuario o contraseña inválidos', 'Allow less secure apps in GMail': 'Activa: Permitir aplicaciones menos segura en GMail', @@ -274,7 +307,7 @@ def catch_exception(f): except Exception as e: name = f.__name__ if IS_WIN: - debug(traceback.format_exc()) + msgbox(traceback.format_exc()) log.error(name, exc_info=True) return func @@ -372,19 +405,6 @@ def _get_class_doc(obj: Any) -> Any: return classes[type_doc](obj) -def _get_class_uno(obj: Any) -> Any: - classes = dict( - SwXTextGraphicObject = LOImage, - SvxShapeText = LOImage, - ) - name = obj.ImplementationName - print(f'ImplementationName = {name}') - instance = obj - if name in classes: - instance = classes[name](obj) - return instance - - def dict_to_property(values: dict, uno_any: bool=False): ps = tuple([PropertyValue(Name=n, Value=v) for n, v in values.items()]) if uno_any: @@ -402,6 +422,14 @@ def _property_to_dict(values): return d +def json_dumps(data): + return json.dumps(data, indent=4, sort_keys=True) + + +def json_loads(data): + return json.loads(data) + + def data_to_dict(data): if isinstance(data, tuple) and isinstance(data[0], tuple): return _array_to_dict(data) @@ -411,18 +439,6 @@ def data_to_dict(data): return {} -def _path_url(path: str) -> str: - if path.startswith('file://'): - return path - return uno.systemPathToFileUrl(path) - - -def _path_system(path: str) -> str: - if path.startswith('file://'): - return str(Path(uno.fileUrlToSystemPath(path)).resolve()) - return path - - def _get_dispatch() -> Any: return create_instance('com.sun.star.frame.DispatchHelper') @@ -519,9 +535,12 @@ def call_macro(args, in_thread=False): return result -def run(command, capture=False): +def run(command, capture=False, split=True): + if not split: + return subprocess.check_output(command, shell=True).decode() + cmd = shlex.split(command) - result = subprocess.run(cmd, capture_output=capture, text=True) + result = subprocess.run(cmd, capture_output=capture, text=True, shell=IS_WIN) if capture: result = result.stdout else: @@ -529,15 +548,15 @@ def run(command, capture=False): return result -# ~ def popen(command, stdin=None): - # ~ try: - # ~ proc = subprocess.Popen(shlex.split(command), shell=IS_WIN, - # ~ stdout=subprocess.PIPE, stderr=subprocess.STDOUT) - # ~ for line in proc.stdout: - # ~ yield line.decode().rstrip() - # ~ except Exception as e: - # ~ error(e) - # ~ yield (e.errno, e.strerror) +def popen(command): + try: + proc = subprocess.Popen(shlex.split(command), shell=IS_WIN, + stdout=subprocess.PIPE, stderr=subprocess.STDOUT) + for line in proc.stdout: + yield line.decode().rstrip() + except Exception as e: + error(e) + yield (e.errno, e.strerror) def sleep(seconds): @@ -607,6 +626,266 @@ def sha512(data): return result +def get_config(key='', default={}, prefix='conf'): + name_file = FILE_NAME_CONFIG.format(prefix) + values = None + path = _P.join(_P.config('UserConfig'), name_file) + if not _P.exists(path): + return default + + values = _P.from_json(path) + if key: + values = values.get(key, default) + + return values + + +def set_config(key, value, prefix='conf'): + name_file = FILE_NAME_CONFIG.format(prefix) + path = _P.join(_P.config('UserConfig'), name_file) + values = get_config(default={}, prefix=prefix) + values[key] = value + result = _P.to_json(path, values) + return result + + +def start(): + global _start + _start = now() + info(_start) + return + + +def end(get_seconds=False): + global _start + e = now() + td = e - _start + result = str(td) + if get_seconds: + result = td.total_seconds() + return result + + +def get_epoch(): + n = now() + return int(time.mktime(n.timetuple())) + + +def render(template, data): + s = Template(template) + return s.safe_substitute(**data) + + +def get_size_screen(): + if IS_WIN: + user32 = ctypes.windll.user32 + res = f'{user32.GetSystemMetrics(0)}x{user32.GetSystemMetrics(1)}' + else: + args = 'xrandr | grep "*" | cut -d " " -f4' + res = run(args, split=False) + return res.strip() + + +def url_open(url, data=None, headers={}, verify=True, get_json=False): + err = '' + req = Request(url) + for k, v in headers.items(): + req.add_header(k, v) + try: + # ~ debug(url) + if verify: + if not data is None and isinstance(data, str): + data = data.encode() + response = urlopen(req, data=data) + else: + context = ssl._create_unverified_context() + response = urlopen(req, context=context) + except HTTPError as e: + error(e) + err = str(e) + except URLError as e: + error(e.reason) + err = str(e.reason) + else: + headers = dict(response.info()) + result = response.read() + if get_json: + result = json.loads(result) + + return result, headers, err + + +def _get_key(password): + from cryptography.hazmat.primitives import hashes + from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC + + kdf = PBKDF2HMAC(algorithm=hashes.SHA256(), length=32, salt=SALT, + iterations=100000) + key = base64.urlsafe_b64encode(kdf.derive(password.encode())) + return key + + +def encrypt(data, password): + from cryptography.fernet import Fernet + + f = Fernet(_get_key(password)) + if isinstance(data, str): + data = data.encode() + token = f.encrypt(data).decode() + return token + + +def decrypt(token, password): + from cryptography.fernet import Fernet, InvalidToken + + data = '' + f = Fernet(_get_key(password)) + try: + data = f.decrypt(token.encode()).decode() + except InvalidToken as e: + error('Invalid Token') + return data + + +class SmtpServer(object): + + def __init__(self, config): + self._server = None + self._error = '' + self._sender = '' + self._is_connect = self._login(config) + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + self.close() + + @property + def is_connect(self): + return self._is_connect + + @property + def error(self): + return self._error + + def _login(self, config): + name = config['server'] + port = config['port'] + is_ssl = config['ssl'] + self._sender = config['user'] + hosts = ('gmail' in name or 'outlook' in name) + try: + if is_ssl and hosts: + self._server = smtplib.SMTP(name, port, timeout=TIMEOUT) + self._server.ehlo() + self._server.starttls() + self._server.ehlo() + elif is_ssl: + self._server = smtplib.SMTP_SSL(name, port, timeout=TIMEOUT) + self._server.ehlo() + else: + self._server = smtplib.SMTP(name, port, timeout=TIMEOUT) + + self._server.login(self._sender, config['password']) + msg = 'Connect to: {}'.format(name) + debug(msg) + return True + except smtplib.SMTPAuthenticationError as e: + if '535' in str(e): + self._error = _('Incorrect user or password') + return False + if '534' in str(e) and 'gmail' in name: + self._error = _('Allow less secure apps in GMail') + return False + except smtplib.SMTPException as e: + self._error = str(e) + return False + except Exception as e: + self._error = str(e) + return False + return False + + def _body(self, msg): + body = msg.replace('\\n', '
') + return body + + def send(self, message): + file_name = 'attachment; filename={}' + email = MIMEMultipart() + email['From'] = self._sender + email['To'] = message['to'] + email['Cc'] = message.get('cc', '') + email['Subject'] = message['subject'] + email['Date'] = formatdate(localtime=True) + if message.get('confirm', False): + email['Disposition-Notification-To'] = email['From'] + email.attach(MIMEText(self._body(message['body']), 'html')) + + for path in message.get('files', ()): + fn = _P(path).file_name + part = MIMEBase('application', 'octet-stream') + part.set_payload(_P.read_bin(path)) + encoders.encode_base64(part) + part.add_header('Content-Disposition', f'attachment; filename={fn}') + email.attach(part) + + receivers = ( + email['To'].split(',') + + email['CC'].split(',') + + message.get('bcc', '').split(',')) + try: + self._server.sendmail(self._sender, receivers, email.as_string()) + msg = 'Email sent...' + debug(msg) + if message.get('path', ''): + self.save_message(email, message['path']) + return True + except Exception as e: + self._error = str(e) + return False + return False + + def save_message(self, email, path): + mbox = mailbox.mbox(path, create=True) + mbox.lock() + try: + msg = mailbox.mboxMessage(email) + mbox.add(msg) + mbox.flush() + finally: + mbox.unlock() + return + + def close(self): + try: + self._server.quit() + msg = 'Close connection...' + debug(msg) + except: + pass + return + + +def _send_email(server, messages): + with SmtpServer(server) as server: + if server.is_connect: + for msg in messages: + server.send(msg) + else: + error(server.error) + return server.error + + +def send_email(server, message): + messages = message + if isinstance(message, dict): + messages = (message,) + t = threading.Thread(target=_send_email, args=(server, messages)) + t.start() + return + + # ~ Classes class LOBaseObject(object): @@ -681,6 +960,10 @@ class LOImage(object): class LODocument(object): + FILTERS = { + 'doc': 'MS Word 97', + 'docx': 'MS Word 2007 XML', + } def __init__(self, obj): self._obj = obj @@ -730,7 +1013,19 @@ class LODocument(object): @property def path(self): - return _path_system(self.obj.URL) + return _P.to_system(self.obj.URL) + + @property + def dir(self): + return _P(self.path).path + + @property + def file_name(self): + return _P(self.path).file_name + + @property + def name(self): + return _P(self.path).name @property def status_bar(self): @@ -767,12 +1062,21 @@ class LODocument(object): else: um.enterHiddenUndoContext() + def clear_undo(self): + self.obj.getUndoManager().clear() + return + @property def selection(self): sel = self.obj.CurrentSelection # ~ return _get_class_uno(sel) return sel + @property + def table_auto_formats(self): + taf = create_instance('com.sun.star.sheet.TableAutoFormats') + return taf.ElementNames + def create_instance(self, name): obj = self.obj.createInstance(name) return obj @@ -790,6 +1094,7 @@ class LODocument(object): sc = create_instance('com.sun.star.datatransfer.clipboard.SystemClipboard') transferable = sc.getContents() self._cc.insertTransferable(transferable) + # ~ return self.obj.getCurrentSelection() return def select(self, obj): @@ -813,12 +1118,29 @@ class LODocument(object): return _P.exists(path_pdf) + def export(self, path: str, ext: str='', args: dict={}): + if not ext: + ext = _P(path).ext + filter_name = self.FILTERS[ext] + filter_data = dict_to_property(args, True) + args = { + 'FilterName': filter_name, + 'FilterData': filter_data, + } + opt = dict_to_property(args) + try: + self.obj.storeToURL(_P.to_url(path), opt) + except Exception as e: + error(e) + path = '' + return _P.exists(path) + def save(self, path: str='', args: dict={}) -> bool: result = True opt = dict_to_property(args) if path: try: - self.obj.storeAsURL(_path_url(path), opt) + self.obj.storeAsURL(_P.to_url(path), opt) except Exception as e: error(e) result = False @@ -831,6 +1153,64 @@ class LODocument(object): return +class LOCellStyle(LOBaseObject): + + def __init__(self, obj): + super().__init__(obj) + + @property + def name(self): + return self.obj.Name + + @property + def properties(self): + properties = self.obj.PropertySetInfo.Properties + data = {p.Name: getattr(self.obj, p.Name) for p in properties} + return data + @properties.setter + def properties(self, values): + _set_properties(self.obj, values) + + +class LOCellStyles(object): + + def __init__(self, obj, doc): + self._obj = obj + self._doc = doc + + def __len__(self): + return len(self.obj) + + def __getitem__(self, index): + return LOCellStyle(self.obj[index]) + + def __setitem__(self, key, value): + self.obj[key] = value + + def __delitem__(self, key): + if not isinstance(key, str): + key = key.Name + del self.obj[key] + + def __contains__(self, item): + return item in self.obj + + @property + def obj(self): + return self._obj + + @property + def names(self): + return self.obj.ElementNames + + def new(self, name: str=''): + obj = self._doc.create_instance('com.sun.star.style.CellStyle') + if name: + self.obj[name] = obj + obj = LOCellStyle(obj) + return obj + + class LOCalc(LODocument): def __init__(self, obj): @@ -841,9 +1221,20 @@ class LOCalc(LODocument): def __getitem__(self, index): return LOCalcSheet(self._sheets[index]) + def __setitem__(self, key, value): + self._sheets[key] = value + def __len__(self): return self._sheets.Count + def __contains__(self, item): + return item in self._sheets + + @property + def names(self): + names = self.obj.Sheets.ElementNames + return names + @property def selection(self): sel = self.obj.CurrentSelection @@ -861,11 +1252,107 @@ class LOCalc(LODocument): def active(self): return LOCalcSheet(self._cc.ActiveSheet) + @property + def headers(self): + return self._cc.ColumnRowHeaders + @headers.setter + def headers(self, value): + self._cc.ColumnRowHeaders = value + + @property + def tabs(self): + return self._cc.SheetTabs + @tabs.setter + def tabs(self, value): + self._cc.SheetTabs = value + + @property + def cs(self): + return self.cell_styles + @property + def cell_styles(self): + obj = self.obj.StyleFamilies['CellStyles'] + return LOCellStyles(obj, self) + @property def db_ranges(self): # ~ return LOCalcDataBaseRanges(self.obj.DataBaseRanges) return self.obj.DatabaseRanges + def activate(self, sheet): + obj = sheet + if isinstance(sheet, LOCalcSheet): + obj = sheet.obj + elif isinstance(sheet, str): + obj = self._sheets[sheet] + self._cc.setActiveSheet(obj) + return + + def new_sheet(self): + s = self.create_instance('com.sun.star.sheet.Spreadsheet') + return s + + def insert(self, name): + names = name + if isinstance(name, str): + names = (name,) + for n in names: + self._sheets[n] = self.new_sheet() + return LOCalcSheet(self._sheets[n]) + + def move(self, name, pos=-1): + index = pos + if pos < 0: + index = len(self) + if isinstance(name, LOCalcSheet): + name = name.name + self._sheets.moveByName(name, index) + return + + def remove(self, name): + if isinstance(name, LOCalcSheet): + name = name.name + self._sheets.removeByName(name) + return + + def copy(self, name, new_name='', pos=-1): + if isinstance(name, LOCalcSheet): + name = name.name + index = pos + if pos < 0: + index = len(self) + self._sheets.copyByName(name, new_name, index) + return LOCalcSheet(self._sheets[new_name]) + + def copy_from(self, doc, source='', target='', pos=-1): + index = pos + if pos < 0: + index = len(self) + + names = source + if not source: + names = doc.names + elif isinstance(source, str): + names = (source,) + + new_names = target + if not target: + new_names = names + elif isinstance(target, str): + new_names = (target,) + + for i, name in enumerate(names): + self._sheets.importSheet(doc.obj, name, index + i) + self[index + i].name = new_names[i] + + return LOCalcSheet(self._sheets[index]) + + def sort(self, reverse=False): + names = sorted(self.names, reverse=reverse) + for i, n in enumerate(names): + self.move(n, i) + return + def render(self, data, sheet=None, clean=True): if sheet is None: sheet = self.active @@ -1081,6 +1568,9 @@ class LOCalcSheet(object): def __exit__(self, exc_type, exc_value, traceback): pass + def __str__(self): + return f'easymacro.LOCalcSheet: {self.name}' + @property def obj(self): return self._obj @@ -1092,6 +1582,46 @@ class LOCalcSheet(object): def name(self, value): self._obj.Name = value + @property + def code_name(self): + return self._obj.CodeName + @code_name.setter + def code_name(self, value): + self._obj.CodeName = value + + @property + def visible(self): + return self._obj.IsVisible + @visible.setter + def visible(self, value): + self._obj.IsVisible = value + + @property + def is_protected(self): + return self._obj.isProtected() + + @property + def password(self): + return '' + @visible.setter + def password(self, value): + self.obj.protect(value) + + def unprotect(self, value): + try: + self.obj.unprotect(value) + return True + except: + pass + return False + + @property + def color(self): + return self._obj.TabColor + @color.setter + def color(self, value): + self._obj.TabColor = get_color(value) + @property def used_area(self): cursor = self.get_cursor() @@ -1112,7 +1642,7 @@ class LOCalcSheet(object): @property def doc(self): - return self.obj.DrawPage.Forms.Parent + return LOCalc(self.obj.DrawPage.Forms.Parent) @property def charts(self): @@ -1126,6 +1656,47 @@ class LOCalcSheet(object): def forms(self): return LOSheetForms(self.obj.DrawPage.Forms) + def activate(self): + self.doc.activate(self._obj) + return + + def clean(self): + doc = self.doc + sheet = doc.create_instance('com.sun.star.sheet.Spreadsheet') + doc._sheets.replaceByName(self.name, sheet) + return + + def move(self, pos=-1): + index = pos + if pos < 0: + index = len(self.doc) + self.doc._sheets.moveByName(self.name, index) + return + + def remove(self): + self.doc._sheets.removeByName(self.name) + return + + def copy(self, new_name='', pos=-1): + index = pos + if pos < 0: + index = len(self.doc) + self.doc._sheets.copyByName(self.name, new_name, index) + return LOCalcSheet(self.doc._sheets[new_name]) + + def copy_to(self, doc, target='', pos=-1): + index = pos + if pos < 0: + index = len(doc) + name = self.name + if not target: + new_name = name + + doc._sheets.importSheet(self.doc.obj, name, index) + sheet = doc[name] + sheet.name = new_name + return sheet + def get_cursor(self, cell=None): if cell is None: cursor = self.obj.createCursor() @@ -1139,11 +1710,39 @@ class LOCalcSheet(object): return rango.render(data, clean) +class LOCalcRows(object): + + def __init__(self, obj): + self._obj = obj + + def __len__(self): + return self.obj.Count + + def __str__(self): + return 'Rows' + + @property + def obj(self): + return self._obj + + @property + def count(self): + return len(self) + + @property + def visible(self): + return self.obj.IsVisible + @visible.setter + def visible(self, value): + self.obj.IsVisible = value + + class LOCalcRange(object): def __init__(self, obj): self._obj = obj self._sd = None + self._is_cell = obj.ImplementationName == OBJ_CELL def __getitem__(self, index): return LOCalcRange(self.obj[index]) @@ -1185,6 +1784,17 @@ class LOCalcRange(object): def is_none(self): return self.obj is None + @property + def is_cell(self): + return self._is_cell + + @property + def back_color(self): + return self._obj.CellBackColor + @back_color.setter + def back_color(self, value): + self._obj.CellBackColor = get_color(value) + @property def dp(self): return self.sheet.dp @@ -1211,9 +1821,18 @@ class LOCalcRange(object): def columns(self): return self.obj.Columns.Count + @property + def column(self): + c1 = self.address.Column + c2 = c1 + 1 + ra = self.current_region.range_address + r1 = ra.StartRow + r2 = ra.EndRow + 1 + return LOCalcRange(self.sheet[r1:r2, c1:c2].obj) + @property def rows(self): - return self.obj.Rows.Count + return LOCalcRows(self.obj.Rows) @property def row(self): @@ -1284,7 +1903,22 @@ class LOCalcRange(object): return self.obj.getDataArray() @data.setter def data(self, values): - self.obj.setDataArray(values) + if self._is_cell: + self.to_size(len(values), len(values[0])).data = values + else: + self.obj.setDataArray(values) + + @property + def dict(self): + rows = self.data + k = rows[0] + data = [dict(zip(k, r)) for r in rows[1:]] + return data + @dict.setter + def dict(self, values): + data = [tuple(values[0].keys())] + data += [tuple(d.values()) for d in values] + self.data = data @property def formula(self): @@ -1344,6 +1978,13 @@ class LOCalcRange(object): } return data + @property + def visible(self): + cursor = self.cursor + rangos = cursor.queryVisibleCells() + rangos = [LOCalcRange(self.sheet[r.AbsoluteName].obj) for r in rangos] + return tuple(rangos) + def select(self): self.doc.select(self.obj) return @@ -1367,10 +2008,31 @@ class LOCalcRange(object): rango.data = self.data return + def copy_from(self, rango, formula=False): + data = rango + if isinstance(rango, LOCalcRange): + if formula: + data = rango.formula + else: + data = rango.data + rows = len(data) + cols = len(data[0]) + if formula: + self.to_size(rows, cols).formula = data + else: + self.to_size(rows, cols).data = data + return + def auto_width(self): self.obj.Columns.OptimalWidth = True return + def clean_render(self, template='\{(\w.+)\}'): + self._sd.SearchRegularExpression = True + self._sd.setSearchString(template) + self.obj.replaceAll(self._sd) + return + def render(self, data, clean=True): self._sd = self.sheet.obj.createSearchDescriptor() self._sd.SearchCaseSensitive = False @@ -1474,6 +2136,19 @@ class LOCalcRange(object): args.clear() return img + def filter_by_color(self, cell): + rangos = cell.column[1:,:].visible + for r in rangos: + for c in r: + if c.back_color != cell.back_color: + c.rows.visible = False + return + + def clear(self, what=1023): + # ~ http://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1sheet_1_1CellFlags.html + self.obj.clearContents(what) + return + class LOWriterPageStyle(LOBaseObject): @@ -1512,6 +2187,18 @@ class LOWriterTextRange(object): self._is_paragraph = self.obj.ImplementationName == 'SwXParagraph' self._is_table = self.obj.ImplementationName == 'SwXTextTable' + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + for i, p in enumerate(self.obj): + if i == self._index: + obj = LOWriterTextRange(p, self._doc) + self._index += 1 + return obj + raise StopIteration + @property def obj(self): return self._obj @@ -1533,7 +2220,7 @@ class LOWriterTextRange(object): @property def text(self): - return self.obj.getText() + return self.obj.Text @property def cursor(self): @@ -1543,6 +2230,10 @@ class LOWriterTextRange(object): def dp(self): return self._doc.dp + @property + def is_table(self): + return self._is_table + def offset(self): cursor = self.cursor.getEnd() return LOWriterTextRange(cursor, self._doc) @@ -1572,7 +2263,23 @@ class LOWriterTextRanges(object): self._doc = doc def __getitem__(self, index): - return LOWriterTextRange(self.obj[index], self._doc) + for i, p in enumerate(self.obj): + if i == index: + obj = LOWriterTextRange(p, self._doc) + break + return obj + + def __iter__(self): + self._index = 0 + return self + + def __next__(self): + for i, p in enumerate(self.obj): + if i == self._index: + obj = LOWriterTextRange(p, self._doc) + self._index += 1 + return obj + raise StopIteration @property def obj(self): @@ -1585,6 +2292,14 @@ class LOWriter(LODocument): super().__init__(obj) self._type = WRITER + @property + def text(self): + return LOWriterTextRange(self.obj.Text, self) + + @property + def paragraphs(self): + return LOWriterTextRanges(self.obj.Text, self) + @property def selection(self): sel = self.obj.CurrentSelection @@ -1690,6 +2405,13 @@ class LOShape(LOBaseObject): value = value.obj self.obj.Anchor = value + @property + def visible(self): + return self.obj.Visible + @visible.setter + def visible(self, value): + self.obj.Visible = value + def remove(self): self.obj.Parent.remove(self.obj) return @@ -1796,7 +2518,8 @@ class LODrawImpress(LODocument): @property def selection(self): sel = self.obj.CurrentSelection[0] - return _get_class_uno(sel) + # ~ return _get_class_uno(sel) + return sel @property def current_page(self): @@ -2206,7 +2929,7 @@ class LODocs(object): http://api.libreoffice.org/docs/idl/ref/interfacecom_1_1sun_1_1star_1_1frame_1_1XComponentLoader.html http://api.libreoffice.org/docs/idl/ref/servicecom_1_1sun_1_1star_1_1document_1_1MediaDescriptor.html """ - path = _path_url(path) + path = _P.to_url(path) opt = dict_to_property(args) doc = cls._desktop.loadComponentFromURL(path, '_default', 0, opt) if doc is None: @@ -2372,9 +3095,10 @@ class EventsFocus(EventsListenerBase, XFocusListener): # ~ HelpURL = ? class UnoBaseObject(object): - def __init__(self, obj): + def __init__(self, obj, path=''): self._obj = obj self._model = obj.Model + # ~ self._path = path def __setattr__(self, name, value): exists = hasattr(self, name) @@ -2577,6 +3301,22 @@ class UnoBaseObject(object): def tab_stop(self, value): self.model.Tabstop = value + @property + def ps(self): + ps = self.obj.getPosSize() + return ps + @ps.setter + def ps(self, ps): + self.obj.setPosSize(ps.X, ps.Y, ps.Width, ps.Height, POSSIZE) + + def set_focus(self): + self.obj.setFocus() + return + + def ps_from(self, source): + self.ps = source.ps + return + def center(self, horizontal=True, vertical=False): p = self.parent.Model w = p.Width @@ -2739,6 +3479,78 @@ class UnoImage(UnoBaseObject): self.m.ImageURL = _P.to_url(value) +class UnoListBox(UnoBaseObject): + + def __init__(self, obj): + super().__init__(obj) + self._path = '' + + def __setattr__(self, name, value): + if name in ('_path',): + self.__dict__[name] = value + else: + super().__setattr__(name, value) + + @property + def type(self): + return 'listbox' + + @property + def value(self): + return self.obj.getSelectedItem() + + @property + def count(self): + return len(self.data) + + @property + def data(self): + return self.model.StringItemList + @data.setter + def data(self, values): + self.model.StringItemList = list(sorted(values)) + + @property + def path(self): + return self._path + @path.setter + def path(self, value): + self._path = value + + def unselect(self): + self.obj.selectItem(self.value, False) + return + + def select(self, pos=0): + if isinstance(pos, str): + self.obj.selectItem(pos, True) + else: + self.obj.selectItemPos(pos, True) + return + + def clear(self): + self.model.removeAllItems() + return + + def _set_image_url(self, image): + if _P.exists(image): + return _P.to_url(image) + + path = _P.join(self._path, DIR['images'], image) + return _P.to_url(path) + + def insert(self, value, path='', pos=-1, show=True): + if pos < 0: + pos = self.count + if path: + self.model.insertItem(pos, value, self._set_image_url(path)) + else: + self.model.insertItemText(pos, value) + if show: + self.select(pos) + return + + UNO_CLASSES = { 'label': UnoLabel, 'link': UnoLabelLink, @@ -2747,6 +3559,7 @@ UNO_CLASSES = { 'check': UnoCheck, 'text': UnoText, 'image': UnoImage, + 'listbox': UnoListBox, } @@ -2760,12 +3573,11 @@ class LODialog(object): 'check': 'com.sun.star.awt.UnoControlCheckBoxModel', 'text': 'com.sun.star.awt.UnoControlEditModel', 'image': 'com.sun.star.awt.UnoControlImageControlModel', + 'listbox': 'com.sun.star.awt.UnoControlListBoxModel', # ~ 'grid': 'com.sun.star.awt.grid.UnoControlGridModel', # ~ 'groupbox': 'com.sun.star.awt.UnoControlGroupBoxModel', - # ~ 'listbox': 'com.sun.star.awt.UnoControlListBoxModel', # ~ 'roadmap': 'com.sun.star.awt.UnoControlRoadmapModel', # ~ 'tree': 'com.sun.star.awt.tree.TreeControlModel', - # ~ 'groupbox': 'com.sun.star.awt.UnoControlGroupBoxModel', # ~ 'pages': 'com.sun.star.awt.UnoMultiPageModel', } @@ -2784,7 +3596,7 @@ class LODialog(object): path = args.pop('Path', '') if path: dp = create_instance(service, True) - dlg = dp.createDialog(_path_url(path)) + dlg = dp.createDialog(_P.to_url(path)) return dlg if 'Location' in args: @@ -2825,6 +3637,9 @@ class LODialog(object): return self._controls @property + def path(self): + return self._path + @property def id(self): return self._id @id.setter @@ -2853,6 +3668,13 @@ class LODialog(object): def visible(self, value): self.obj.Visible = value + @property + def step(self): + return self.model.Step + @step.setter + def step(self, value): + self.model.Step = value + @property def events(self): return self._events @@ -2918,6 +3740,8 @@ class LODialog(object): control = self.obj.getControl(name) _add_listeners(self.events, control, name) control = UNO_CLASSES[tipo](control) + if tipo in ('listbox',): + control.path = self.path if tipo == 'tree' and root: control.root = root @@ -3234,7 +4058,50 @@ class classproperty: return self +class ClipBoard(object): + SERVICE = 'com.sun.star.datatransfer.clipboard.SystemClipboard' + CLIPBOARD_FORMAT_TEXT = 'text/plain;charset=utf-16' + + class TextTransferable(unohelper.Base, XTransferable): + + def __init__(self, text): + df = DataFlavor() + df.MimeType = ClipBoard.CLIPBOARD_FORMAT_TEXT + df.HumanPresentableName = "encoded text utf-16" + self.flavors = (df,) + self._data = text + + def getTransferData(self, flavor): + return self._data + + def getTransferDataFlavors(self): + return self.flavors + + + @classmethod + def set(cls, value): + ts = cls.TextTransferable(value) + sc = create_instance(cls.SERVICE) + sc.setContents(ts, None) + return + + @classproperty + def contents(cls): + df = None + text = '' + sc = create_instance(cls.SERVICE) + transferable = sc.getContents() + data = transferable.getTransferDataFlavors() + for df in data: + if df.MimeType == cls.CLIPBOARD_FORMAT_TEXT: + break + if df: + text = transferable.getTransferData(df) + return text + + class Paths(object): + FILE_PICKER = 'com.sun.star.ui.dialogs.FilePicker' def __init__(self, path=''): if path.startswith('file://'): @@ -3255,7 +4122,7 @@ class Paths(object): @property def ext(self): - return self._path.suffix + return self._path.suffix[1:] @property def info(self): @@ -3265,14 +4132,32 @@ class Paths(object): def url(self): return self._path.as_uri() + @property + def size(self): + return self._path.stat().st_size + @classproperty def home(self): return str(Path.home()) + @classproperty + def documents(self): + return self.config() + @classproperty def temp_dir(self): return tempfile.gettempdir() + @classproperty + def python(self): + if IS_WIN: + path = self.join(self.config('Module'), PYTHON) + elif IS_MAC: + path = self.join(self.config('Module'), '..', 'Resources', PYTHON) + else: + path = sys.executable + return path + @classmethod def dir_tmp(self, only_name=False): dt = tempfile.TemporaryDirectory() @@ -3285,10 +4170,6 @@ class Paths(object): tmp = tempfile.NamedTemporaryFile(suffix=ext) return tmp.name - @classproperty - def python(self): - return sys.executable - @classmethod def config(cls, name='Work'): """ @@ -3298,6 +4179,80 @@ class Paths(object): path = create_instance('com.sun.star.util.PathSettings') return cls.to_system(getattr(path, name)) + @classmethod + def get(cls, init_dir='', filters=()): + """ + Options: http://api.libreoffice.org/docs/idl/ref/namespacecom_1_1sun_1_1star_1_1ui_1_1dialogs_1_1TemplateDescription.html + filters: Example + ( + ('XML', '*.xml'), + ('TXT', '*.txt'), + ) + """ + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + file_picker = create_instance(cls.FILE_PICKER) + file_picker.setTitle(_('Select path')) + file_picker.setDisplayDirectory(init_dir) + file_picker.initialize((2,)) + if filters: + file_picker.setCurrentFilter(filters[0][0]) + for f in filters: + file_picker.appendFilter(f[0], f[1]) + + path = '' + if file_picker.execute(): + path = cls.to_system(file_picker.getSelectedFiles()[0]) + return path + + @classmethod + def get_dir(cls, init_dir=''): + folder_picker = create_instance(cls.FILE_PICKER) + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + folder_picker.setTitle(_('Select directory')) + folder_picker.setDisplayDirectory(init_dir) + + path = '' + if folder_picker.execute(): + path = cls.to_system(folder_picker.getDisplayDirectory()) + return path + + @classmethod + def get_file(cls, init_dir='', filters=(), multiple=False): + """ + init_folder: folder default open + multiple: True for multiple selected + filters: Example + ( + ('XML', '*.xml'), + ('TXT', '*.txt'), + ) + """ + if not init_dir: + init_dir = cls.documents + init_dir = cls.to_url(init_dir) + + file_picker = create_instance(cls.FILE_PICKER) + file_picker.setTitle(_('Select file')) + file_picker.setDisplayDirectory(init_dir) + file_picker.setMultiSelectionMode(multiple) + + if filters: + file_picker.setCurrentFilter(filters[0][0]) + for f in filters: + file_picker.appendFilter(f[0], f[1]) + + path = '' + if file_picker.execute(): + files = file_picker.getSelectedFiles() + path = [cls.to_system(f) for f in files] + if not multiple: + path = path[0] + return path + @classmethod def replace_ext(cls, path, new_ext): p = Paths(path) @@ -3347,6 +4302,16 @@ class Paths(object): result = bool(Path(path).write_bytes(data)) return result + @classmethod + def read(cls, path, encoding='utf-8'): + data = Path(path).read_text(encoding=encoding) + return data + + @classmethod + def read_bin(cls, path): + data = Path(path).read_bytes() + return data + @classmethod def to_url(cls, path): if not path.startswith('file://'): @@ -3394,6 +4359,71 @@ class Paths(object): path = _P.to_system(pip.getPackageLocation(id_ext)) return path + @classmethod + def from_json(cls, path): + data = json.loads(cls.read(path)) + return data + + @classmethod + def to_json(cls, path, data): + data = json.dumps(data, indent=4, ensure_ascii=False, sort_keys=True) + return cls.save(path, data) + + @classmethod + def from_csv(cls, path, args={}): + # ~ See https://docs.python.org/3.7/library/csv.html#csv.reader + with open(path) as f: + rows = tuple(csv.reader(f, **args)) + return rows + + @classmethod + def to_csv(cls, path, data, args={}): + with open(path, 'w') as f: + writer = csv.writer(f, **args) + writer.writerows(data) + return + + @classmethod + def zip(cls, source, target='', pwd=''): + path_zip = target + if not isinstance(source, (tuple, list)): + path, _, name, _ = _P(source).info + start = len(path) + 1 + if not target: + path_zip = f'{path}/{name}.zip' + + if isinstance(source, (tuple, list)): + files = [(f, f[len(_P(f).path)+1:]) for f in source] + elif _P.is_file(source): + files = ((source, source[start:]),) + else: + files = [(f, f[start:]) for f in _P.walk(source)] + + compression = zipfile.ZIP_DEFLATED + with zipfile.ZipFile(path_zip, 'w', compression=compression) as z: + for f in files: + z.write(f[0], f[1]) + return + + @classmethod + def zip_content(cls, path): + with zipfile.ZipFile(path) as z: + names = z.namelist() + return names + + @classmethod + def unzip(cls, source, target='', members=None, pwd=None): + path = target + if not target: + path = _P(source).path + with zipfile.ZipFile(source) as z: + if not pwd is None: + pwd = pwd.encode() + if isinstance(members, str): + members = (members,) + z.extractall(path, members=members, pwd=pwd) + return True + @classmethod def copy(cls, source, target='', name=''): p, f, n, e = _P(source).info @@ -3415,6 +4445,8 @@ def __getattr__(name): return LODocs().active.active if name == 'selection': return LODocs().active.selection + if name == 'current_region': + return LODocs().active.selection.current_region if name in ('rectangle', 'pos_size'): return Rectangle() if name == 'paths': @@ -3429,6 +4461,8 @@ def __getattr__(name): return LOMenus() if name == 'shortcuts': return LOShortCuts() + if name == 'clipboard': + return ClipBoard raise AttributeError(f"module '{__name__}' has no attribute '{name}'") @@ -3792,4 +4826,3 @@ class LOServer(object): else: instance = self._sm.createInstance(name) return instance - diff --git a/zaz.py b/zaz.py index 0953e17..667fe97 100755 --- a/zaz.py +++ b/zaz.py @@ -4,6 +4,8 @@ # ~ This file is part of ZAZ. +# ~ https://git.elmau.net/elmau/zaz + # ~ ZAZ 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 3 of the License, or @@ -44,6 +46,10 @@ from conf import ( log) +EASYMACRO_TMP = 'easymacro2.py' +EASYMACRO = 'easymacro.py' + + class LiboXML(object): CONTEXT = { 'calc': 'com.sun.star.sheet.SpreadsheetDocument', @@ -549,7 +555,7 @@ def _update_files(): copyfile(source, target) if FILES['easymacro']: - source = 'easymacro2.py' + source = EASYMACRO target = _join(path_source, 'pythonpath', source) copyfile(source, target) @@ -595,7 +601,7 @@ def _update_files(): return -def _new(): +def _create(): if not _validate_new(): return @@ -617,7 +623,6 @@ def _get_info_path(path): def _zip_embed(source, files): PATH = 'Scripts/python/' - EASYMACRO = 'easymacro2.py' FILE_PYC = 'easymacro.pyc' p, f, name, e = _get_info_path(source) @@ -693,8 +698,6 @@ def _embed(args): def _locales(args): - EASYMACRO = 'easymacro2.py' - if args.files: files = args.files.split(',') else: @@ -726,8 +729,39 @@ def _update(): return +def _new(args): + if not args.target: + msg = 'Add argument target: -t PATH_TARGET' + log.error(msg) + return + + if not args.name: + msg = 'Add argument name: -n name-new-extension' + log.error(msg) + return + + path = _join(args.target, args.name) + _mkdir(path) + _mkdir(_join(path, 'files')) + _mkdir(_join(path, 'images')) + path_logo = 'images/pymacros.png' + copyfile(path_logo, _join(path, 'images/logo.png')) + copyfile('zaz.py', _join(path, 'zaz.py')) + copyfile(EASYMACRO_TMP, _join(path, 'easymacro.py')) + copyfile('conf.py.example', _join(path, 'conf.py')) + + msg = 'Folders and files copy successfully for new extension.' + log.info(msg) + msg = f'Change to folder: {path}' + log.info(msg) + return + def main(args): + if args.new: + _new(args) + return + if args.update: _update() return @@ -740,8 +774,8 @@ def main(args): _embed(args) return - if args.new: - _new() + if args.create: + _create() return if not _validate_update(): @@ -762,9 +796,13 @@ def main(args): def _process_command_line_arguments(): parser = argparse.ArgumentParser( description='Make LibreOffice extensions') - parser.add_argument('-i', '--install', dest='install', action='store_true', + parser.add_argument('-new', '--new', dest='new', action='store_true', default=False, required=False) - parser.add_argument('-n', '--new', dest='new', action='store_true', + parser.add_argument('-t', '--target', dest='target', default='') + parser.add_argument('-n', '--name', dest='name', default='', required=False) + parser.add_argument('-c', '--create', dest='create', action='store_true', + default=False, required=False) + parser.add_argument('-i', '--install', dest='install', action='store_true', default=False, required=False) parser.add_argument('-e', '--embed', dest='embed', action='store_true', default=False, required=False)