diff --git a/externals/Makefile b/externals/Makefile index 363736cf580cf0f63198b8c36d8412b8140a0543..3cefc37fc5ed505805856f0ce57b20172a2eeba5 100644 --- a/externals/Makefile +++ b/externals/Makefile @@ -134,11 +134,9 @@ ifeq ($(LIGHT),yes) lib_targets = loaders-libdir pddp INCREMENTAL = yes else -lib_targets = adaptive arraysize autotune bassemu boids bsaylor comport creb cxc cyclone disis earplug ekext ext13 fftease fluid freeverb ggee hcs iem_ambi iem_bin_ambi iemlib iemgui iemguts iem_adaptfilt iemmatrix iemxmlrpc iem_delay iem_roomsim iem_spec2 iem_tab jasch_lib loaders-libdir lyonpotpourri mapping markex maxlib mjlib moocow moonlib motex mrpeach oscx pan pdcontainer pddp pdlua pdogg plugin pmpd rjlib sigpack smlib tof unauthorized vbap windowing zexy +lib_targets = adaptive arraysize autotune bassemu boids bsaylor comport creb cxc cyclone disis earplug ekext ext13 fftease flatgui fluid freeverb ggee hcs iem_ambi iem_bin_ambi iemlib iemgui iemguts iem_adaptfilt iemmatrix iemxmlrpc iem_delay iem_roomsim iem_spec2 iem_tab jasch_lib loaders-libdir lyonpotpourri mapping markex maxlib mjlib moocow moonlib motex mrpeach oscx pan pdcontainer pddp pdlua pdogg plugin pmpd rjlib sigpack smlib tof unauthorized vbap windowing zexy endif -# DISABLED: flatgui - # NEW (IN-PROGRESS): flext # this is for libraries that don't compile (yet) on all platforms @@ -538,16 +536,21 @@ fftease_clean: #------------------------------------------------------------------------------# # FLATGUI +# +# Right now we're just building footils/knob and throwing it in the flatgui +# external directory for compatibility +# flatgui: - make -C $(externals_src)/flatgui CFLAGS="$(CFLAGS)" \ - PD_PATH=$(pd_src) PD_INCLUDE=$(pd_src)/src + make -C $(externals_src)/footils/knob CFLAGS="$(CFLAGS)" \ + PD_PATH=$(pd_src) pdbinpath=$(pd_src)/src \ + PD_INCLUDE=$(pd_src)/src flatgui_install: - make -C $(externals_src)/flatgui STRIP="$(STRIP)" \ + make -C $(externals_src)/footils/knob STRIP="$(STRIP)" \ DESTDIR="$(DESTDIR)" objectsdir="$(objectsdir)" install flatgui_clean: - make -C $(externals_src)/flatgui clean + make -C $(externals_src)/footils/knob clean #------------------------------------------------------------------------------# # FLEXT and FLEXT externals diff --git a/externals/footils/knob/Makefile.pdlibbuilder.revised b/externals/footils/knob/Makefile.pdlibbuilder.revised new file mode 100644 index 0000000000000000000000000000000000000000..ed11b2259b0c5f674cd6005c810f9d9ff1e20379 --- /dev/null +++ b/externals/footils/knob/Makefile.pdlibbuilder.revised @@ -0,0 +1,1207 @@ +# Makefile.pdlibbuilder dated 2016-06-26 + +version = 0.2.5 + +# Helper makefile for Pure Data external libraries. +# Written by Katja Vetter March-June 2015 for the public domain. No warranties. +# Inspired by Hans Christoph Steiner's Makefile Template and Stephan Beal's +# ShakeNMake. +# +# GNU make version >= 3.81 required. +# +# +#=== characteristics =========================================================== +# +# +# - defines build settings based on autodetected OS and architecture +# - defines rules to build Pd class- or lib executables from C or C++ sources +# - defines rules for libdir installation +# - defines convenience targets for developer and user +# - evaluates implicit dependencies for non-clean builds +# +# +#=== basic usage =============================================================== +# +# +# In your Makefile, define your Pd lib name and class files, and include +# Makefile.pdlibbuilder at the end of the Makefile. Like so: +# +# ________________________________________________________________________ +# +# # Makefile for mylib +# +# lib.name = mylib +# +# class.sources = myclass1.c myclass2.c +# +# datafiles = myclass1-help.pd myclass2-help.pd README.txt LICENSE.txt +# +# include Makefile.pdlibbuilder +# ________________________________________________________________________ +# +# +# For files in class.sources it is assumed that class basename == source file +# basename. The default target builds all classes as individual executables +# with Pd's default extension for the platform. For anything more than the +# most basic usage, continue reading. +# +# +#=== list of Makefile.pdlibbuilder API variables =============================== +# +# +# Variables available for definition in your library Makefile: +# +# - lib.name +# - lib.setup.sources +# - class.sources +# - common.sources +# - shared.sources +# - <classname>.class.sources +# - <classname>.class.ldflags +# - <classname>.class.ldlibs +# - cflags +# - ldflags +# - ldlibs +# - datafiles +# - datadirs +# - makefiles +# - makefiledirs +# - externalsdir +# +# Optional multiline defines evaluated per operating system: +# +# - forLinux +# - forDarwin +# - forWindows +# +# +# Variables avaialable for (re)definition via command arguments: +# +# - pdbinpath (Windows only) +# - pdincludepath +# - DESTDIR +# - prefix +# - libdir +# - pkglibdir +# - CPPFLAGS +# - CFLAGS +# - LDFLAGS +# - CC +# - CXX +# - INSTALL +# - INSTALL_PROGRAM +# - INSTALL_DATA +# - INSTALL_DIR +# +# Variables available for your makefile or as command argument: +# +# - objectsdir +# - make-lib-executable +# - suppress-wunused +# +# +#=== descriptions of Makefile.pdlibbuilder API variables ======================= +# +# +# lib.name: +# Name of the library directory as it will be installed / distributed. Also the +# name of the lib executable in the case where all classes are linked into +# a single binary. +# +# lib.setup.sources: +# Source file(s) (C or C++) which must be compiled only when linking all classes +# into a single lib binary. +# +# class.sources: +# All sources files (C or C++) for which the condition holds that +# class name == source file basename. +# +# <classname>.class.sources: +# Source file(s) (C or C++) specific to class <classname>. Use this for +# multiple-source classes or when class name != source file basename. +# +# common.sources: +# Source file(s) which must be statically linked to each class in the library. +# +# shared.sources: +# Source file(s) (C or C++) to build a shared dynamic link lib, to be linked +# with all class executables. +# +# cflags, ldflags, ldlibs: +# Define cflags (preprocessor&compiler), ldflags (linker) and ldlibs (dynamic +# link libs) for the whole library. These flags are added to platform-specific +# flags defined by Makefile.pdlibbuilder. +# +# <classname>.class.ldflags and <classname>.class.ldlibs: +# Define ldflags resp. ldlibs specific to class <classname>. These flags are +# added to platform-specific flags defined by Makefile.pdlibbuilder, and flags +# defined in your Makefile for the whole library. Note: cflags can not be +# defined per class in the current implementation. +# +# datafiles and datadirs: +# All extra files you want to include in binary distributions of the +# library: abstractions and help patches, example patches, meta patch, readme +# and license texts, manuals, sound files, etcetera. Use 'datafiles' for all +# files that should go into your lib rootdir and 'datadirs' for complete +# directories you want to copy from source to distribution. +# +# externalsdir: +# Relative path to directory 'externals' in the context of pd-extended SVN, or +# any other centralized build layout for multiple libraries. Default value +# is '..', meaning the direct parent. The value is used in search paths for +# pd core components (header files, and executable in the case of Windows). +# +# forLinux, forDarwin, forWindows: +# Shorthand for 'variable definitions for Linux only' etc. Use like: +# define forLinux +# cflags += -DLINUX +# class.sources += linuxthing.c +# endef +# +# makefiles and makefiledirs: +# Extra makefiles or directories with makefiles that should be made in sub-make +# processes. +# +# pdbinpath: +# For Windows only. Directory where pd.dll can be found for linking. +# +# pdincludepath: +# Directory where Pd API m_pd.h can be found, and other Pd header files. +# +# DESTDIR, prefix, libdir: +# Components of the path for installation as conventionally used on Linux. +# +# pkglibdir: +# Base path for installation of Pd library directories. Default is specified +# per OS, see section about paths below. +# +# objectsdir: +# Alias of pkglibdir. Can be defined in your makefile to enable project- +# dependent relative install locations. +# +# CPPFLAGS: +# Preprocessor flags which are not strictly required for building. +# +# CFLAGS: +# Compiler flags which are not strictly required for building. Compiler flags +# defined by Makefile.pdlibbuilder for warning, optimization and architecture +# specification are overriden by CFLAGS. +# +# LDFLAGS: +# Linker flags which are not strictly required for building. Linker flags +# defined by Makefile.pdlibbuilder for architecture specification are overriden +# by LDFLAGS. +# +# CC and CXX: +# C and C++ compiler programs as defined in your build environment. +# +# INSTALL, INSTALL_PROGRAM, INSTALL_DATA, INSTALL_DIR: +# Definitions of install program, may be overriden via command argument. +# +# make-lib-executable: +# When this variable is defined 'yes' in your makefile or as command argument, +# Makefile.pdlibbuilder will try to build all classes into a single library +# executable (but it will force exit if lib.setup.sources is undefined). +# If your makefile defines 'make-lib-executable=yes' as the library default, +# this can still be overriden with 'make-lib-executable=no' as command argument +# to build individual class executables (the Makefile.pdlibbuilder default.) +# +# suppress-wunused: +# When this variable is defined ('yes' or any other value), -Wunused-variable, +# -Wunused-parameter, -Wunused-value and -Wunused-function are suppressed, +# but the other warnings from -Wall are retained. +# +# +#=== paths ===================================================================== +# +# +# Source files in directories other than current working directory must be +# prefixed with their relative path. Do not rely on VPATH or vpath. +# Object (.o) files are built in the directory of their source files. +# Executables are built in current working directory. +# +# Variable 'pdincludepath' stores a location where m_pd.h is expected to reside. +# Locations where Makefile.pdlibbuilder tries to find it, in order of priority: +# +# any OS: $(externalsdir)../pd/src +# +# Linux: /usr/include/pdextended +# /usr/include/pd +# +# OSX: /Applications/Pd-extended.app/Contents/Resources/include/pdextended +# /Applications/Pd.app/Contents/Resources/src +# +# Windows: %PROGRAMFILES%/pd/include/pdextended +# %PROGRAMFILES%/pd/src +# +# The path for installation of all library components is constructed as: +# +# installpath := $(DESTDIR)$(objectsdir)/$(lib.name) +# +# Default for 'objectsdir' is defined per platform and follows this convention: +# https://puredata.info/docs/faq/how-do-i-install-externals-and-help-files +# +# Linux: /usr/local/lib/pd-externals +# OSX: ~/Library/Pd +# Windows: %APPDATA%/Pd +# +# The rationale for not installing to ~/pd-externals by default on Linux +# is that some people share the home dir between 32 and 64 bit installations. +# +# +#=== targets =================================================================== +# +# +# all: build $(executables) plus optional post target +# post: target to build after $(executables) +# alldebug: build all with -g option turned on for debug symbols +# <classname>: force clean build of an individual class +# <sourcefile>.pre: make preprocessor output file in current working directory +# <sourcefile>.lst: make asm/source output file in current working directory +# +# install: install executables and data files +# clean: remove build products from source tree +# +# help: print help text +# vars: print makefile variables +# allvars: print all variables +# depend: print generated prerequisites +# coffee: dummy target +# +# Variable $(executables) expands to class executables plus optional shared lib, +# or alternatively to single lib executable when make-lib-executable=true. +# Targets pre and post can be defined by library makefile. Make sure to include +# Makefile.pdlibbuilder first so default target all will not be redefined. +# +# +#=== Pd-extended libdir concept ================================================ +# +# +# For libdir layout as conceived by Hans-Christoph Steiner, see: +# +# https://puredata.info/docs/developer/Libdir +# +# Files README.txt, LICENSE.txt and <lib.name>-meta.pd are part of the libdir +# convention. Help patches for each class and abstraction are supposed to be +# available. Makefile.pdlibbuilder does not force the presence of these files +# however. It does not automatically include such files in libdir installations. +# Data files you want to include in distributions must be defined explicitly in +# your Makefile. +# +# +#=== Makefile.pdlibbuilder syntax conventions ================================== +# +# +# Makefile.pdlibbuilder variable names are lower case. Default make variables, +# environment variables, and standard user variables (CC, CXX, CFLAGS, DESTDIR) +# are upper case. Use target 'allvars' to print all variables and their values. +# +# 'Fields' in data variables are separated by dots, like in 'foo.class.sources'. +# Words in variables expressing a function or command are separated by dashes, +# like in 'make-lib-executable'. +# +# +#=== useful make options ======================================================= +# +# +# Use 'make -d <target>' to print debug details of the make process. +# Use 'make -p <target>' to print make's database. +# +# +#=== TODO ====================================================================== +# +# +# - decide whether to use -static-libgcc or shared dll in MinGW +# - cygwin support +# - android support +# - Windows 64 bit support +# - figure out how to handle '$' in filenames +# - add makefile template targets dpkg-source dist libdir distclean tags? +# +# +#=== end of documentation sections ============================================= +# +# +################################################################################ +################################################################################ +################################################################################ + + +# GNU make version 3.81 (2006) or higher is required because of the following: +# - function 'info' +# - variable '.DEFAULT_GOAL' + +# force exit when make version is < 3.81 +ifneq ($(firstword $(sort 3.81 $(MAKE_VERSION))), 3.81) + $(error GNU make version 3.81 or higher is required) +endif + +# Relative path to externals root dir in multi-lib source tree like +# pd-extended SVN. Default is parent of current working directory. May be +# defined differently in including makefile. This variable is used to probe for +# paths. +externalsdir ?= .. + +# variable you can use to check if Makefile.pdlibbuilder is already included +Makefile.pdlibbuilder = true + + +################################################################################ +### variables: library name and version ######################################## +################################################################################ + + +# strip possibles spaces from lib.name, they mess up calculated file names +lib.name := $(strip $(lib.name)) + +# if meta file exists, check library version +metafile := $(wildcard $(lib.name)-meta.pd) + +ifdef metafile + lib.version := $(shell sed -n \ + 's|^\#X text [0-9][0-9]* [0-9][0-9]* VERSION \(.*\);|\1|p' \ + $(metafile)) +endif + + +################################################################################ +### variables: files ########################################################### +################################################################################ + + +#=== sources =================================================================== + + +# (re)define <classname>.class.sources using file names in class.sources + +define add-class-source +$(notdir $(basename $v)).class.sources += $v +endef + +$(foreach v, $(class.sources), $(eval $(add-class-source))) + +# derive class names from <classname>.class.sources variables +sourcevariables := $(filter %.class.sources, $(.VARIABLES)) +classes := $(basename $(basename $(sourcevariables))) + +# accumulate all source files specified in makefile +classes.sources := $(sort $(foreach v, $(sourcevariables), $($v))) +all.sources := $(classes.sources) $(lib.setup.sources) \ + $(shared.sources) $(common.sources) + + +#=== object files ============================================================== + + +# construct object filenames from all C and C++ source file names +classes.objects := $(addsuffix .o, $(basename $(classes.sources))) +common.objects := $(addsuffix .o, $(basename $(common.sources))) +shared.objects := $(addsuffix .o, $(basename $(shared.sources))) +lib.setup.objects := $(addsuffix .o, $(basename $(lib.setup.sources))) +all.objects = $(classes.objects) $(common.objects) $(shared.objects) \ + $(lib.setup.objects) + + +#=== executables =============================================================== + + +# use recursive variables here because executable extension is not yet known + +# construct class executable names from class names +classes.executables = $(addsuffix .$(extension), $(classes)) + +# construct shared lib executable name if shared sources are defined +ifdef shared.sources + shared.lib = lib$(lib.name).$(shared.extension) +else + shared.lib = +endif + + +################################################################################ +### variables per platform ##################################################### +################################################################################ + + +#=== flags per architecture ==================================================== + + +# Set architecture-dependent cflags, mainly for Linux. For Mac and Windows, +# arch.c.flags are overriden below. + +machine := $(shell uname -m) + +# Raspberry Pi 1st generation +ifeq ($(machine), armv6l) + arch.c.flags = -march=armv6 -mfpu=vfp -mfloat-abi=hard +endif + +# Beagle, Udoo, RPi2 etc. +ifeq ($(machine), armv7l) + arch.c.flags = -march=armv7-a -mfpu=vfpv3 -mfloat-abi=hard +endif + +# Intel 32 bit, build with SSE and SSE2 instructions +ifeq ($(findstring $(machine), i386 i686), $(machine)) + arch.c.flags = -march=pentium4 -mfpmath=sse -msse -msse2 +endif + +# Intel/AMD 64 bit, build with SSE, SSE2 and SSE3 instructions +ifeq ($(findstring $(machine), ia64 x86_64), $(machine)) + arch.c.flags = -march=core2 -mfpmath=sse -msse -msse2 -msse3 +endif + + +#=== operating system ========================================================== + + +# The following systems are defined: Linux, Darwin, Windows. GNU and +# GNU/kFreeBSD are treated as Linux to get the same options. System-specific +# multiline defines (optionally set in library makefile) are conditionally +# evaluated here. + +uname := $(shell uname) + +ifeq ($(findstring $(uname), Linux GNU GNU/kFreeBSD), $(uname)) + system = Linux + $(eval $(forLinux)) +endif + +ifeq ($(uname), Darwin) + system = Darwin + $(eval $(forDarwin)) +endif + +ifeq ($(findstring MINGW, $(uname)), MINGW) + system = Windows + $(eval $(forWindows)) +endif + +# TODO: Cygwin, Android + + +#=== flags and paths for Linux ================================================= + + +ifeq ($(system), Linux) + prefix = /usr/local + libdir := $(prefix)/lib + pkglibdir = $(libdir)/pd-externals + pdincludepath := $(firstword $(wildcard \ + $(externalsdir)/../pd/src \ + /usr/include/pdextended \ + /usr/include/pd)) + extension = pd_linux + cpp.flags := -DUNIX + c.flags := -fpic + c.ldflags := -rdynamic -shared -fpic -Wl,-rpath,"\$$ORIGIN",--enable-new-dtags + c.ldlibs := -lc -lm + cxx.flags := -fpic -fcheck-new + cxx.ldflags := -rdynamic -shared -fpic -Wl,-rpath,"\$$ORIGIN",--enable-new-dtags + cxx.ldlibs := -lc -lm -lstdc++ + shared.extension = so + shared.ldflags := -rdynamic -fpic -shared -Wl,-soname,$(shared.lib) + stripflags = --strip-unneeded -R .note -R .comment +endif + + +#=== flags and paths for Darwin ================================================ + + +# On OSX we try to build fat binaries by default. It is assumed that OSX i386 +# can build for ppc and OSX x86_64 can't. TODO: try to refine this condition. +# LLVM-clang doesn't support -fcheck-new, therefore this flag is omitted for +# OSX x86_64. + +ifeq ($(system), Darwin) + pkglibdir = $(HOME)/Library/Pd + pdincludepath := $(firstword $(wildcard \ + $(externalsdir)/../pd/src \ + /Applications/Pd-extended*.app/Contents/Resources/include/pdextended \ + /Applications/Pd*.app/Contents/Resources/src)) + extension = pd_darwin + cpp.flags := -DUNIX -DMACOSX -I /sw/include + c.flags := + c.ldflags := -undefined suppress -flat_namespace -bundle + c.ldlibs := -lc + cxx.ldflags := -undefined suppress -flat_namespace -bundle + cxx.ldlibs := -lc + shared.extension = dylib + shared.ldflags = -dynamiclib -undefined dynamic_lookup \ + -install_name @loader_path/$(shared.lib) \ + -compatibility_version 1 -current_version 1.0 + stripflags = -x + ifeq ($(machine), i386) + cxx.flags := -fcheck-new + arch.c.flags := -arch ppc -arch i386 -arch x86_64 -mmacosx-version-min=10.4 + arch.ld.flags := -arch ppc -arch i386 -arch x86_64 -mmacosx-version-min=10.4 + endif + ifeq ($(machine), x86_64) + arch.c.flags := -arch x86_64 -mmacosx-version-min=10.5 + arch.ld.flags := -arch x86_64 -mmacosx-version-min=10.5 + endif +endif + + +#=== flags and paths for Windows =============================================== + + +# Standard paths on Windows contain spaces, and GNU make functions treat such +# paths as lists, with unintended effects. Therefore we must use shell function +# ls instead of make's wildcard, and probe for each standard path individually. +# Using double quotes around paths with spaces is obligatory. Since some path +# variables are assembled or re-expanded later, great care must be taken to put +# quotes at appropriate points throughout the makefile. Thanks, Bill. + +# paths for 32-bit executables on 64-bit Windows aren't yet defined here (TODO) +ifeq ($(system), Windows) + pkglibdir := $(APPDATA)/Pd + pdbinpath := $(wildcard $(externalsdir)/../pd/bin) + pdincludepath := $(wildcard $(externalsdir)/../pd/src) + ifndef pdbinpath + pdbinpath := $(shell ls -d "$(PROGRAMFILES)/pd/bin") + endif + ifndef pdincludepath + pdincludepath := $(shell ls -d "$(PROGRAMFILES)/pd/include/pdextended") + endif + ifndef pdincludepath + pdincludepath := $(shell ls -d "$(PROGRAMFILES)/pd/src") + endif +endif + +# On Windows we build 32 bit by default to match Pd(-extended) binary +# distributions. This may change in the future. +# TODO: decide whether -mms-bitfields should be specified. +ifeq ($(system), Windows) + extension = dll + CC = gcc + CXX = g++ + arch.c.flags := -march=pentium4 -msse -msse2 -mfpmath=sse + cpp.flags := -DMSW -DNT + c.flags := + c.ldflags := -static-libgcc -shared \ + -Wl,--enable-auto-import "$(pdbinpath)/pd.dll" + c.ldlibs := + cxx.flags := -fcheck-new + cxx.ldflags := -static-libstdc++ -shared \ + -Wl,--enable-auto-import "$(pdbinpath)/pd.dll" + cxx.ldlibs := + shared.extension = dll + shared.ldflags := -static-libgcc -shared "$(pdbinpath)/pd.dll" + stripflags = --strip-unneeded -R .note -R .comment +endif + + +#=== paths ===================================================================== + + +# Default pkglibdir is specified above per operating system. It is aliased as +# 'objectsdir' to retain compatibility with pd-extended template. Assignment +# operator '?=' is used to enable a project-relative path definition in the +# including makefile. +objectsdir ?= $(pkglibdir) + +# base path where all components of the lib will be installed by default +installpath := $(DESTDIR)$(objectsdir)/$(lib.name) + +# check if pdincludepath contains spaces (as is often the case on Windows) +# if so, store the path so we can later do checks with it +pdincludepathwithspaces := $(if $(word 2, $(pdincludepath)), $(pdincludepath)) + + +#=== accumulated build flags =================================================== + + +# From GNU make docs: 'Users expect to be able to specify CFLAGS freely +# themselves.' So we use CFLAGS to define options which are not strictly +# required for compilation: optimizations, architecture specifications, and +# warnings. CFLAGS can be safely overriden using a make command argument. +# Variables cflags, ldflags and ldlibs may be defined in including makefile. + +optimization.flags = -O3 -ffast-math -funroll-loops -fomit-frame-pointer +warn.flags = -Wall -Wextra -Wshadow -Winline -Wstrict-aliasing + +# suppress -Wunused-variable & Co if you don't want to clutter a build log +ifdef suppress-wunused + warn.flags += $(addprefix -Wno-unused-, function parameter value variable) +endif + +CFLAGS = $(warn.flags) $(optimization.flags) $(arch.c.flags) + +# preprocessor flags +cpp.flags += -DPD -I "$(pdincludepath)" $(CPPFLAGS) + +# architecture specifications for linker are overridable by LDFLAGS +LDFLAGS := $(arch.ld.flags) + +# now add the same ld flags to shared dynamic lib +shared.ldflags := $(shared.ldflags) $(LDFLAGS) + +# accumulated flags for C compiler / linker +c.flags := $(cpp.flags) $(c.flags) $(cflags) $(CFLAGS) +c.ldflags := $(c.ldflags) $(ldflags) $(LDFLAGS) +c.ldlibs := $(c.ldlibs) $(ldlibs) + +# accumulated flags for C++ compiler / linker +cxx.flags := $(cpp.flags) $(cxx.flags) $(cflags) $(CFLAGS) +cxx.ldflags := $(cxx.ldflags) $(ldflags) $(LDFLAGS) +cxx.ldlibs := $(cxx.ldlibs) $(ldlibs) + + +################################################################################ +### variables: tools ########################################################### +################################################################################ + + +# aliases so we can later define 'compile-$1' and set 'c' or 'cxx' as argument +compile-c := $(CC) +compile-cxx := $(CXX) + + +################################################################################ +### checks ##################################################################### +################################################################################ + + +# At this point most variables are defined. Now do some checks and info's +# before rules begin. + +# 'forward declaration' of default target, needed to do checks +all: + +# To avoid unpredictable results, make sure the default target is not redefined +# by including makefile. +ifneq ($(.DEFAULT_GOAL), all) + $(error Default target must be 'all'.) +endif + +# find out which target(s) will be made +ifdef MAKECMDGOALS + goals := $(MAKECMDGOALS) +else + goals := all +endif + +# store path to Pd API m_pd.h if it is found +ifdef pdincludepath + mpdh := $(shell ls "$(pdincludepath)/m_pd.h") +endif + +# print Makefile.pdlibbuilder version +$(info ++++ info: using Makefile.pdlibbuilder version $(version)) + +# when making target all, check if m_pd.h is found and print info about it +ifeq ($(goals), all) + $(if $(mpdh), \ + $(info ++++ info: using Pd API $(mpdh)), \ + $(warning Where is Pd API m_pd.h? Do 'make help' for info.)) +endif + +# print target info +$(info ++++ info: making target $(goals) $(if $(lib.name),in lib $(lib.name))) + +# when installing, print installpath info +$(if $(filter install install-lib, $(goals)), $(info ++++ info: \ + installpath is '$(installpath)')) + + +#=== define executables ======================================================== + + +# By default we build class executables, and optionally a shared dynamic link +# lib. When make-lib-executable=yes we build all classes into a single lib +# executable, on the condition that variable lib.setup.sources is defined. + +ifeq ($(make-lib-executable),yes) + $(if $(lib.setup.sources), ,\ + $(error Can not build library blob because lib.setup.sources is undefined)) + executables := $(lib.name).$(extension) +else + executables := $(classes.executables) $(shared.lib) +endif + + +################################################################################ +### rules: special targets ##################################################### +################################################################################ + + +# Disable built-in rules. If some target can't be built with the specified +# rules, it should not be built at all. +MAKEFLAGS += --no-builtin-rules + +.PRECIOUS: +.SUFFIXES: +.PHONY: all post build-lib \ + $(classes) $(makefiledirs) $(makefiles) \ + install install-executables install-datafiles install-datadirs \ + force clean vars allvars depend help + + +################################################################################ +### rules: build targets ####################################################### +################################################################################ + + +# Target all forces the build of targets [$(executables) post] in +# deterministic order. Target $(executables) builds class executables plus +# optional shared lib or alternatively a single lib executable when +# make-lib-executable=true. Target post is optionally defined by +# library makefile. + +all: post +post: $(executables) + +all: + $(info ++++info: target all in lib $(lib.name) completed) + +# build all with -g option turned on for debug symbols +alldebug: c.flags += -g +alldebug: cxx.flags += -g +alldebug: all + + +#=== class executable ========================================================== + + +# recipe for linking objects in class executable +# argument $1 = compiler type (c or cxx) +# argument $2 = class basename +define link-class + $(compile-$1) \ + $($1.ldflags) $($2.class.ldflags) \ + -o $2.$(extension) \ + $(addsuffix .o, $(basename $($2.class.sources))) \ + $(addsuffix .o, $(basename $(common.sources))) \ + $($1.ldlibs) $($2.class.ldlibs) $(shared.lib) +endef + +# general rule for linking object files in class executable +%.$(extension): $(shared.lib) + $(info ++++ info: linking objects in $@ for lib $(lib.name)) + $(if $(filter %.cc %.cpp, $($*.class.sources)), \ + $(call link-class,cxx,$*), \ + $(call link-class,c,$*)) + + +#=== library blob ============================================================== + + +# build all classes into single executable +build-lib: $(lib.name).$(extension) + $(info ++++ info: library blob $(lib.name).$(extension) completed) + +# recipe for linking objects in lib executable +# argument $1 = compiler type (c or cxx) +define link-lib + $(compile-$1) \ + $($1.ldflags) $(lib.ldflags) \ + -o $(lib.name).$(extension) $(all.objects) \ + $($1.ldlibs) $(lib.ldlibs) +endef + +# rule for linking objects in lib executable +# declared conditionally to avoid name clashes +ifeq ($(make-lib-executable),yes) +$(lib.name).$(extension): $(all.objects) + $(if $(filter %.cc %.cpp, $(all.sources)), \ + $(call link-lib,cxx), \ + $(call link-lib,c)) +endif + + +#=== shared dynamic lib ======================================================== + + +# recipe for linking objects in shared executable +# argument $1 = compiler type (c or cxx) +define link-shared + $(compile-$1) \ + $(shared.ldflags) \ + -o lib$(lib.name).$(shared.extension) $(shared.objects) \ + $($1.ldlibs) $(shared.ldlibs) +endef + +# rule for linking objects in shared executable +# build recipe is in macro 'link-shared' +lib$(lib.name).$(shared.extension): $(shared.objects) + $(info ++++ info: linking objects in shared lib $@) + $(if $(filter %.cc %.cpp, $(shared.sources)), \ + $(call link-shared,cxx), \ + $(call link-shared,c)) + + +#=== object files ============================================================== + + +# recipe to make .o file from source +# argument $1 is compiler type (c or cxx) +define make-object-file + $(info ++++ info: making $@ in lib $(lib.name)) + $(compile-$1) \ + $($1.flags) \ + -o $@ -c $< +endef + +# Three rules to create .o files. These are double colon 'terminal' rules, +# meaning they are the last in a rules chain. + +%.o:: %.c + $(call make-object-file,c) + +%.o:: %.cc + $(call make-object-file,cxx) + +%.o:: %.cpp + $(call make-object-file,cxx) + + +#=== explicit prerequisites for class executables ============================== + + +# For class executables, prerequisite rules are declared in run time. Target +# 'depend' prints these rules for debugging purposes. + +# declare explicit prerequisites rule like 'class: class.extension' +# argument $v is class basename +define declare-class-target +$v: $v.$(extension) +endef + +# declare explicit prerequisites rule like 'class.extension: object1.o object2.o' +# argument $v is class basename +define declare-class-executable-target +$v.$(extension): $(addsuffix .o, $(basename $($v.class.sources))) \ + $(addsuffix .o, $(basename $(common.sources))) +endef + +# evaluate explicit prerequisite rules for all classes +$(foreach v, $(classes), $(eval $(declare-class-target))) +$(foreach v, $(classes), $(eval $(declare-class-executable-target))) + + +#=== implicit prerequisites for class executables ============================== + + +# Evaluating implicit prerequisites (header files) with help from the +# preprocessor is 'expensive' so this is done conditionally and selectively. +# Note that it is also possible to trigger a build via install targets, in +# which case implicit prerequisites are not checked. + +# When the Pd include path contains spaces it will mess up the implicit +# prerequisites rules. Also it is known that multiple arch flags are +# incompatible with preprocessor option -MM on OSX <= 10.5. Dependency +# tracking must be disabled in those cases. + +oldfat := $(and $(filter ppc i386, $(machine)), \ + $(filter-out 0 1, $(words $(filter -arch, $(c.flags))))) + +disable-dependency-tracking := $(strip $(pdincludepathwithspaces) $(oldfat)) + +ifndef disable-dependency-tracking + must-build-everything := $(filter all, $(goals)) + must-build-class := $(filter $(classes), $(goals)) + must-build-sources := $(foreach v, $(must-build-class), $($v.class.sources)) +endif + +# declare implicit prerequisites rule like 'object.o: header1.h header2.h ...' +# argument $1 is input source file(s) +# dir is explicitly added because option -MM strips it by default +define declare-object-target +$(dir $1)$(filter %.o: %.h, $(shell $(CPP) $(c.flags) -MM $1)) $(MAKEFILE_LIST) +endef + +# evaluate implicit prerequisite rules when rebuilding everything +ifdef must-build-everything + $(if $(wildcard $(all.objects)), \ + $(info ++++ info: evaluating implicit prerequisites in lib $(lib.name).....) \ + $(foreach v, $(all.sources), $(eval $(call declare-object-target, $v)))) +endif + +# evaluate implicit prerequisite rules when selectively building classes +ifdef must-build-class + $(foreach v, $(must-build-sources), \ + $(eval $(call declare-object-target, $v))) + $(foreach v, $(shared.sources), \ + $(eval $(call declare-object-target, $v))) +endif + + +################################################################################ +### rules: preprocessor and assembly files ##################################### +################################################################################ + + +# Preprocessor and assembly output files for bug tracing etc. They are not part +# of the build processes for executables. By default these files are created in +# the current working directory. Dependency tracking is not performed, the build +# is forced instead to make sure it's up to date. + +force: + + +#=== preprocessor file ========================================================= + + +# make preprocessor output file with extension .pre +# argument $1 = compiler type (c or cxx) +define make-preprocessor-file + $(info ++++ info: making preprocessor output file $(notdir $*.pre) \ + in current working directory) + $(compile-$1) -E $< $(c.flags) $($1.flags) -o $(notdir $*.pre) +endef + +%.pre:: %.c force + $(call make-preprocessor-file,c) + +%.pre:: %.cc force + $(call make-preprocessor-file,cxx) + +%.pre:: %.cpp force + $(call make-preprocessor-file,cxx) + + +#=== assembly file ============================================================= + + +# make C / assembly interleaved output file with extension .lst +# argument $1 = compiler type (c or cxx) +define make-assembly-file + $(info ++++ info: making assembly output file $(notdir $*.lst) \ + in current working directory) + $(compile-$1) \ + -c -Wa,-a,-ad -fverbose-asm \ + $($1.flags) \ + $< > $(notdir $*.lst) +endef + +%.lst:: %.c force + $(call make-assembly-file,c) + +%.lst:: %.cc force + $(call make-assembly-file,cxx) + +%.lst:: %.cpp force + $(call make-assembly-file,cxx) + + +################################################################################ +### rules: installation targets ################################################ +################################################################################ + + +# Install targets depend on successful exit status of target all because nothing +# must be installed in case of a build error. + + +# -p = preserve time stamps +# -m = set permission mode (as in chmod) +# -d = create all components of specified directories +INSTALL = install +INSTALL_PROGRAM := $(INSTALL) -p -m 644 +INSTALL_DATA := $(INSTALL) -p -m 644 +INSTALL_DIR := $(INSTALL) -m 755 -d + +# strip spaces from file names +executables := $(strip $(executables)) +datafiles := $(strip $(datafiles)) +datadirs := $(strip $(datadirs)) + +# Do not make any install sub-target with empty variable definition because the +# install program would exit with an error. +install: $(if $(executables), install-executables) +install: $(if $(datafiles), install-datafiles) +install: $(if $(datadirs), install-datadirs) + +install-executables: all + $(INSTALL_DIR) -v "$(installpath)" + $(INSTALL_PROGRAM) $(executables) "$(installpath)" + $(info ++++ info: executables of lib $(lib.name) installed \ + from $(CURDIR) to $(installpath)) + +install-datafiles: all + $(INSTALL_DIR) -v "$(installpath)" + $(INSTALL_DATA) $(datafiles) "$(installpath)" + $(info ++++ info: data files of lib $(lib.name) installed \ + from $(CURDIR) to $(installpath)) + +install-datadirs: all + $(foreach v, $(datadirs), $(INSTALL_DIR) "$(installpath)/$v";) + $(foreach v, $(datadirs), \ + $(INSTALL_DATA) $(wildcard $v/*) "$(installpath)/$v";) + $(info ++++ info: data directories of lib $(lib.name) installed \ + from $(CURDIR) to $(installpath)) + + +################################################################################ +### rules: distribution targets ################################################ +################################################################################ + + +# TODO +# These targets are implemented in Makefile Template, but I have to figure out +# how to do it under the not-so-strict conditions of Makefile.pdlibbuilder. + +# make source package +dist: + @echo "target dist not yet implemented" + +# make Debian source package +dpkg-source: + @echo "target dpkg-source not yet implemented" + +$(ORIGDIR): + +$(DISTDIR): + + +################################################################################ +### rules: clean targets ####################################################### +################################################################################ + + +# delete build products from build tree +clean: + rm -f $(all.objects) + rm -f $(classes.executables) $(lib.name).$(extension) $(shared.lib) + rm -f *.pre *.lst + +# remove distribution directories and tarballs from build tree +distclean: clean + @echo "target distclean not yet implemented" + + +################################################################################ +### rules: submake targets ##################################################### +################################################################################ + + +# Iterate over sub-makefiles or makefiles in other directories. + +# When 'continue-make=yes' is set, sub-makes will report 'true' to the parent +# process regardless of their real exit status. This prevents the parent make +# from being aborted by a sub-make error. Useful when you want to quickly find +# out which sub-makes from a large set will succeed. +ifeq ($(continue-make),yes) + continue = || true +endif + +# These targets will trigger sub-make processes for entries in 'makefiledirs' +# and 'makefiles'. +all alldebug install clean distclean dist dkpg-source: \ + $(makefiledirs) $(makefiles) + +# this expands to identical rules for each entry in 'makefiledirs' +$(makefiledirs): + $(MAKE) --directory=$@ $(MAKECMDGOALS) $(continue) + +# this expands to identical rules for each entry in 'makefiles' +$(makefiles): + $(MAKE) --directory=$(dir $@) --makefile=$(notdir $@) $(MAKECMDGOALS) $(continue) + + +################################################################################ +### rules: convenience targets ################################################# +################################################################################ + + +#=== show variables ============================================================ + + +# Several 'function' macro's cause errors when expanded within a rule or without +# proper arguments. Variables which are set with the define directive are only +# shown by name for that reason. +functions = \ +add-class-source \ +declare-class-target \ +declare-class-executable-target \ +declare-object-target \ +link-class \ +link-lib \ +link-shared \ +make-object-file \ +make-preprocessor-file \ +make-assembly-file + + +# show variables from makefiles +vars: + $(info ++++ info: showing makefile variables:) + $(foreach v,\ + $(sort $(filter-out $(functions) functions, $(.VARIABLES))),\ + $(if $(filter file, $(origin $v)),\ + $(info variable $v = $($v)))) + $(foreach v, $(functions), $(info 'function' name: $v)) + @echo + +# show all variables +allvars: + $(info ++++ info: showing default, automatic and makefile variables:) + $(foreach v, \ + $(sort $(filter-out $(functions) functions, $(.VARIABLES))), \ + $(info variable ($(origin $v)) $v = $($v))) + $(foreach v, $(functions), $(info 'function' name: $v)) + @echo + + +#=== show dependencies ========================================================= + + +# show generated prerequisites rules +depend: + $(info ++++ info: generated prerequisite rules) + $(foreach v, $(classes), $(info $(declare-class-target))) + $(foreach v, $(classes), $(info $(declare-class-executable-target))) + $(foreach v, $(all.sources), $(info $(call declare-object-target, $v))) + @echo + + +#=== show help text ============================================================ + + +# brief info about targets and paths + +ifdef mpdh + mpdhinfo := $(mpdh) +else + mpdhinfo := m_pd.h was not found. Is Pd(-extended) installed? +endif + +help: + @echo + @echo " Main targets:" + @echo " all: build executables (default target)" + @echo " install: install all components of the library" + @echo " vars: print makefile variables for troubleshooting" + @echo " allvars: print all variables for troubleshooting" + @echo " help: print this help text" + @echo + @echo " Pd API m_pd.h:" + @echo " $(mpdhinfo)" + @echo " You may specify your preferred Pd include path as argument to" + @echo " the make command, like 'pdincludepath=path/to/pd/src'." + @echo + @echo " Path for installation of your libdir(s):" + @echo " $(objectsdir)" + @echo " Alternatively you may specify your path for installation as argument" + @echo " to the make command, like 'objectsdir=path/to/pd-externals'." + @echo + @echo " Default paths are listed in the doc sections in Makefile.pdlibbuilder." + @echo + + +#=== dummy target ============================================================== + + +coffee: + @echo "Makefile.pdlibbuilder: Can not make coffee. Sorry." + + +################################################################################ +### end of rules sections ###################################################### +################################################################################ + + +# for syntax highlighting in vim and github +# vim: set filetype=make: + diff --git a/externals/footils/knob/knob.c b/externals/footils/knob/knob.c index b3c0c2ca10818ee367cfbf3c618d1f45b27ce360..bdb953e032ee15222f3ef2b3190c766c9a899c0e 100644 --- a/externals/footils/knob/knob.c +++ b/externals/footils/knob/knob.c @@ -26,7 +26,7 @@ #include "g_canvas.h" -#include "../../old_g_all_guis.inc" +#include "g_all_guis.h" #include <math.h> #ifdef WIN32 @@ -47,13 +47,14 @@ t_widgetbehavior knob_widgetbehavior; static t_class *knob_class; t_symbol *iemgui_key_sym=0; /* taken from g_all_guis.c */ -typedef struct _knob /* taken from Frank's modyfied g_all_guis.h */ +typedef struct _knob /* taken from Frank's modified g_all_guis.h */ { t_iemgui x_gui; int x_pos; int x_val; int x_lin0_log1; int x_steady; + int x_explained_surprising_behavior; double x_min; double x_max; double x_k; @@ -63,34 +64,39 @@ typedef struct _knob /* taken from Frank's modyfied g_all_guis.h */ static void knob_draw_update(t_knob *x, t_glist *glist) { - if (glist_isvisible(glist)) - { + /* All this complicated data and no normalized value stored anywhere! + So we roll our own here for the sake of sanity... */ + double normalized_value = (double)(((x->x_val)*0.01*x->x_k + x->x_min) + / x->x_max); /* compute dial:*/ float radius = 0.5*(float)x->x_gui.x_h; - double angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*( (double)x->x_val*0.01/(double)x->x_gui.x_h ); - int start = -80; - int extent = 350 - (int)(360.0*angle/(2.0*M_PI)); + double angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*((double)x->x_val*0.01/(double)x->x_gui.x_h ); +// int start = -80; +// int extent = 350 - (int)(360.0*angle/(2.0*M_PI)); /* center point: */ - int x1 = text_xpix(&x->x_gui.x_obj, glist) + radius; - int y1 = text_ypix(&x->x_gui.x_obj, glist) + radius; - int x2 = text_xpix(&x->x_gui.x_obj, glist) + radius + radius * sin( -angle); - int y2 = text_ypix(&x->x_gui.x_obj, glist) + radius + radius * cos( angle); - - sys_vgui(".x%lx.c coords %xKNOB %d %d %d %d\n", - glist_getcanvas(glist), x, - x1, /* x1 */ - y1, /* y1 */ - x2, /* x2 */ - y2 /* y2 */ - ); - + int x1 = radius; + int y1 = radius; + int x2 = radius + radius * sin(-angle); + int y2 = radius + radius * cos(angle); + + gui_vmess("gui_turn_mknob", "xxiiiiif", + glist_getcanvas(glist), + x, + x1, + y1, + x2, + y2, + 1, + normalized_value + ); /* post("knob: (%d, %d) (%d, %d)", x1,y1,x2,y2); */ - sys_vgui(".x%lx.c itemconfigure %xBASE -start %d -extent %d \n", glist_getcanvas(glist), x, - start, extent); - } + //sys_vgui(".x%lx.c itemconfigure %xBASE -start %d -extent %d \n", + // glist_getcanvas(glist), x, + // start, extent); } +static void knob_draw_config(t_knob *x,t_glist *glist); static void knob_draw_new(t_knob *x, t_glist *glist) { int xpos=text_xpix(&x->x_gui.x_obj, glist); @@ -99,130 +105,101 @@ static void knob_draw_new(t_knob *x, t_glist *glist) t_canvas *canvas=glist_getcanvas(glist); /* compute dial:*/ - float radius = 0.5*(float)x->x_gui.x_h; +// float radius = 0.5*(float)x->x_gui.x_h; /* center point: */ - int x1 = xpos + radius; - int y1 = ypos + radius; +// int x1 = xpos + radius; +// int y1 = ypos + radius; /* dial */ - double angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*((float)x->x_val*0.01/(float)(x->x_gui.x_h)); - int x2 = x1 + radius * sin(-angle); - int y2 = y1 + radius * cos(angle); +// double angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*((float)x->x_val*0.01/(float)(x->x_gui.x_h)); +// int x2 = x1 + radius * sin(-angle); +// int y2 = y1 + radius * cos(angle); + /* reuse mknob drawing routine */ + gui_vmess("gui_mknob_new", "xxiiiiii", + canvas, + x, + xpos, + ypos, + glist_istoplevel(glist), + !iemgui_has_snd(&x->x_gui), + !iemgui_has_rcv(&x->x_gui), + 1 + ); + knob_draw_config(x, glist); /* BASE2 */ + /* sys_vgui(".x%lx.c create arc %d %d %d %d -outline #%6.6x -style arc -width 3 -start -80 -extent 340 -tags %xBASE2\n", canvas, - xpos, ypos, /* upper left */ - xpos + x->x_gui.x_h, ypos + x->x_gui.x_h, /* lower right */ + xpos, ypos, + xpos + x->x_gui.x_h, ypos + x->x_gui.x_h, x->x_gui.x_fcol, x); - + */ /* BASE */ + /* sys_vgui(".x%lx.c create arc %d %d %d %d -fill #%6.6x -style arc -width 3 -start -80 -extent 340 -tags %xBASE\n", - canvas, - xpos, ypos, /* upper left */ - xpos + x->x_gui.x_h, ypos + x->x_gui.x_h, /* lower right */ + canvas, + xpos, ypos, + xpos + x->x_gui.x_h, ypos + x->x_gui.x_h, x->x_gui.x_bcol, x); + */ /* LINE */ + /* sys_vgui(".x%lx.c create line %d %d %d %d -width 3 -fill #%6.6x -capstyle round -tags %xKNOB\n", - canvas, - x1, /* x1 */ - y1, /* y1 */ - x2, /* x2 */ - y2, /* y2 */ - x->x_gui.x_fcol, /* color */ - x); - + canvas, + x1, + y1, + x2, + y2, + x->x_gui.x_fcol, + x); + sys_vgui(".x%lx.c create text %d %d -text {%s} -anchor w \ -font {%s %d bold} -fill #%6.6x -tags %xLABEL\n", canvas, xpos+x->x_gui.x_ldx, ypos+x->x_gui.x_ldy, strcmp(x->x_gui.x_lab->s_name, "empty")?x->x_gui.x_lab->s_name:"", - x->x_gui.x_font, x->x_gui.x_fontsize, x->x_gui.x_lcol, x); - if(!x->x_gui.x_fsf.x_snd_able) + iem_fstyletoint(&x->x_gui), x->x_gui.x_fontsize, x->x_gui.x_lcol, + x); + if (!iemgui_has_snd(&x->x_gui)) sys_vgui(".x%lx.c create rectangle %d %d %d %d -tags %xOUT%d\n", canvas, xpos, ypos + x->x_gui.x_h+2, xpos+7, ypos + x->x_gui.x_h+3, x, 0); - if(!x->x_gui.x_fsf.x_rcv_able) + if (!iemgui_has_rcv(&x->x_gui)) sys_vgui(".x%lx.c create rectangle %d %d %d %d -tags %xIN%d\n", canvas, xpos, ypos-2, xpos+7, ypos-1, x, 0); + */ } -static void knob_draw_move(t_knob *x, t_glist *glist) +static void knob_draw_config(t_knob *x,t_glist *glist) { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); - /* int r = ypos + x->x_gui.x_h - (x->x_val + 50)/100; */ - t_canvas *canvas=glist_getcanvas(glist); - /* compute dial:*/ - float radius = 0.5*(float)x->x_gui.x_h; - /* float angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*((float)x->x_val*0.01/(float)(x->x_gui.x_h)); */ - double angle = 7.0/36.0 + 34.0/36.0*2.0*M_PI*( (double)x->x_val*0.01/(double)x->x_gui.x_h ); - /* center point: */ - int x1 = text_xpix(&x->x_gui.x_obj, glist) + radius; - int y1 = text_ypix(&x->x_gui.x_obj, glist) + radius; - int x2 = text_xpix(&x->x_gui.x_obj, glist) + radius + radius * sin( -angle); - int y2 = text_ypix(&x->x_gui.x_obj, glist) + radius + radius * cos( angle); - - - - sys_vgui(".x%lx.c coords %xKNOB %d %d %d %d\n", - canvas, x, - x1, /* x1 */ - y1, /* y1 */ - x2, /* x2 */ - y2 /* y2 */ - ); - /* post("knob: (%d, %d) (%d, %d)", x1,y1,x2,y2); */ - - sys_vgui(".x%lx.c coords %xBASE %d %d %d %d\n", - canvas, x, - xpos, ypos, - xpos + x->x_gui.x_h, ypos + x->x_gui.x_h); - - sys_vgui(".x%lx.c coords %xBASE2 %d %d %d %d\n", - canvas, x, - xpos, ypos, - xpos + x->x_gui.x_h, ypos + x->x_gui.x_h); - - sys_vgui(".x%lx.c coords %xLABEL %d %d\n", - canvas, x, xpos+x->x_gui.x_ldx, ypos+x->x_gui.x_ldy); - if(!x->x_gui.x_fsf.x_snd_able) - sys_vgui(".x%lx.c coords %xOUT%d %d %d %d %d\n", - canvas, x, 0, - xpos, ypos + x->x_gui.x_h+2, - xpos+7, ypos + x->x_gui.x_h+3); - if(!x->x_gui.x_fsf.x_rcv_able) - sys_vgui(".x%lx.c coords %xIN%d %d %d %d %d\n", - canvas, x, 0, - xpos, ypos-2, - xpos+7, ypos-1); -} + t_canvas *canvas = glist_getcanvas(glist); + char bcol[8], fcol[8], lcol[8]; -static void knob_draw_erase(t_knob* x,t_glist* glist) -{ - t_canvas *canvas=glist_getcanvas(glist); + sprintf(bcol, "#%6.6x", x->x_gui.x_bcol); + sprintf(fcol, "#%6.6x", x->x_gui.x_fcol); + sprintf(lcol, "#%6.6x", x->x_gui.x_lcol); - sys_vgui(".x%lx.c delete %xBASE\n", canvas, x); - sys_vgui(".x%lx.c delete %xBASE2\n", canvas, x); - sys_vgui(".x%lx.c delete %xKNOB\n", canvas, x); - sys_vgui(".x%lx.c delete %xLABEL\n", canvas, x); - if(!x->x_gui.x_fsf.x_snd_able) - sys_vgui(".x%lx.c delete %xOUT%d\n", canvas, x, 0); - if(!x->x_gui.x_fsf.x_rcv_able) - sys_vgui(".x%lx.c delete %xIN%d\n", canvas, x, 0); -} - -static void knob_draw_config(t_knob* x,t_glist* glist) -{ - t_canvas *canvas=glist_getcanvas(glist); + /* reuse mknob interface */ + gui_vmess("gui_configure_mknob", "xxissi", + canvas, + x, + x->x_gui.x_h, + bcol, + fcol, + 1 /* flag to tell we are a footils/knob */ + ); + knob_draw_update(x, glist); +/* sys_vgui(".x%lx.c itemconfigure %xLABEL -font {%s %d bold} -fill #%6.6x -text {%s} \n", - canvas, x, x->x_gui.x_font, x->x_gui.x_fontsize, - x->x_gui.x_fsf.x_selected?IEM_GUI_COLOR_SELECTED:x->x_gui.x_lcol, + canvas, x, iem_fstyletoint(&x->x_gui), x->x_gui.x_fontsize, + glist_isselected(canvas, (t_gobj *)x) ? + 0 : x->x_gui.x_lcol, strcmp(x->x_gui.x_lab->s_name, "empty")?x->x_gui.x_lab->s_name:""); sys_vgui(".x%lx.c itemconfigure %xKNOB -fill #%6.6x\n", canvas, x, x->x_gui.x_fcol); @@ -230,68 +207,77 @@ static void knob_draw_config(t_knob* x,t_glist* glist) x, x->x_gui.x_fcol); sys_vgui(".x%lx.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, x->x_gui.x_bcol); +*/ } +/* static void knob_draw_io(t_knob* x,t_glist* glist, int old_snd_rcv_flags) { int xpos=text_xpix(&x->x_gui.x_obj, glist); int ypos=text_ypix(&x->x_gui.x_obj, glist); t_canvas *canvas=glist_getcanvas(glist); - if((old_snd_rcv_flags & IEM_GUI_OLD_SND_FLAG) && !x->x_gui.x_fsf.x_snd_able) + if((old_snd_rcv_flags & IEM_GUI_OLD_SND_FLAG) && !iemgui_has_snd(&x->x_gui)) sys_vgui(".x%lx.c create rectangle %d %d %d %d -tags %xOUT%d\n", canvas, xpos, ypos + x->x_gui.x_h+2, xpos+7, ypos + x->x_gui.x_h+3, x, 0); - if(!(old_snd_rcv_flags & IEM_GUI_OLD_SND_FLAG) && x->x_gui.x_fsf.x_snd_able) + if(!(old_snd_rcv_flags & IEM_GUI_OLD_SND_FLAG) && iemgui_has_rcv(&x->x_gui)) sys_vgui(".x%lx.c delete %xOUT%d\n", canvas, x, 0); - if((old_snd_rcv_flags & IEM_GUI_OLD_RCV_FLAG) && !x->x_gui.x_fsf.x_rcv_able) + if((old_snd_rcv_flags & IEM_GUI_OLD_RCV_FLAG) && !iemgui_has_rcv(&x->x_gui)) sys_vgui(".x%lx.c create rectangle %d %d %d %d -tags %xIN%d\n", canvas, xpos, ypos-2, xpos+7, ypos-1, x, 0); - if(!(old_snd_rcv_flags & IEM_GUI_OLD_RCV_FLAG) && x->x_gui.x_fsf.x_rcv_able) + if(!(old_snd_rcv_flags & IEM_GUI_OLD_RCV_FLAG) && iemgui_has_rcv(&x->x_gui)) sys_vgui(".x%lx.c delete %xIN%d\n", canvas, x, 0); } +*/ +/* static void knob_draw_select(t_knob *x, t_glist *glist) { t_canvas *canvas=glist_getcanvas(glist); - if(x->x_gui.x_fsf.x_selected) + if(glist_isselected(canvas, (t_gobj *)x)) { pd_bind(&x->x_gui.x_obj.ob_pd, iemgui_key_sym); - sys_vgui(".x%lx.c itemconfigure %xBASE2 -outline #%6.6x\n", canvas, x, IEM_GUI_COLOR_SELECTED); - sys_vgui(".x%lx.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, IEM_GUI_COLOR_SELECTED); - sys_vgui(".x%lx.c itemconfigure %xLABEL -fill #%6.6x\n", canvas, x, IEM_GUI_COLOR_SELECTED); + sys_vgui(".x%lx.c itemconfigure %xBASE2 -outline #%6.6x\n", canvas, x, 0); + sys_vgui(".x%lx.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, 0); + sys_vgui(".x%lx.c itemconfigure %xLABEL -fill #%6.6x\n", canvas, x, 0); } else { pd_unbind(&x->x_gui.x_obj.ob_pd, iemgui_key_sym); - sys_vgui(".x%lx.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, IEM_GUI_COLOR_NORMAL); + sys_vgui(".x%lx.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, 0); sys_vgui(".x%lx.c itemconfigure %xBASE2 -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); sys_vgui(".x%lx.c itemconfigure %xLABEL -fill #%6.6x\n", canvas, x, x->x_gui.x_lcol); } } +*/ -void knob_draw(t_knob *x, t_glist *glist, int mode) +static void knob_draw(t_knob *x, t_glist *glist, int mode) { if(mode == IEM_GUI_DRAW_MODE_UPDATE) knob_draw_update(x, glist); else if(mode == IEM_GUI_DRAW_MODE_MOVE) - knob_draw_move(x, glist); + iemgui_base_draw_move(&x->x_gui); else if(mode == IEM_GUI_DRAW_MODE_NEW) knob_draw_new(x, glist); +/* else if(mode == IEM_GUI_DRAW_MODE_SELECT) knob_draw_select(x, glist); else if(mode == IEM_GUI_DRAW_MODE_ERASE) knob_draw_erase(x, glist); +*/ else if(mode == IEM_GUI_DRAW_MODE_CONFIG) knob_draw_config(x, glist); +/* else if(mode >= IEM_GUI_DRAW_MODE_IO) knob_draw_io(x, glist, mode - IEM_GUI_DRAW_MODE_IO); +*/ } /* ------------------------ knob widgetbehaviour----------------------------- */ @@ -311,27 +297,25 @@ static void knob_getrect(t_gobj *z, t_glist *glist, static void knob_save(t_gobj *z, t_binbuf *b) { t_knob *x = (t_knob *)z; - int bflcol[3], *ip1, *ip2; + int bflcol[3]; t_symbol *srl[3]; iemgui_save(&x->x_gui, srl, bflcol); - ip1 = (int *)(&x->x_gui.x_isa); - ip2 = (int *)(&x->x_gui.x_fsf); binbuf_addv(b, "ssiisiiffiisssiiiiiiiii", gensym("#X"),gensym("obj"), (t_int)x->x_gui.x_obj.te_xpix, (t_int)x->x_gui.x_obj.te_ypix, atom_getsymbol(binbuf_getvec(x->x_gui.x_obj.te_binbuf)), x->x_gui.x_h, x->x_gui.x_h, (float)x->x_min, (float)x->x_max, - x->x_lin0_log1, (*ip1)&IEM_INIT_ARGS_ALL, + x->x_lin0_log1, iem_symargstoint(&x->x_gui), srl[0], srl[1], srl[2], x->x_gui.x_ldx, x->x_gui.x_ldy, - (*ip2)&IEM_FSTYLE_FLAGS_ALL, x->x_gui.x_fontsize, + iem_fstyletoint(&x->x_gui), x->x_gui.x_fontsize, bflcol[0], bflcol[1], bflcol[2], x->x_val, x->x_steady); binbuf_addv(b, ";"); } -void knob_check_height(t_knob *x, int h) +static void knob_check_height(t_knob *x, int h) { if(h < IEM_SL_MINSIZE) h = IEM_SL_MINSIZE; @@ -347,7 +331,7 @@ void knob_check_height(t_knob *x, int h) x->x_k = (x->x_max - x->x_min)/(double)(x->x_gui.x_h - 1); } -void knob_check_minmax(t_knob *x, double min, double max) +static void knob_check_minmax(t_knob *x, double min, double max) { if(x->x_lin0_log1) { @@ -367,9 +351,9 @@ void knob_check_minmax(t_knob *x, double min, double max) x->x_min = min; x->x_max = max; if(x->x_min > x->x_max) /* bugfix */ - x->x_gui.x_isa.x_reverse = 1; + x->x_gui.x_reverse = 1; else - x->x_gui.x_isa.x_reverse = 0; + x->x_gui.x_reverse = 0; if(x->x_lin0_log1) x->x_k = log(x->x_max/x->x_min)/(double)(x->x_gui.x_h - 1); else @@ -379,11 +363,43 @@ void knob_check_minmax(t_knob *x, double min, double max) static void knob_properties(t_gobj *z, t_glist *owner) { t_knob *x = (t_knob *)z; - char buf[800]; +// char buf[800]; + char *gfx_tag; t_symbol *srl[3]; iemgui_properties(&x->x_gui, srl); - + gfx_tag = gfxstub_new2(&x->x_gui.x_obj.ob_pd, x); + + gui_start_vmess("gui_iemgui_dialog", "s", gfx_tag); + gui_start_array(); + + gui_s("type"); gui_s("knob"); + /* Since mknob reuses the iemgui dialog code, we just + use the "width" slot for "size" and the "height" slot + for the number of steps and re-label it on the GUI side. */ +// gui_s("width"); gui_i(x->x_gui.x_w); +// gui_s("height"); gui_i(x->x_gui.x_h); + gui_s("size"); gui_i(x->x_gui.x_h); + gui_s("minimum_range"); gui_f(x->x_min); + gui_s("maximum_range"); gui_f(x->x_max); + gui_s("log_scaling"); gui_i(x->x_lin0_log1); + gui_s("init"); gui_i(x->x_gui.x_loadinit); + gui_s("steady_on_click"); gui_i(x->x_steady); + gui_s("send_symbol"); gui_s(srl[0]->s_name); + gui_s("receive_symbol"); gui_s(srl[1]->s_name); + gui_s("label"); gui_s(srl[2]->s_name); + gui_s("x_offset"); gui_i(x->x_gui.x_ldx); + gui_s("y_offset"); gui_i(x->x_gui.x_ldy); + gui_s("font_style"); gui_i(x->x_gui.x_font_style); + gui_s("font_size"); gui_i(x->x_gui.x_fontsize); + gui_s("background_color"); gui_i(0xffffff & x->x_gui.x_bcol); + gui_s("foreground_color"); gui_i(0xffffff & x->x_gui.x_fcol); + gui_s("label_color"); gui_i(0xffffff & x->x_gui.x_lcol); + + gui_end_array(); + gui_end_vmess(); + +/* sprintf(buf, "pdtk_iemgui_dialog %%s KNOB \ --------dimensions(pix)(pix):-------- %d %d NONE: %d %d height: \ -----------output-range:----------- %g left: %g right: %g \ @@ -393,13 +409,14 @@ static void knob_properties(t_gobj *z, t_glist *owner) %d %d \ %d %d %d\n", x->x_gui.x_w, IEM_SL_MINSIZE, x->x_gui.x_h, IEM_GUI_MINSIZE, - x->x_min, x->x_max, 0.0,/*no_schedule*/ - x->x_lin0_log1, x->x_gui.x_isa.x_loadinit, x->x_steady, -1,/*no multi, but iem-characteristic*/ + x->x_min, x->x_max, 0.0, //no_schedule + x->x_lin0_log1, x->x_gui.x_loadinit, x->x_steady, -1,//no multi, but iem-characteristic srl[0]->s_name, srl[1]->s_name, srl[2]->s_name, x->x_gui.x_ldx, x->x_gui.x_ldy, - x->x_gui.x_fsf.x_font_style, x->x_gui.x_fontsize, + iem_fstyletoint(&x->x_gui), x->x_gui.x_fontsize, 0xffffff & x->x_gui.x_bcol, 0xffffff & x->x_gui.x_fcol, 0xffffff & x->x_gui.x_lcol); gfxstub_new(&x->x_gui.x_obj.ob_pd, x, buf); +*/ } static void knob_bang(t_knob *x) @@ -415,15 +432,32 @@ static void knob_bang(t_knob *x) //post("knob_bang -- x_k: %f", x->x_k); outlet_float(x->x_gui.x_obj.ob_outlet, out); - if(x->x_gui.x_fsf.x_snd_able && x->x_gui.x_snd->s_thing) + if(iemgui_has_snd(&x->x_gui) && x->x_gui.x_snd->s_thing) pd_float(x->x_gui.x_snd->s_thing, out); } +static void complain_about_surprising_behavior(t_knob *x) +{ + if (!x->x_explained_surprising_behavior) + { + pd_error(x, "knob: value is coupled to its height. This " + "means its output value can change when you change its size. " + "If you don't need to change the widget's size when running " + "your patch this is probably ok. Otherwise, " + "consider using moonlib/knob instead. " + "(Or perhaps question why audio interfaces are the only UIs " + "which still employ 'knob' widgets.)"); + x->x_explained_surprising_behavior = 1; + } +} + static void knob_dialog(t_knob *x, t_symbol *s, int argc, t_atom *argv) { - t_symbol *srl[3]; + if (atom_getintarg(19, argc, argv)) + canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x); int w = (int)atom_getintarg(0, argc, argv); int h = (int)atom_getintarg(1, argc, argv); + int old_h = x->x_gui.x_h; double min = (double)atom_getfloatarg(2, argc, argv); double max = (double)atom_getfloatarg(3, argc, argv); int lilo = (int)atom_getintarg(4, argc, argv); @@ -436,13 +470,20 @@ static void knob_dialog(t_knob *x, t_symbol *s, int argc, t_atom *argv) x->x_steady = 1; else x->x_steady = 0; - sr_flags = iemgui_dialog(&x->x_gui, srl, argc, argv); + sr_flags = iemgui_dialog(&x->x_gui, argc, argv); x->x_gui.x_h = iemgui_clip_size(w); knob_check_height(x, h); knob_check_minmax(x, min, max); - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_CONFIG); + if (old_h != x->x_gui.x_h) + complain_about_surprising_behavior(x); + iemgui_draw_config(&x->x_gui); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_CONFIG); + iemgui_draw_io(&x->x_gui, sr_flags); +/* (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_IO + sr_flags); - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_MOVE); +*/ +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_MOVE); + iemgui_io_draw_move(&x->x_gui); canvas_fixlinesfor(x->x_gui.x_glist, (t_text*)x); } @@ -450,7 +491,7 @@ static void knob_motion(t_knob *x, t_floatarg dx, t_floatarg dy) { int old = x->x_val; - if(x->x_gui.x_fsf.x_finemoved) + if(x->x_gui.x_finemoved) x->x_pos -= (int)dy; else x->x_pos -= 100*(int)dy; @@ -495,14 +536,14 @@ static int knob_newclick(t_gobj *z, struct _glist *glist, { t_knob* x = (t_knob *)z; - if(doit) + if (doit) { - knob_click( x, (t_floatarg)xpix, (t_floatarg)ypix, (t_floatarg)shift, + knob_click (x, (t_floatarg)xpix, (t_floatarg)ypix, (t_floatarg)shift, 0, (t_floatarg)alt); - if(shift) - x->x_gui.x_fsf.x_finemoved = 1; + if (shift) + x->x_gui.x_finemoved = 1; else - x->x_gui.x_fsf.x_finemoved = 0; + x->x_gui.x_finemoved = 0; } return (1); } @@ -511,21 +552,21 @@ static void knob_set(t_knob *x, t_floatarg f) { double g; - if(x->x_gui.x_isa.x_reverse) /* bugfix */ + if (x->x_gui.x_reverse) /* bugfix */ { - if(f > x->x_min) + if (f > x->x_min) f = x->x_min; - if(f < x->x_max) + if (f < x->x_max) f = x->x_max; } else { - if(f > x->x_max) + if (f > x->x_max) f = x->x_max; - if(f < x->x_min) + if (f < x->x_min) f = x->x_min; } - if(x->x_lin0_log1) + if (x->x_lin0_log1) g = log(f/x->x_min)/x->x_k; else g = (f - x->x_min) / x->x_k; @@ -535,14 +576,15 @@ static void knob_set(t_knob *x, t_floatarg f) // x->x_val = (int)(100.0*g); x->x_pos = x->x_val; //post("knob_set x_val: %f", x->x_val ); - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); + if (glist_isvisible(x->x_gui.x_glist)) + (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); } static void knob_float(t_knob *x, t_floatarg f) { - /* post("knob_set infloat f: %f", f ); */ + /* post("knob_set infloat f: %f", f ); */ knob_set(x, f); - if(x->x_gui.x_fsf.x_put_in2out) + if (x->x_gui.x_put_in2out) knob_bang(x); } @@ -550,15 +592,20 @@ static void knob_size(t_knob *x, t_symbol *s, int ac, t_atom *av) { x->x_gui.x_h = iemgui_clip_size((int)atom_getintarg(0, ac, av)); if(ac > 1) - knob_check_height(x, (int)atom_getintarg(1, ac, av)); - iemgui_size((void *)x, &x->x_gui); + knob_check_height(x, (int)atom_getintarg(1, ac, av)); + complain_about_surprising_behavior(x); + iemgui_size(&x->x_gui); } static void knob_delta(t_knob *x, t_symbol *s, int ac, t_atom *av) -{iemgui_delta((void *)x, &x->x_gui, s, ac, av);} +{ + iemgui_delta(&x->x_gui, s, ac, av); +} static void knob_pos(t_knob *x, t_symbol *s, int ac, t_atom *av) -{iemgui_pos((void *)x, &x->x_gui, s, ac, av);} +{ + iemgui_pos(&x->x_gui, s, ac, av); +} static void knob_range(t_knob *x, t_symbol *s, int ac, t_atom *av) { @@ -567,22 +614,34 @@ static void knob_range(t_knob *x, t_symbol *s, int ac, t_atom *av) } static void knob_color(t_knob *x, t_symbol *s, int ac, t_atom *av) -{iemgui_color((void *)x, &x->x_gui, s, ac, av);} +{ + iemgui_color(&x->x_gui, s, ac, av); +} static void knob_send(t_knob *x, t_symbol *s) -{iemgui_send(x, &x->x_gui, s);} +{ + iemgui_send(&x->x_gui, s); +} static void knob_receive(t_knob *x, t_symbol *s) -{iemgui_receive(x, &x->x_gui, s);} +{ + iemgui_receive(&x->x_gui, s); +} static void knob_label(t_knob *x, t_symbol *s) -{iemgui_label((void *)x, &x->x_gui, s);} +{ + iemgui_label(&x->x_gui, s); +} static void knob_label_pos(t_knob *x, t_symbol *s, int ac, t_atom *av) -{iemgui_label_pos((void *)x, &x->x_gui, s, ac, av);} +{ + iemgui_label_pos(&x->x_gui, s, ac, av); +} static void knob_label_font(t_knob *x, t_symbol *s, int ac, t_atom *av) -{iemgui_label_font((void *)x, &x->x_gui, s, ac, av);} +{ + iemgui_label_font(&x->x_gui, s, ac, av); +} static void knob_log(t_knob *x) { @@ -598,7 +657,7 @@ static void knob_lin(t_knob *x) static void knob_init(t_knob *x, t_floatarg f) { - x->x_gui.x_isa.x_loadinit = (f==0.0)?0:1; + x->x_gui.x_loadinit = (f==0.0)?0:1; } static void knob_steady(t_knob *x, t_floatarg f) @@ -610,7 +669,7 @@ static void knob_steady(t_knob *x, t_floatarg f) static void knob_loadbang(t_knob *x, t_floatarg action) { - if (action == LB_LOAD && x->x_gui.x_isa.x_loadinit) + if (action == LB_LOAD && x->x_gui.x_loadinit) { (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); knob_bang(x); @@ -635,24 +694,101 @@ static void knob_list(t_knob *x, t_symbol *s, int ac, t_atom *av) } */ +/* we may no longer need h_dragon... */ +static void knob__clickhook(t_scalehandle *sh, int newstate) +{ + t_knob *x = (t_knob *)(sh->h_master); + if (newstate) + { + canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x); + if (!sh->h_scale) /* click on a label handle */ + scalehandle_click_label(sh); + } + /* We no longer need this "clickhook", as we can handle the dragging + either in the GUI (for the label handle) or or in canvas_doclick */ + //iemgui__clickhook3(sh,newstate); + sh->h_dragon = newstate; +} + +static void knob__motionhook(t_scalehandle *sh, + t_floatarg mouse_x, t_floatarg mouse_y) +{ + if (sh->h_scale) + { + t_knob *x = (t_knob *)(sh->h_master); + int width = x->x_gui.x_h, + height = x->x_gui.x_h; + int x1, y1, x2, y2, d; + x1 = text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); + y1 = text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); + x2 = x1 + width; + y2 = y1 + height; + + /* This is convoluted, but I can't think of another + way to get this behavior... */ + if (mouse_x <= x2) + { + if (mouse_y > y2) + d = mouse_y - y2; + else if (abs(mouse_y - y2) < abs(mouse_x - x2)) + d = mouse_y - y2; + else + d = mouse_x - x2; + } + else + { + if (mouse_y <= y2) + d = mouse_x - x2; + else + d = maxi(mouse_y - y2, mouse_x - x2); + } + sh->h_dragx = d; + sh->h_dragy = d; + scalehandle_drag_scale(sh); + + width = maxi(width + d, IEM_GUI_MINSIZE); + height = width; + + /* Explain some suprising behavior to the user... */ + + x->x_gui.x_w = width; + x->x_gui.x_h = height; + /* recalculate x_k and stuff */ + knob_check_height(x, height); + knob_check_minmax(x, x->x_min, x->x_max); + + complain_about_surprising_behavior(x); + + if (glist_isvisible(x->x_gui.x_glist)) + { + knob_draw_config(x, x->x_gui.x_glist); + scalehandle_unclick_scale(sh); + } + + int properties = gfxstub_haveproperties((void *)x); + if (properties) + { + int new_w = x->x_gui.x_w + sh->h_dragx; + // This should actually be "size", but we're using the + // "width" input in dialog_iemgui and just relabelling it + // as a kluge. + properties_set_field_int(properties, "width", new_w); + } + } + scalehandle_dragon_label(sh, mouse_x, mouse_y); +} + static void *knob_new(t_symbol *s, int argc, t_atom *argv) { t_knob *x = (t_knob *)pd_new(knob_class); int bflcol[]={-262144, -1, -1}; - t_symbol *srl[3]; int w=IEM_KNOB_DEFAULTSIZE, h=IEM_KNOB_DEFAULTSIZE; int lilo=0, ldx=0, ldy=-8; int fs=8, v=0, steady=1; double min=0.0, max=(double)(IEM_SL_DEFAULTSIZE-1); - char str[144]; - - //srl[0] = gensym("empty"); - //srl[1] = gensym("empty"); - //srl[2] = gensym("empty"); - - iem_inttosymargs(&x->x_gui.x_isa, 0); - iem_inttofstyle(&x->x_gui.x_fsf, 0); + iem_inttosymargs(&x->x_gui, 0); + iem_inttofstyle(&x->x_gui, 0); if(((argc == 17)||(argc == 18))&&IS_A_FLOAT(argv,0)&&IS_A_FLOAT(argv,1) &&IS_A_FLOAT(argv,2)&&IS_A_FLOAT(argv,3) @@ -669,11 +805,11 @@ static void *knob_new(t_symbol *s, int argc, t_atom *argv) min = (double)atom_getfloatarg(2, argc, argv); max = (double)atom_getfloatarg(3, argc, argv); lilo = (int)atom_getintarg(4, argc, argv); - iem_inttosymargs(&x->x_gui.x_isa, atom_getintarg(5, argc, argv)); + iem_inttosymargs(&x->x_gui, atom_getintarg(5, argc, argv)); iemgui_new_getnames(&x->x_gui, 6, argv); ldx = (int)atom_getintarg(9, argc, argv); ldy = (int)atom_getintarg(10, argc, argv); - iem_inttofstyle(&x->x_gui.x_fsf, atom_getintarg(11, argc, argv)); + iem_inttofstyle(&x->x_gui, atom_getintarg(11, argc, argv)); fs = (int)atom_getintarg(12, argc, argv); bflcol[0] = (int)atom_getintarg(13, argc, argv); bflcol[1] = (int)atom_getintarg(14, argc, argv); @@ -681,16 +817,18 @@ static void *knob_new(t_symbol *s, int argc, t_atom *argv) v = (int)atom_getintarg(16, argc, argv); } else iemgui_new_getnames(&x->x_gui, 6, 0); - if((argc == 18)&&IS_A_FLOAT(argv,17)) + if ((argc == 18)&&IS_A_FLOAT(argv,17)) steady = (int)atom_getintarg(17, argc, argv); x->x_gui.x_draw = (t_iemfunptr)knob_draw; +/* x->x_gui.x_fsf.x_snd_able = 1; x->x_gui.x_fsf.x_rcv_able = 1; +*/ x->x_gui.x_glist = (t_glist *)canvas_getcurrent(); - if(x->x_gui.x_isa.x_loadinit) + if (x->x_gui.x_loadinit) x->x_val = v; else x->x_val = 0; @@ -699,19 +837,30 @@ static void *knob_new(t_symbol *s, int argc, t_atom *argv) x->x_lin0_log1 = lilo; if(steady != 0) steady = 1; x->x_steady = steady; +/* if (!strcmp(x->x_gui.x_snd->s_name, "empty")) x->x_gui.x_fsf.x_snd_able = 0; if (!strcmp(x->x_gui.x_rcv->s_name, "empty")) x->x_gui.x_fsf.x_rcv_able = 0; - if(x->x_gui.x_fsf.x_font_style == 1) strcpy(x->x_gui.x_font, "helvetica"); - else if(x->x_gui.x_fsf.x_font_style == 2) strcpy(x->x_gui.x_font, "times"); - else { x->x_gui.x_fsf.x_font_style = 0; - strcpy(x->x_gui.x_font, "courier"); } - if(x->x_gui.x_fsf.x_rcv_able) +*/ + if (iem_fstyletoint(&x->x_gui) == 1) + { + //strcpy(x->x_gui.x_font, "helvetica"); + } + else if (iem_fstyletoint(&x->x_gui) == 2) + { + //strcpy(x->x_gui.x_font, "times"); + } + else + { + x->x_gui.x_font_style = 0; + //strcpy(x->x_gui.x_font, "courier"); + } + if (iemgui_has_rcv(&x->x_gui)) pd_bind(&x->x_gui.x_obj.ob_pd, x->x_gui.x_rcv); x->x_gui.x_ldx = ldx; x->x_gui.x_ldy = ldy; - if(fs < 4) + if (fs < 4) fs = 4; x->x_gui.x_fontsize = fs; x->x_gui.x_h = iemgui_clip_size(h); @@ -721,6 +870,14 @@ static void *knob_new(t_symbol *s, int argc, t_atom *argv) //x->x_thick = 0; iemgui_verify_snd_ne_rcv(&x->x_gui); outlet_new(&x->x_gui.x_obj, &s_float); + x->x_gui.x_obj.te_iemgui = 1; + x->x_explained_surprising_behavior = 0; + + x->x_gui.x_handle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 1, knob__clickhook, knob__motionhook); + x->x_gui.x_lhandle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 0, knob__clickhook, knob__motionhook); + return (x); } @@ -728,9 +885,9 @@ static void *knob_new(t_symbol *s, int argc, t_atom *argv) static void knob_free(t_knob *x) { - if(x->x_gui.x_fsf.x_selected) + if (iemgui_has_snd(&x->x_gui)) pd_unbind(&x->x_gui.x_obj.ob_pd, iemgui_key_sym); - if(x->x_gui.x_fsf.x_rcv_able) + if (iemgui_has_rcv(&x->x_gui)) pd_unbind(&x->x_gui.x_obj.ob_pd, x->x_gui.x_rcv); gfxstub_deleteforkey(x); } @@ -771,6 +928,7 @@ void knob_setup(void) iemgui_key_sym = gensym("#keyname"); knob_widgetbehavior.w_getrectfn = knob_getrect; knob_widgetbehavior.w_displacefn = iemgui_displace; + knob_widgetbehavior.w_displacefnwtag = iemgui_displace_withtag; knob_widgetbehavior.w_selectfn = iemgui_select; knob_widgetbehavior.w_activatefn = NULL; knob_widgetbehavior.w_deletefn = iemgui_delete; diff --git a/externals/footils/knob/makefile b/externals/footils/knob/makefile index 5fdf212db7d7c5c085d4f8b9ba88109f308eb96c..cdb8e075f5c344eff79071fc690d164e03a6b350 100644 --- a/externals/footils/knob/makefile +++ b/externals/footils/knob/makefile @@ -1,91 +1,13 @@ -NAME=knob -CSYM=knob - -# !!! -# change these two -PD_DIR=../../../pd -#current: pd_nt pd_linux -current: pd_linux - -# ----------------------- NT ----------------------- - -pd_nt: $(NAME).dll - -.SUFFIXES: .dll - -PDNTCFLAGS = /W3 /WX /O2 /G6 /DNT /DPD /nologo - -# where is VC++ ??? -VC="C:\Programme\Microsoft Visual Studio\VC98" - -# where is your m_pd.h ??? -PDNTINCLUDE = /I. /Ic:\pd\tcl\include /Ic:\pd\src /I$(VC)\include /Iinclude - -PDNTLDIR = $(VC)\Lib -PDNTLIB = $(PDNTLDIR)\libc.lib \ - $(PDNTLDIR)\oldnames.lib \ - $(PDNTLDIR)\kernel32.lib \ - $(PDNTLDIR)\user32.lib \ - $(PDNTLDIR)\uuid.lib \ - $(PDNTLDIR)\ws2_32.lib \ - c:\pd\bin\pd.lib - -.c.dll: - cl $(PDNTCFLAGS) $(PDNTINCLUDE) /c $*.c - link /dll /export:$(CSYM)_setup $*.obj $(PDNTLIB) - -# ----------------------- IRIX 5.x ----------------------- - -pd_irix5: $(NAME).pd_irix5 - -.SUFFIXES: .pd_irix5 - -SGICFLAGS5 = -o32 -DPD -DUNIX -DIRIX -O2 - -SGIINCLUDE = -I../../src - -.c.pd_irix5: - cc $(SGICFLAGS5) $(SGIINCLUDE) -o $*.o -c $*.c - ld -elf -shared -rdata_shared -o $*.pd_irix5 $*.o - rm $*.o - -# ----------------------- IRIX 6.x ----------------------- - -pd_irix6: $(NAME).pd_irix6 - -.SUFFIXES: .pd_irix6 - -SGICFLAGS6 = -n32 -DPD -DUNIX -DIRIX -DN32 -woff 1080,1064,1185 \ - -OPT:roundoff=3 -OPT:IEEE_arithmetic=3 -OPT:cray_ivdep=true \ - -Ofast=ip32 - -.c.pd_irix6: - cc $(SGICFLAGS6) $(SGIINCLUDE) -o $*.o -c $*.c - ld -n32 -IPA -shared -rdata_shared -o $*.pd_irix6 $*.o - rm $*.o - -# ----------------------- LINUX i386 ----------------------- - -pd_linux: $(NAME).pd_linux - -.SUFFIXES: .pd_linux - -LINUXCFLAGS = -DPD -DUNIX -O2 -funroll-loops -fomit-frame-pointer \ - -Wall -W -Wshadow -Wstrict-prototypes -Werror \ - -Wno-unused -Wno-parentheses -Wno-switch - -# where is your m_pd.h ??? -LINUXINCLUDE = -I../../src -I$(PD_DIR)/src - -.c.pd_linux: - cc -O2 -Wall -DPD -fPIC $(LINUXINCLUDE) -c $*.c - ld --export-dynamic -shared -o $*.pd_linux $*.o -lc - strip $*.pd_linux - -# ---------------------------------------------------------- - -install: - cp help-*.pd ../../doc/5.reference - -clean: - rm -f *.o *.pd_* so_locations +# Makefile for mylib + +lib.name = flatgui + +class.sources = knob.c + +# ldlibs = -lfluidsynth + +datafiles = knob-help.pd clock.pd README + +externalsdir = ../../ + +include Makefile.pdlibbuilder.revised diff --git a/externals/iem/iemgui/src/room_sim_2d.c b/externals/iem/iemgui/src/room_sim_2d.c index fff669083b63bee23b8f27b0f79d1b2ccaa05730..7f57c69296527284c511a119488afb5a8d2798ec 100644 --- a/externals/iem/iemgui/src/room_sim_2d.c +++ b/externals/iem/iemgui/src/room_sim_2d.c @@ -7,7 +7,7 @@ iemgui written by Thomas Musil, Copyright (c) IEM KUG Graz Austria 2000 - 2006 * #include "iemlib.h" #include "iemgui.h" #include "g_canvas.h" -#include "../../../old_g_all_guis.inc" +#include "g_all_guis.h" #include <stdio.h> #include <stdlib.h> @@ -60,72 +60,163 @@ static void room_sim_2d_out_rho(t_room_sim_2d *x) static void room_sim_2d_out_para(t_room_sim_2d *x) { int i, n = x->x_nr_src; - int w2=x->x_gui.x_w/2, h2=x->x_gui.x_h/2; - + int w2 = x->x_gui.x_w / 2, + h2 = x->x_gui.x_h/2; + SETFLOAT(x->x_at, 0.0f); SETSYMBOL(x->x_at+1, x->x_s_head_xy); SETFLOAT(x->x_at+2, (t_float)(h2 - x->x_pix_src_y[0])/x->x_cnvrt_roomlx2pixh); SETFLOAT(x->x_at+3, (t_float)(w2 - x->x_pix_src_x[0])/x->x_cnvrt_roomlx2pixh); outlet_list(x->x_out_para, &s_list, 4, x->x_at); - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { SETFLOAT(x->x_at, (t_float)i); SETSYMBOL(x->x_at+1, x->x_s_src_xy); - SETFLOAT(x->x_at+2, (t_float)(h2 - x->x_pix_src_y[i])/x->x_cnvrt_roomlx2pixh); - SETFLOAT(x->x_at+3, (t_float)(w2 - x->x_pix_src_x[i])/x->x_cnvrt_roomlx2pixh); + SETFLOAT(x->x_at+2, + (t_float)(h2 - x->x_pix_src_y[i]) / x->x_cnvrt_roomlx2pixh); + SETFLOAT(x->x_at+3, + (t_float)(w2 - x->x_pix_src_x[i]) / x->x_cnvrt_roomlx2pixh); outlet_list(x->x_out_para, &s_list, 4, x->x_at); } } static void room_sim_2d_draw_update(t_room_sim_2d *x, t_glist *glist) { - if(glist_isvisible(glist)) + if (glist_isvisible(glist)) { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); + int xpos = text_xpix(&x->x_gui.x_obj, glist); + int ypos = text_ypix(&x->x_gui.x_obj, glist); int dx, dy; t_canvas *canvas=glist_getcanvas(glist); - dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); + dx = -(int)((t_float)x->x_pix_rad * (t_float)sin(x->x_rho_head * + 0.0174533f) + 0.49999f); + dy = -(int)((t_float)x->x_pix_rad * (t_float)cos(x->x_rho_head * + 0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); + + gui_vmess("gui_room_sim_update", "xxiiiii", + canvas, + x, + x->x_pix_src_x[0], + x->x_pix_src_y[0], + dx, + dy, + x->x_pix_rad + ); } } -void room_sim_2d_draw_new(t_room_sim_2d *x, t_glist *glist) +void room_sim_2d_draw_unmap(t_room_sim_2d *x, t_glist *glist) { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); + gui_vmess("gui_room_sim_erase", "xx", + x->x_gui.x_glist, x); +} + +void room_sim_2d_draw_map(t_room_sim_2d *x, t_glist *glist) +{ + int xpos = text_xpix(&x->x_gui.x_obj, glist); + int ypos = text_ypix(&x->x_gui.x_obj, glist); int dx, dy; - int i, n=x->x_nr_src; - int fs=x->x_fontsize; + int i, n = x->x_nr_src; + int fs = x->x_fontsize; t_canvas *canvas=glist_getcanvas(glist); - - sys_vgui(".x%x.c create rectangle %d %d %d %d -fill #%6.6x -outline #%6.6x -tags %xBASE\n", - canvas, xpos, ypos, xpos + x->x_gui.x_w, ypos + x->x_gui.x_h, - x->x_gui.x_bcol, x->x_gui.x_fsf.x_selected?IEM_GUI_COLOR_SELECTED:IEM_GUI_COLOR_NORMAL, x); - for(i=1; i<=n; i++) + char fcol[MAXPDSTRING]; + char bcol[MAXPDSTRING]; + char elemcol[MAXPDSTRING]; + sprintf(fcol, "#%6.6x", x->x_gui.x_fcol); + sprintf(bcol, "#%6.6x", x->x_gui.x_bcol); + + gui_start_vmess("gui_room_sim_map", "xxiiifiiiss", + canvas, + x, + x->x_gui.x_w, + x->x_gui.x_h, + x->x_pix_rad, + x->x_rho_head, + x->x_pix_src_x[0], + x->x_pix_src_y[0], + x->x_fontsize, + fcol, + bcol + ); + gui_start_array(); + for(i = 1; i <= n; i++) { - sys_vgui(".x%x.c create text %d %d -text {%d} -anchor c \ - -font {times %d bold} -fill #%6.6x -tags %xSRC%d\n", - canvas, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i], i, fs, - x->x_col_src[i], x, i); +// sys_vgui(".x%x.c create text %d %d -text {%d} -anchor c \ +// -font {times %d bold} -fill #%6.6x -tags %xSRC%d\n", +// canvas, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i], i, fs, +// x->x_col_src[i], x, i); + gui_start_array(); + gui_i(x->x_pix_src_x[i]); + gui_i(x->x_pix_src_y[i]); + sprintf(elemcol, "#%6.6x", x->x_col_src[i]); + gui_s(elemcol); + gui_end_array(); } + gui_end_array(); + gui_end_vmess(); - sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD\n", - canvas, xpos+x->x_pix_src_x[0]-x->x_pix_rad, ypos+x->x_pix_src_y[0]-x->x_pix_rad, - xpos+x->x_pix_src_x[0]+x->x_pix_rad-1, ypos+x->x_pix_src_y[0]+x->x_pix_rad-1, - x->x_gui.x_fcol, x); +// sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD\n", +// canvas, xpos+x->x_pix_src_x[0]-x->x_pix_rad, ypos+x->x_pix_src_y[0]-x->x_pix_rad, +// xpos+x->x_pix_src_x[0]+x->x_pix_rad-1, ypos+x->x_pix_src_y[0]+x->x_pix_rad-1, +// x->x_gui.x_fcol, x); dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c create line %d %d %d %d -width 3 -fill #%6.6x -tags %xNOSE\n", - canvas, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy, - x->x_gui.x_fcol, x); +// sys_vgui(".x%x.c create line %d %d %d %d -width 3 -fill #%6.6x -tags %xNOSE\n", +// canvas, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy, +// x->x_gui.x_fcol, x); +} + +void room_sim_2d_draw_new(t_room_sim_2d *x, t_glist *glist) +{ + int xpos = text_xpix(&x->x_gui.x_obj, glist); + int ypos = text_ypix(&x->x_gui.x_obj, glist); + int dx, dy; + int i, n = x->x_nr_src; + int fs = x->x_fontsize; + t_canvas *canvas=glist_getcanvas(glist); + + gui_vmess("gui_room_sim_new", "xxiiiii", + canvas, + x, + xpos, + ypos, + x->x_gui.x_w, + x->x_gui.x_h, + glist_istoplevel(glist) + ); + + room_sim_2d_draw_map(x, glist); +/* sys_vgui(".x%x.c create rectangle %d %d %d %d -fill #%6.6x -outline #%6.6x -tags %xBASE\n", + canvas, xpos, ypos, xpos + x->x_gui.x_w, ypos + x->x_gui.x_h, + x->x_gui.x_bcol, x->x_gui.x_fsf.x_selected?IEM_GUI_COLOR_SELECTED:IEM_GUI_COLOR_NORMAL, x); +*/ + +// for(i=1; i<=n; i++) +// { +// sys_vgui(".x%x.c create text %d %d -text {%d} -anchor c \ +// -font {times %d bold} -fill #%6.6x -tags %xSRC%d\n", +// canvas, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i], i, fs, +// x->x_col_src[i], x, i); +// } + +// sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD\n", +// canvas, xpos+x->x_pix_src_x[0]-x->x_pix_rad, ypos+x->x_pix_src_y[0]-x->x_pix_rad, +// xpos+x->x_pix_src_x[0]+x->x_pix_rad-1, ypos+x->x_pix_src_y[0]+x->x_pix_rad-1, +// x->x_gui.x_fcol, x); +// dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c create line %d %d %d %d -width 3 -fill #%6.6x -tags %xNOSE\n", +// canvas, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy, +// x->x_gui.x_fcol, x); } +/* void room_sim_2d_draw_move(t_room_sim_2d *x, t_glist *glist) { int xpos=text_xpix(&x->x_gui.x_obj, glist); @@ -152,51 +243,22 @@ void room_sim_2d_draw_move(t_room_sim_2d *x, t_glist *glist) canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); } - -void room_sim_2d_draw_erase(t_room_sim_2d* x, t_glist* glist) -{ - int i, n; - t_canvas *canvas=glist_getcanvas(glist); - - sys_vgui(".x%x.c delete %xBASE\n", canvas, x); - n = x->x_nr_src; - for(i=1; i<=n; i++) - { - sys_vgui(".x%x.c delete %xSRC%d\n", canvas, x, i); - } - sys_vgui(".x%x.c delete %xHEAD\n", canvas, x); - sys_vgui(".x%x.c delete %xNOSE\n", canvas, x); -} - -void room_sim_2d_draw_select(t_room_sim_2d* x, t_glist* glist) -{ - t_canvas *canvas=glist_getcanvas(glist); - - if(x->x_gui.x_fsf.x_selected) - { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); - - sys_vgui(".x%x.c itemconfigure %xBASE -outline #%6.6x\n", canvas, x, IEM_GUI_COLOR_SELECTED); - } - else - { - sys_vgui(".x%x.c itemconfigure %xBASE -outline #%6.6x\n", canvas, x, IEM_GUI_COLOR_NORMAL); - } -} +*/ void room_sim_2d_draw(t_room_sim_2d *x, t_glist *glist, int mode) { - if(mode == IEM_GUI_DRAW_MODE_UPDATE) + if (mode == IEM_GUI_DRAW_MODE_UPDATE) room_sim_2d_draw_update(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_MOVE) - room_sim_2d_draw_move(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_NEW) + else if (mode == IEM_GUI_DRAW_MODE_MOVE) + iemgui_base_draw_move(&x->x_gui); + else if (mode == IEM_GUI_DRAW_MODE_NEW) room_sim_2d_draw_new(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_SELECT) +/* + else if (mode == IEM_GUI_DRAW_MODE_SELECT) room_sim_2d_draw_select(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_ERASE) + else if (mode == IEM_GUI_DRAW_MODE_ERASE) room_sim_2d_draw_erase(x, glist); +*/ } /* ------------------------ cnv widgetbehaviour----------------------------- */ @@ -238,14 +300,14 @@ static void room_sim_2d_save(t_gobj *z, t_binbuf *b) static void room_sim_2d_motion(t_room_sim_2d *x, t_floatarg dx, t_floatarg dy) { - int sel=x->x_sel_index; - int pixrad=x->x_pix_rad; - int xpos=text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); - int ypos=text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); - int ddx, ddy; + int sel = x->x_sel_index; + int pixrad = x->x_pix_rad; + int xpos = text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); + int ypos = text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); +// int ddx, ddy; t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); - if(x->x_gui.x_fsf.x_finemoved && (sel == 0)) + if (x->x_gui.x_finemoved && (sel == 0)) { x->x_rho_head -= dy; @@ -256,47 +318,59 @@ static void room_sim_2d_motion(t_room_sim_2d *x, t_floatarg dx, t_floatarg dy) room_sim_2d_out_rho(x); (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); } - else if(sel == 0) + else if (sel == 0) { x->x_pos_x += (int)dx; x->x_pos_y += (int)dy; x->x_pix_src_x[0] = x->x_pos_x; x->x_pix_src_y[0] = x->x_pos_y; - if(x->x_pix_src_x[0] < 0) + if (x->x_pix_src_x[0] < 0) x->x_pix_src_x[0] = 0; - if(x->x_pix_src_x[0] > x->x_gui.x_w) + if (x->x_pix_src_x[0] > x->x_gui.x_w) x->x_pix_src_x[0] = x->x_gui.x_w; - if(x->x_pix_src_y[0] < 0) + if (x->x_pix_src_y[0] < 0) x->x_pix_src_y[0] = 0; - if(x->x_pix_src_y[0] > x->x_gui.x_h) + if (x->x_pix_src_y[0] > x->x_gui.x_h) x->x_pix_src_y[0] = x->x_gui.x_h; room_sim_2d_out_para(x); - sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, - xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); - ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); +// sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, +// xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); +// ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); + (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); } else { + char col[MAXPDSTRING]; x->x_pos_x += (int)dx; x->x_pos_y += (int)dy; x->x_pix_src_x[sel] = x->x_pos_x; x->x_pix_src_y[sel] = x->x_pos_y; - if(x->x_pix_src_x[sel] < 0) + if (x->x_pix_src_x[sel] < 0) x->x_pix_src_x[sel] = 0; - if(x->x_pix_src_x[sel] > x->x_gui.x_w) + if (x->x_pix_src_x[sel] > x->x_gui.x_w) x->x_pix_src_x[sel] = x->x_gui.x_w; - if(x->x_pix_src_y[sel] < 0) + if (x->x_pix_src_y[sel] < 0) x->x_pix_src_y[sel] = 0; - if(x->x_pix_src_y[sel] > x->x_gui.x_h) + if (x->x_pix_src_y[sel] > x->x_gui.x_h) x->x_pix_src_y[sel] = x->x_gui.x_h; room_sim_2d_out_para(x); - sys_vgui(".x%x.c coords %xSRC%d %d %d\n", - canvas, x, sel, xpos+x->x_pix_src_x[sel], ypos+x->x_pix_src_y[sel]); +// sys_vgui(".x%x.c coords %xSRC%d %d %d\n", +// canvas, x, sel, xpos+x->x_pix_src_x[sel], ypos+x->x_pix_src_y[sel]); + sprintf(col, "#%6.6x", x->x_col_src[sel]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + sel - 1, + x->x_pix_src_x[sel], + x->x_pix_src_y[sel], + x->x_fontsize, + col + ); } } @@ -305,81 +379,88 @@ static void room_sim_2d_click(t_room_sim_2d *x, t_floatarg xpos, t_floatarg ypos { int w = (int)xpos - text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); int h = (int)ypos - text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); - int i, n=x->x_nr_src; - int pixrad=x->x_pix_rad; - int fsi=x->x_fontsize; - int diff, maxdiff=10000, sel=-1; + int i, n = x->x_nr_src; + int pixrad = x->x_pix_rad; + int fsi = x->x_fontsize; + int diff, maxdiff = 10000, sel =- 1; i = 0;/* head */ - if((w >= (x->x_pix_src_x[i]-pixrad)) && (w <= (x->x_pix_src_x[i]+pixrad)) && (h >= (x->x_pix_src_y[i]-pixrad)) && (h <= (x->x_pix_src_y[i]+pixrad))) + if ((w >= (x->x_pix_src_x[i] - pixrad)) && + (w <= (x->x_pix_src_x[i] + pixrad)) && + (h >= (x->x_pix_src_y[i] - pixrad)) && + (h <= (x->x_pix_src_y[i] + pixrad))) { diff = w - x->x_pix_src_x[i]; - if(diff < 0) + if (diff < 0) diff *= -1; - if(diff < maxdiff) + if (diff < maxdiff) { maxdiff = diff; sel = i; } diff = h - x->x_pix_src_y[i]; - if(diff < 0) + if (diff < 0) diff *= -1; - if(diff < maxdiff) + if (diff < maxdiff) { maxdiff = diff; sel = i; } } - fsi *= 2; fsi /= 3; - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { - if((w >= (x->x_pix_src_x[i]-fsi)) && (w <= (x->x_pix_src_x[i]+fsi)) && - (h >= (x->x_pix_src_y[i]-fsi)) && (h <= (x->x_pix_src_y[i]+fsi))) + if ((w >= (x->x_pix_src_x[i] - fsi)) && + (w <= (x->x_pix_src_x[i] + fsi)) && + (h >= (x->x_pix_src_y[i] - fsi)) && + (h <= (x->x_pix_src_y[i] + fsi))) { diff = w - x->x_pix_src_x[i]; - if(diff < 0) + if (diff < 0) diff *= -1; - if(diff < maxdiff) + if (diff < maxdiff) { maxdiff = diff; sel = i; } diff = h - x->x_pix_src_y[i]; - if(diff < 0) + if (diff < 0) diff *= -1; - if(diff < maxdiff) + if (diff < maxdiff) { maxdiff = diff; sel = i; } } } - if(sel >= 0) + if (sel >= 0) { x->x_sel_index = sel; x->x_pos_x = x->x_pix_src_x[sel]; x->x_pos_y = x->x_pix_src_y[sel]; - glist_grab(x->x_gui.x_glist, &x->x_gui.x_obj.te_g, (t_glistmotionfn)room_sim_2d_motion, 0, xpos, ypos); + glist_grab(x->x_gui.x_glist, &x->x_gui.x_obj.te_g, + (t_glistmotionfn)room_sim_2d_motion, 0, xpos, ypos); } } -static int room_sim_2d_newclick(t_gobj *z, struct _glist *glist, int xpix, int ypix, int shift, int alt, int dbl, int doit) +static int room_sim_2d_newclick(t_gobj *z, struct _glist *glist, int xpix, + int ypix, int shift, int alt, int dbl, int doit) { t_room_sim_2d* x = (t_room_sim_2d *)z; - if(doit) + if (doit) { - room_sim_2d_click( x, (t_floatarg)xpix, (t_floatarg)ypix, (t_floatarg)shift, 0, (t_floatarg)alt); - if(shift) + room_sim_2d_click(x, (t_floatarg)xpix, (t_floatarg)ypix, (t_floatarg)shift, + 0, (t_floatarg)alt); + if (shift) { - x->x_gui.x_fsf.x_finemoved = 1; + x->x_gui.x_finemoved = 1; room_sim_2d_out_rho(x); } else { - x->x_gui.x_fsf.x_finemoved = 0; + x->x_gui.x_finemoved = 0; room_sim_2d_out_para(x); } } @@ -394,27 +475,29 @@ static void room_sim_2d_bang(t_room_sim_2d *x) static void room_sim_2d_src_font(t_room_sim_2d *x, t_floatarg ff) { - int fs=(int)(ff + 0.49999f); - int i, n=x->x_nr_src; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + int fs = (int)(ff + 0.49999f); + int i, n = x->x_nr_src; + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if(fs < 8) + if (fs < 8) fs = 8; - if(fs > 250) + if (fs > 250) fs = 250; x->x_fontsize = fs; - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { - sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fs); +// sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fs); + gui_vmess("gui_room_sim_fontsize", "xxii", + canvas, x, i, fs); } } static void room_sim_2d_set_rho(t_room_sim_2d *x, t_floatarg rho) { - while(rho <= -180.0f) + while (rho <= -180.0f) rho += 360.0f; - while(rho > 180.0f) + while (rho > 180.0f) rho -= 360.0f; x->x_rho_head = rho; (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); @@ -428,38 +511,50 @@ static void room_sim_2d_rho(t_room_sim_2d *x, t_floatarg rho) static void room_sim_2d_set_src_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_atom *argv) { + char col[MAXPDSTRING]; t_float xsrc, ysrc; t_float roomx2=0.5f*x->x_room_x, roomy2=0.5f*x->x_room_y; - int i, n=x->x_nr_src; - int xpos=text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); - int ypos=text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); - int w2=x->x_gui.x_w/2, h2=x->x_gui.x_h/2; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + int i, n = x->x_nr_src; + int xpos = text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); + int ypos = text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); + int w2 = x->x_gui.x_w/2, h2 = x->x_gui.x_h/2; + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if(argc < 3) + if (argc < 3) { - post("room_sim_2d ERROR: src_xy-input needs 1 index + 2 float-dimensions: src_index, x [m], y [m]"); + post("room_sim_2d ERROR: src_xy-input needs 1 index + 2 float-dimensions: " + "src_index, x [m], y [m]"); return; } i = (int)atom_getint(argv++); - if((i > 0)&&(i <= n)) + if ((i > 0) && (i <= n)) { ysrc = atom_getfloat(argv++); xsrc = atom_getfloat(argv); - if(xsrc < -roomy2) + if (xsrc < -roomy2) xsrc = -roomy2; - if(xsrc > roomy2) + if (xsrc > roomy2) xsrc = roomy2; - if(ysrc < -roomx2) + if (ysrc < -roomx2) ysrc = -roomx2; - if(ysrc > roomx2) + if (ysrc > roomx2) ysrc = roomx2; x->x_pix_src_x[i] = w2 - (int)(x->x_cnvrt_roomlx2pixh * xsrc + 0.49999f); x->x_pix_src_y[i] = h2 - (int)(x->x_cnvrt_roomlx2pixh * ysrc + 0.49999f); - sys_vgui(".x%x.c coords %xSRC%d %d %d\n", - canvas, x, i, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i]); +// sys_vgui(".x%x.c coords %xSRC%d %d %d\n", +// canvas, x, i, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i]); + sprintf(col, "#%6.6x", x->x_col_src[i]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + i - 1, + x->x_pix_src_x[i], + x->x_pix_src_y[i], + x->x_fontsize, + col + ); } } @@ -469,18 +564,20 @@ static void room_sim_2d_src_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_atom * room_sim_2d_out_para(x); } -static void room_sim_2d_set_head_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_atom *argv) +static void room_sim_2d_set_head_xy(t_room_sim_2d *x, t_symbol *s, int argc, + t_atom *argv) { - int pixrad=x->x_pix_rad; - int xpos=text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); - int ypos=text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); - int w2=x->x_gui.x_w/2, h2=x->x_gui.x_h/2; + int pixrad = x->x_pix_rad; + int xpos = text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); + int ypos = text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); + int w2 = x->x_gui.x_w / 2, h2=x->x_gui.x_h / 2; int ddx, ddy; t_float xh, yh; - t_float roomx2=0.5f*x->x_room_x, roomy2=0.5f*x->x_room_y; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + t_float roomx2 = 0.5f * x->x_room_x, + roomy2 = 0.5f * x->x_room_y; + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if(argc < 2) + if (argc < 2) { post("room_sim_2d ERROR: head_xy-input needs 2 float-dimensions: x [m], y [m]"); return; @@ -488,25 +585,26 @@ static void room_sim_2d_set_head_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_a yh = atom_getfloat(argv++); xh = atom_getfloat(argv); - if(xh < -roomy2) + if (xh < -roomy2) xh = -roomy2; - if(xh > roomy2) + if (xh > roomy2) xh = roomy2; - if(yh < -roomx2) + if (yh < -roomx2) yh = -roomx2; - if(yh > roomx2) + if (yh > roomx2) yh = roomx2; x->x_pix_src_x[0] = w2 - (int)(x->x_cnvrt_roomlx2pixh * xh + 0.49999f); x->x_pix_src_y[0] = h2 - (int)(x->x_cnvrt_roomlx2pixh * yh + 0.49999f); - sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, - xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); - ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); +// sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, +// xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); +// ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); + (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); } static void room_sim_2d_head_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_atom *argv) @@ -517,9 +615,9 @@ static void room_sim_2d_head_xy(t_room_sim_2d *x, t_symbol *s, int argc, t_atom static void room_sim_2d_room_dim(t_room_sim_2d *x, t_symbol *s, int argc, t_atom *argv) { - int i, n=x->x_nr_src; + int i, n = x->x_nr_src; - if(argc < 2) + if (argc < 2) { post("room_sim_2d ERROR: room_dim-input needs 2 float-dimensions: x-Length [m], y-Width [m]"); return; @@ -527,19 +625,19 @@ static void room_sim_2d_room_dim(t_room_sim_2d *x, t_symbol *s, int argc, t_atom x->x_room_x = atom_getfloat(argv++); x->x_room_y = atom_getfloat(argv); - if(x->x_room_x < 1.0f) + if (x->x_room_x < 1.0f) x->x_room_x = 1.0f; - if(x->x_room_y < 1.0f) + if (x->x_room_y < 1.0f) x->x_room_y = 1.0f; x->x_gui.x_h = (int)(x->x_cnvrt_roomlx2pixh * (t_float)x->x_room_x + 0.49999f); x->x_gui.x_w = (int)(x->x_cnvrt_roomlx2pixh * (t_float)x->x_room_y + 0.49999f); - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { - if(x->x_pix_src_x[i] > x->x_gui.x_w) + if (x->x_pix_src_x[i] > x->x_gui.x_w) x->x_pix_src_x[i] = x->x_gui.x_w; - if(x->x_pix_src_y[i] > x->x_gui.x_h) + if (x->x_pix_src_y[i] > x->x_gui.x_h) x->x_pix_src_y[i] = x->x_gui.x_h; } @@ -550,86 +648,123 @@ static void room_sim_2d_room_dim(t_room_sim_2d *x, t_symbol *s, int argc, t_atom /*static void room_sim_2d_n_src(t_room_sim_2d *x, t_floatarg fnsrc) { -int n_src=(int)fnsrc; + int n_src = (int)fnsrc; - if(n_src < 1) + if (n_src < 1) n_src = 1; - if(n_src > 30) + if (n_src > 30) n_src = 30; - if(n_src != x->x_nr_src) + if (n_src != x->x_nr_src) { - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); + room_sim_2d_draw_unmap(x, x->x_gui.x_glist); x->x_nr_src = n_src; - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); - } +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); + room_sim_2d_draw_map(x, x->x_gui.x_glist); } +} */ static void room_sim_2d_room_col(t_room_sim_2d *x, t_floatarg fcol) { - int col=(int)fcol; + int col = (int)fcol; + char fgstring[MAXPDSTRING]; + char bgstring[MAXPDSTRING]; int i; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if(col < 0) + if (col < 0) { i = -1 - col; x->x_gui.x_bcol = ((i & 0x3f000) << 6)|((i & 0xfc0) << 4)|((i & 0x3f) << 2); } else { - if(col > 29) + if (col > 29) col = 29; x->x_gui.x_bcol = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, x->x_gui.x_bcol); + sprintf(fgstring, "#%6.6x", x->x_gui.x_fcol); + sprintf(bgstring, "#%6.6x", x->x_gui.x_bcol); + gui_vmess("gui_room_sim_colors", "xxss", + canvas, + x, + fgstring, + bgstring + ); +// sys_vgui(".x%x.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, x->x_gui.x_bcol); } static void room_sim_2d_head_col(t_room_sim_2d *x, t_floatarg fcol) { - int col=(int)fcol; + int col = (int)fcol; + char fgstring[MAXPDSTRING]; + char bgstring[MAXPDSTRING]; int i; t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); - if(col < 0) + if (col < 0) { i = -1 - col; x->x_gui.x_fcol = ((i & 0x3f000) << 6)|((i & 0xfc0) << 4)|((i & 0x3f) << 2); } else { - if(col > 29) + if (col > 29) col = 29; x->x_gui.x_fcol = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xHEAD -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); - sys_vgui(".x%x.c itemconfigure %xNOSE -fill #%6.6x\n", canvas, x, x->x_gui.x_fcol); + sprintf(fgstring, "#%6.6x", x->x_gui.x_fcol); + sprintf(bgstring, "#%6.6x", x->x_gui.x_bcol); + gui_vmess("gui_room_sim_colors", "xxss", + canvas, + x, + fgstring, + bgstring + ); + //sys_vgui(".x%x.c itemconfigure %xHEAD -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); + //sys_vgui(".x%x.c itemconfigure %xNOSE -fill #%6.6x\n", canvas, x, x->x_gui.x_fcol); } static void room_sim_2d_src_col(t_room_sim_2d *x, t_symbol *s, int argc, t_atom *argv) { int col; - int i, j, n=x->x_nr_src; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + char colstring[MAXPDSTRING]; + int i, j, n = x->x_nr_src; + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if((argc >= 2)&&IS_A_FLOAT(argv,0)&&IS_A_FLOAT(argv,1)) + if ((argc >= 2) && IS_A_FLOAT(argv,0) && IS_A_FLOAT(argv,1)) { j = (int)atom_getintarg(0, argc, argv); - if((j > 0)&&(j <= n)) + if ((j > 0)&&(j <= n)) { col = (int)atom_getintarg(1, argc, argv); - if(col < 0) + if (col < 0) { i = -1 - col; - x->x_col_src[j] = ((i & 0x3f000) << 6)|((i & 0xfc0) << 4)|((i & 0x3f) << 2); + x->x_col_src[j] = ((i & 0x3f000) << 6) | + ((i & 0xfc0) << 4) | + ((i & 0x3f) << 2); } else { - if(col > 29) + if (col > 29) + { col = 29; + } x->x_col_src[j] = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xSRC%d -fill #%6.6x\n", canvas, x, j, x->x_col_src[j]); + sprintf(colstring, "#%6.6x", x->x_col_src[j]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + j, + x->x_pix_src_x[j], + x->x_pix_src_y[j], + x->x_fontsize, + colstring + ); +// sys_vgui(".x%x.c itemconfigure %xSRC%d -fill #%6.6x\n", canvas, x, j, x->x_col_src[j]); } } } @@ -637,18 +772,18 @@ static void room_sim_2d_src_col(t_room_sim_2d *x, t_symbol *s, int argc, t_atom static void room_sim_2d_pix_per_m_ratio(t_room_sim_2d *x, t_floatarg ratio) { t_float rr; - int i, n=x->x_nr_src; + int i, n = x->x_nr_src; - if(ratio < 1.0f) + if (ratio < 1.0f) ratio = 1.0f; - if(ratio > 200.0f) + if (ratio > 200.0f) ratio = 200.0f; rr = ratio / x->x_cnvrt_roomlx2pixh; x->x_cnvrt_roomlx2pixh = ratio; x->x_gui.x_w = (int)(x->x_cnvrt_roomlx2pixh * x->x_room_y + 0.49999f); x->x_gui.x_h = (int)(x->x_cnvrt_roomlx2pixh * x->x_room_x + 0.49999f); x->x_pix_rad = (int)(x->x_cnvrt_roomlx2pixh * x->x_r_ambi + 0.49999f); - for(i=0; i<=n; i++) + for (i = 0; i <= n; i++) { x->x_pix_src_x[i] = (int)((t_float)x->x_pix_src_x[i]*rr + 0.49999f); x->x_pix_src_y[i] = (int)((t_float)x->x_pix_src_y[i]*rr + 0.49999f); @@ -659,7 +794,7 @@ static void room_sim_2d_pix_per_m_ratio(t_room_sim_2d *x, t_floatarg ratio) static void room_sim_2d_r_ambi(t_room_sim_2d *x, t_floatarg r_ambi) { - if(r_ambi < 0.1f) + if (r_ambi < 0.1f) r_ambi = 0.1f; x->x_r_ambi = r_ambi; x->x_pix_rad = (int)(x->x_cnvrt_roomlx2pixh*r_ambi + 0.49999f); @@ -672,51 +807,103 @@ static void room_sim_2d_nr_src(t_room_sim_2d *x, t_floatarg fnr_src) int nr_src = (int)fnr_src; int old_nr_src, i, j; - if(nr_src < 1) + if (nr_src < 1) nr_src = 1; - else if(nr_src > IEM_GUI_ROOMSIM_2D_MAX_NR_SRC) + else if (nr_src > IEM_GUI_ROOMSIM_2D_MAX_NR_SRC) nr_src = IEM_GUI_ROOMSIM_2D_MAX_NR_SRC; - if(nr_src != x->x_nr_src) + if (nr_src != x->x_nr_src) { - if(glist_isvisible(x->x_gui.x_glist)) - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); + if (glist_isvisible(x->x_gui.x_glist)) + room_sim_2d_draw_unmap(x, x->x_gui.x_glist); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); old_nr_src = x->x_nr_src; x->x_nr_src = nr_src; j = (old_nr_src + 1) % 7; - for(i=old_nr_src+1; i<=nr_src; i++) + for (i = old_nr_src + 1; i <= nr_src; i++) { x->x_col_src[i] = simularca_color_hex[j]; - if(i & 1) - x->x_pix_src_x[i] = 125 + (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i)*4; + if (i & 1) + x->x_pix_src_x[i] = 125 + (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i) * 4; else - x->x_pix_src_x[i] = 125 - (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i)*4; + x->x_pix_src_x[i] = 125 - (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i) * 4; x->x_pix_src_y[i] = 100; j++; j %= 7; } - - if(glist_isvisible(x->x_gui.x_glist)) - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); + if (glist_isvisible(x->x_gui.x_glist)) + room_sim_2d_draw_map(x, x->x_gui.x_glist); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); } } +/* we may no longer need h_dragon... */ +static void room_sim_2d__clickhook(t_scalehandle *sh, int newstate) +{ + t_room_sim_2d *x = (t_room_sim_2d *)(sh->h_master); + if (newstate) + { + canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x); + if (!sh->h_scale) /* click on a label handle */ + scalehandle_click_label(sh); + } + /* We no longer need this "clickhook", as we can handle the dragging + either in the GUI (for the label handle) or or in canvas_doclick */ + //iemgui__clickhook3(sh,newstate); + sh->h_dragon = newstate; +} + +static void room_sim_2d__motionhook(t_scalehandle *sh, + t_floatarg mouse_x, t_floatarg mouse_y) +{ + if (sh->h_scale) + { + t_room_sim_2d *x = (t_room_sim_2d *)(sh->h_master); + int width = mouse_x - text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist), + height = mouse_y - text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist), + minx = IEM_GUI_MINSIZE, + miny = IEM_GUI_MINSIZE; + x->x_gui.x_w = maxi(width, minx); + x->x_gui.x_h = maxi(height, miny); +// slider_check_length(x, x->x_orient ? x->x_gui.x_h : x->x_gui.x_w); + if (glist_isvisible(x->x_gui.x_glist)) + { + room_sim_2d_draw_unmap(x, x->x_gui.x_glist); + room_sim_2d_draw_map(x, x->x_gui.x_glist); + scalehandle_unclick_scale(sh); + } + + int properties = gfxstub_haveproperties((void *)x); + if (properties) + { + /* No properties for room_sim externals atm */ + //properties_set_field_int(properties,"width",new_w); + //properties_set_field_int(properties,"height",new_h); + } + } + scalehandle_dragon_label(sh,mouse_x, mouse_y); +} + +/* from the old header... */ +#define IEM_GUI_COLNR_GREEN 16 +#define IEM_GUI_COLNR_D_ORANGE 24 + static void *room_sim_2d_new(t_symbol *s, int argc, t_atom *argv) { t_room_sim_2d *x = (t_room_sim_2d *)pd_new(room_sim_2d_class); - int i, j, n=1, c; + int i, j, n = 1, c; - if((argc >= 1)&&IS_A_FLOAT(argv,0)) + if ((argc >= 1) && IS_A_FLOAT(argv,0)) { n = (int)atom_getintarg(0, argc, argv); - if(n < 1) + if (n < 1) n = 1; - if(n > IEM_GUI_ROOMSIM_2D_MAX_NR_SRC) + if (n > IEM_GUI_ROOMSIM_2D_MAX_NR_SRC) n = IEM_GUI_ROOMSIM_2D_MAX_NR_SRC; x->x_nr_src = n; } - if(argc == (3*n + 11)) + if (argc == (3 * n + 11)) { x->x_cnvrt_roomlx2pixh = atom_getfloatarg(1, argc, argv); x->x_rho_head = atom_getfloatarg(2, argc, argv); @@ -730,12 +917,14 @@ static void *room_sim_2d_new(t_symbol *s, int argc, t_atom *argv) x->x_gui.x_fcol = ((c & 0x3f000) << 6)|((c & 0xfc0) << 4)|((c & 0x3f) << 2); x->x_pix_src_x[0] = (int)atom_getintarg(9, argc, argv); x->x_pix_src_y[0] = (int)atom_getintarg(10, argc, argv); - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { c = (int)atom_getintarg(8+3*i, argc, argv); - x->x_col_src[i] = ((c & 0x3f000) << 6)|((c & 0xfc0) << 4)|((c & 0x3f) << 2); - x->x_pix_src_x[i] = (int)atom_getintarg(9+3*i, argc, argv); - x->x_pix_src_y[i] = (int)atom_getintarg(10+3*i, argc, argv); + x->x_col_src[i] = ((c & 0x3f000) << 6) | + ((c & 0xfc0) << 4) | + ((c & 0x3f) << 2); + x->x_pix_src_x[i] = (int)atom_getintarg(9 + 3 * i, argc, argv); + x->x_pix_src_y[i] = (int)atom_getintarg(10 + 3 * i, argc, argv); } } else @@ -751,13 +940,13 @@ static void *room_sim_2d_new(t_symbol *s, int argc, t_atom *argv) x->x_pix_src_x[0] = 125; x->x_pix_src_y[0] = 200; j = 0; - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { x->x_col_src[i] = simularca_color_hex[j]; - if(i & 1) - x->x_pix_src_x[i] = 125 + (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i)*4; + if (i & 1) + x->x_pix_src_x[i] = 125 + (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i) * 4; else - x->x_pix_src_x[i] = 125 - (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i)*4; + x->x_pix_src_x[i] = 125 - (IEM_GUI_ROOMSIM_2D_MAX_NR_SRC - i) * 4; x->x_pix_src_y[i] = 100; j++; j %= 7; @@ -776,6 +965,15 @@ static void *room_sim_2d_new(t_symbol *s, int argc, t_atom *argv) x->x_s_head_xy = gensym("head_xy"); x->x_s_src_xy = gensym("src_xy"); + + x->x_gui.x_lab = s_empty; + + x->x_gui.x_obj.te_iemgui = 1; + x->x_gui.x_handle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 1, room_sim_2d__clickhook, room_sim_2d__motionhook); + x->x_gui.x_lhandle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 0, room_sim_2d__clickhook, room_sim_2d__motionhook); + return (x); } @@ -786,31 +984,49 @@ static void room_sim_2d_ff(t_room_sim_2d *x) void room_sim_2d_setup(void) { - room_sim_2d_class = class_new(gensym("room_sim_2d"), (t_newmethod)room_sim_2d_new, - (t_method)room_sim_2d_ff, sizeof(t_room_sim_2d), 0, A_GIMME, 0); + room_sim_2d_class = class_new(gensym("room_sim_2d"), + (t_newmethod)room_sim_2d_new, (t_method)room_sim_2d_ff, + sizeof(t_room_sim_2d), 0, A_GIMME, 0); // class_addcreator((t_newmethod)room_sim_2d_new, gensym("room_sim_2d"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_click, gensym("click"), + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_click, + gensym("click"), A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_motion, gensym("motion"), + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_motion, + gensym("motion"), A_FLOAT, A_FLOAT, 0); class_addbang(room_sim_2d_class, (t_method)room_sim_2d_bang); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_room_dim, gensym("room_dim"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_r_ambi, gensym("r_ambi"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_room_col, gensym("room_col"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_head_col, gensym("head_col"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_col, gensym("src_col"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_rho, gensym("rho"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_xy, gensym("src_xy"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_head_xy, gensym("head_xy"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_rho, gensym("set_rho"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_src_xy, gensym("set_src_xy"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_head_xy, gensym("set_head_xy"), A_GIMME, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_pix_per_m_ratio, gensym("pix_per_m_ratio"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_font, gensym("src_font"), A_DEFFLOAT, 0); - class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_nr_src, gensym("nr_src"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_room_dim, + gensym("room_dim"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_r_ambi, + gensym("r_ambi"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_room_col, + gensym("room_col"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_head_col, + gensym("head_col"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_col, + gensym("src_col"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_rho, + gensym("rho"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_xy, + gensym("src_xy"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_head_xy, + gensym("head_xy"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_rho, + gensym("set_rho"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_src_xy, + gensym("set_src_xy"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_set_head_xy, + gensym("set_head_xy"), A_GIMME, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_pix_per_m_ratio, + gensym("pix_per_m_ratio"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_src_font, + gensym("src_font"), A_DEFFLOAT, 0); + class_addmethod(room_sim_2d_class, (t_method)room_sim_2d_nr_src, + gensym("nr_src"), A_DEFFLOAT, 0); room_sim_2d_widgetbehavior.w_getrectfn = room_sim_2d_getrect; room_sim_2d_widgetbehavior.w_displacefn = iemgui_displace; + room_sim_2d_widgetbehavior.w_displacefnwtag = iemgui_displace_withtag; room_sim_2d_widgetbehavior.w_selectfn = iemgui_select; room_sim_2d_widgetbehavior.w_activatefn = NULL; room_sim_2d_widgetbehavior.w_deletefn = iemgui_delete; diff --git a/externals/iem/iemgui/src/room_sim_3d.c b/externals/iem/iemgui/src/room_sim_3d.c index 043d8660b39a983fff3b60375c435d8d07a00555..7b845683760bd0e0aabcabc774872bbdf52bef70 100644 --- a/externals/iem/iemgui/src/room_sim_3d.c +++ b/externals/iem/iemgui/src/room_sim_3d.c @@ -7,7 +7,7 @@ iemgui written by Thomas Musil, Copyright (c) IEM KUG Graz Austria 2000 - 2006 * #include "iemlib.h" #include "iemgui.h" #include "g_canvas.h" -#include "../../../old_g_all_guis.inc" +#include "g_all_guis.h" #include <stdio.h> #include <stdlib.h> @@ -54,6 +54,7 @@ typedef struct _room_sim_3d t_symbol *x_s_head_xyz; t_symbol *x_s_src_xyz; t_atom x_at[6]; + int x_steady; } t_room_sim_3d; static void room_sim_3d_out_rho(t_room_sim_3d *x) @@ -94,57 +95,131 @@ static void room_sim_3d_draw_update(t_room_sim_3d *x, t_glist *glist) dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); + + gui_vmess("gui_room_sim_update", "xxiiiii", + canvas, + x, + x->x_pix_src_x[0], + x->x_pix_src_y[0], + dx, + dy, + x->x_pix_rad + ); } } -void room_sim_3d_draw_new(t_room_sim_3d *x, t_glist *glist) +void room_sim_3d_draw_unmap(t_room_sim_3d *x, t_glist *glist) { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); + gui_vmess("gui_room_sim_erase", "xx", + x->x_gui.x_glist, x); +} + +void room_sim_3d_draw_map(t_room_sim_3d *x, t_glist *glist) +{ + int xpos = text_xpix(&x->x_gui.x_obj, glist); + int ypos = text_ypix(&x->x_gui.x_obj, glist); int dx, dy; - int H=(int)(0.5f * x->x_room_z * x->x_cnvrt_roomlx2pixh + 0.49999f); - int H2=H*H; + int H = (int)(0.5f * x->x_room_z * x->x_cnvrt_roomlx2pixh + 0.49999f); + int H2 = H*H; int rad2; - int i, n=x->x_nr_src; - int fsi, fs=x->x_fontsize; - t_canvas *canvas=glist_getcanvas(glist); + int i, n = x->x_nr_src; + int fsi, fs = x->x_fontsize; + t_canvas *canvas = glist_getcanvas(glist); + char bcol[MAXPDSTRING]; + char fcol[MAXPDSTRING]; + char elemcol[MAXPDSTRING]; + sprintf(fcol, "#%6.6x", x->x_gui.x_fcol); + sprintf(bcol, "#%6.6x", x->x_gui.x_bcol); - sys_vgui(".x%x.c create rectangle %d %d %d %d -fill #%6.6x -outline #%6.6x -tags %xBASE\n", - canvas, xpos, ypos, xpos + x->x_gui.x_w, ypos + x->x_gui.x_h, - x->x_gui.x_bcol, x->x_gui.x_fsf.x_selected?IEM_GUI_COLOR_SELECTED:IEM_GUI_COLOR_NORMAL, x); - for(i=1; i<=n; i++) + gui_start_vmess("gui_room_sim_map", "xxiiifiiiss", + canvas, + x, + x->x_gui.x_w, + x->x_gui.x_h, + x->x_pix_rad, + x->x_rho_head, + x->x_pix_src_x[0], + x->x_pix_src_y[0], + x->x_fontsize, + fcol, + bcol + ); + gui_start_array(); + for (i = 1; i <= n; i++) { fsi = H2 + (H + 2 * x->x_pix_src_z[i]) * x->x_pix_src_z[i]; fsi *= fs; fsi /= 2*H2; - sys_vgui(".x%x.c create text %d %d -text {%d} -anchor c \ - -font {times %d bold} -fill #%6.6x -tags %xSRC%d\n", - canvas, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i], i, fsi, - x->x_col_src[i], x, i); + gui_start_array(); + gui_i(x->x_pix_src_x[i]); + gui_i(x->x_pix_src_y[i]); + sprintf(elemcol, "#%6.6x", x->x_col_src[i]); + gui_s(elemcol); + gui_i(fsi); + gui_end_array(); +// sys_vgui(".x%x.c create text %d %d -text {%d} -anchor c \ +// -font {times %d bold} -fill #%6.6x -tags %xSRC%d\n", +// canvas, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i], i, fsi, +// x->x_col_src[i], x, i); } + gui_end_array(); - sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD\n", - canvas, xpos+x->x_pix_src_x[0]-x->x_pix_rad, ypos+x->x_pix_src_y[0]-x->x_pix_rad, - xpos+x->x_pix_src_x[0]+x->x_pix_rad-1, ypos+x->x_pix_src_y[0]+x->x_pix_rad-1, - x->x_gui.x_fcol, x); +// sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD\n", +// canvas, xpos+x->x_pix_src_x[0]-x->x_pix_rad, ypos+x->x_pix_src_y[0]-x->x_pix_rad, +// xpos+x->x_pix_src_x[0]+x->x_pix_rad-1, ypos+x->x_pix_src_y[0]+x->x_pix_rad-1, +// x->x_gui.x_fcol, x); + rad2 = H2 + (H + 2 * x->x_pix_src_z[0]) * x->x_pix_src_z[0]; rad2 *= x->x_pix_rad; rad2 /= 8*H2; - sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD2\n", - canvas, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, - xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1, - x->x_gui.x_fcol, x); - dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c create line %d %d %d %d -width 3 -fill #%6.6x -tags %xNOSE\n", - canvas, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy, - x->x_gui.x_fcol, x); + gui_start_array(); + gui_i(x->x_pix_src_x[0] - rad2); + gui_i(x->x_pix_src_y[0] - rad2); + gui_i(x->x_pix_src_x[0] + rad2 - 1); + gui_i(x->x_pix_src_y[0] + rad2 - 1); + gui_end_array(); + gui_end_vmess(); +// sys_vgui(".x%x.c create oval %d %d %d %d -outline #%6.6x -tags %xHEAD2\n", +// canvas, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, +// xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1, +// x->x_gui.x_fcol, x); +// dx = -(int)((t_float)x->x_pix_rad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// dy = -(int)((t_float)x->x_pix_rad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c create line %d %d %d %d -width 3 -fill #%6.6x -tags %xNOSE\n", +// canvas, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy, +// x->x_gui.x_fcol, x); +} + +void room_sim_3d_draw_new(t_room_sim_3d *x, t_glist *glist) +{ + int xpos=text_xpix(&x->x_gui.x_obj, glist); + int ypos=text_ypix(&x->x_gui.x_obj, glist); + int dx, dy; + int H=(int)(0.5f * x->x_room_z * x->x_cnvrt_roomlx2pixh + 0.49999f); + int H2=H*H; + int rad2; + int i, n=x->x_nr_src; + int fsi, fs=x->x_fontsize; + t_canvas *canvas=glist_getcanvas(glist); + + gui_vmess("gui_room_sim_new", "xxiiiii", + canvas, + x, + xpos, + ypos, + x->x_gui.x_w, + x->x_gui.x_h, + glist_istoplevel(glist) + ); + + room_sim_3d_draw_map(x, glist); } +/* void room_sim_3d_draw_move(t_room_sim_3d *x, t_glist *glist) { int xpos=text_xpix(&x->x_gui.x_obj, glist); @@ -185,52 +260,22 @@ void room_sim_3d_draw_move(t_room_sim_3d *x, t_glist *glist) canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], xpos+x->x_pix_src_x[0]+dx, ypos+x->x_pix_src_y[0]+dy); } - -void room_sim_3d_draw_erase(t_room_sim_3d* x, t_glist* glist) -{ - int i, n; - t_canvas *canvas=glist_getcanvas(glist); - - sys_vgui(".x%x.c delete %xBASE\n", canvas, x); - n = x->x_nr_src; - for(i=1; i<=n; i++) - { - sys_vgui(".x%x.c delete %xSRC%d\n", canvas, x, i); - } - sys_vgui(".x%x.c delete %xHEAD\n", canvas, x); - sys_vgui(".x%x.c delete %xHEAD2\n", canvas, x); - sys_vgui(".x%x.c delete %xNOSE\n", canvas, x); -} - -void room_sim_3d_draw_select(t_room_sim_3d* x, t_glist* glist) -{ - t_canvas *canvas=glist_getcanvas(glist); - - if(x->x_gui.x_fsf.x_selected) - { - int xpos=text_xpix(&x->x_gui.x_obj, glist); - int ypos=text_ypix(&x->x_gui.x_obj, glist); - - sys_vgui(".x%x.c itemconfigure %xBASE -outline #%6.6x\n", canvas, x, IEM_GUI_COLOR_SELECTED); - } - else - { - sys_vgui(".x%x.c itemconfigure %xBASE -outline #%6.6x\n", canvas, x, IEM_GUI_COLOR_NORMAL); - } -} +*/ void room_sim_3d_draw(t_room_sim_3d *x, t_glist *glist, int mode) { - if(mode == IEM_GUI_DRAW_MODE_UPDATE) + if (mode == IEM_GUI_DRAW_MODE_UPDATE) room_sim_3d_draw_update(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_MOVE) - room_sim_3d_draw_move(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_NEW) + else if (mode == IEM_GUI_DRAW_MODE_MOVE) + iemgui_base_draw_move(&x->x_gui); + else if (mode == IEM_GUI_DRAW_MODE_NEW) room_sim_3d_draw_new(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_SELECT) +/* + else if (mode == IEM_GUI_DRAW_MODE_SELECT) room_sim_3d_draw_select(x, glist); - else if(mode == IEM_GUI_DRAW_MODE_ERASE) + else if (mode == IEM_GUI_DRAW_MODE_ERASE) room_sim_3d_draw_erase(x, glist); +*/ } /* ------------------------ cnv widgetbehaviour----------------------------- */ @@ -272,44 +317,44 @@ static void room_sim_3d_save(t_gobj *z, t_binbuf *b) static void room_sim_3d_motion(t_room_sim_3d *x, t_floatarg dx, t_floatarg dy) { - int i, n=x->x_nr_src; - int pixrad=x->x_pix_rad; - int sel=x->x_sel_index; - int xpos=text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); - int ypos=text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); + int i, n = x->x_nr_src; + int pixrad = x->x_pix_rad; + int sel = x->x_sel_index; + int xpos = text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist); + int ypos = text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist); int ddx, ddy; - int fs=x->x_fontsize, fsi; - int H=(int)(0.5f * x->x_room_z * x->x_cnvrt_roomlx2pixh + 0.49999f); - int H2=H*H; + int fs = x->x_fontsize, fsi; + int H = (int)(0.5f * x->x_room_z * x->x_cnvrt_roomlx2pixh + 0.49999f); + int H2 = H*H; int rad2; - t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); + t_canvas *canvas = glist_getcanvas(x->x_gui.x_glist); - if(x->x_gui.x_fsf.x_finemoved && (sel == 0)) + if (x->x_gui.x_finemoved && (sel == 0)) { - if(x->x_gui.x_fsf.x_steady)/*alt-key, rhoy, rhox*/ + if (x->x_steady)/*alt-key, rhoy, rhox*/ { } else { x->x_rho_head -= dy; - if(x->x_rho_head <= -180.0f) + if (x->x_rho_head <= -180.0f) x->x_rho_head += 360.0f; - if(x->x_rho_head > 180.0f) + if (x->x_rho_head > 180.0f) x->x_rho_head -= 360.0f; room_sim_3d_out_rho(x); (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); } } - else if(sel == 0) + else if (sel == 0) { - if(x->x_gui.x_fsf.x_steady)/*alt-key, move head in z*/ + if (x->x_steady) /*alt-key, move head in z */ { x->x_pix_src_x[0] = x->x_pos_x; x->x_pix_src_y[0] = x->x_pos_y; x->x_pix_src_z[0] -= (int)dy; - if(x->x_pix_src_z[0] < 0) + if (x->x_pix_src_z[0] < 0) x->x_pix_src_z[0] = 0; - if(x->x_pix_src_z[0] > x->x_height_z) + if (x->x_pix_src_z[0] > x->x_height_z) x->x_pix_src_z[0] = x->x_height_z; } else @@ -319,46 +364,66 @@ static void room_sim_3d_motion(t_room_sim_3d *x, t_floatarg dx, t_floatarg dy) x->x_pos_y += (int)dy; x->x_pix_src_x[0] = x->x_pos_x; x->x_pix_src_y[0] = x->x_pos_y; - if(x->x_pix_src_x[0] < 0) + if (x->x_pix_src_x[0] < 0) x->x_pix_src_x[0] = 0; - if(x->x_pix_src_x[0] > x->x_gui.x_w) + if (x->x_pix_src_x[0] > x->x_gui.x_w) x->x_pix_src_x[0] = x->x_gui.x_w; - if(x->x_pix_src_y[0] < 0) + if (x->x_pix_src_y[0] < 0) x->x_pix_src_y[0] = 0; - if(x->x_pix_src_y[0] > x->x_gui.x_h) + if (x->x_pix_src_y[0] > x->x_gui.x_h) x->x_pix_src_y[0] = x->x_gui.x_h; } room_sim_3d_out_para(x); - sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, - xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); + (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); +// sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, +// xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); rad2 = H2 + (H + 2 * x->x_pix_src_z[0]) * x->x_pix_src_z[0]; rad2 *= x->x_pix_rad; rad2 /= 8*H2; - sys_vgui(".x%x.c coords %xHEAD2 %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, - xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1); - ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); +// sys_vgui(".x%x.c coords %xHEAD2 %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, +// xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1); + gui_vmess("gui_room_sim_head2", "xxiiii", + canvas, + x, + x->x_pix_src_x[0] - rad2, + x->x_pix_src_y[0] - rad2, + x->x_pix_src_x[0] + rad2 - 1, + x->x_pix_src_y[0] + rad2 - 1 + ); +// ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); } else { - if(x->x_gui.x_fsf.x_steady)/*alt-key, move src in z*/ + if (x->x_steady) /*alt-key, move src in z*/ { x->x_pix_src_z[sel] -= (int)dy; - if(x->x_pix_src_z[sel] < 0) + if (x->x_pix_src_z[sel] < 0) x->x_pix_src_z[sel] = 0; - if(x->x_pix_src_z[sel] > x->x_height_z) + if (x->x_pix_src_z[sel] > x->x_height_z) x->x_pix_src_z[sel] = x->x_height_z; room_sim_3d_out_para(x); fsi = H2 + (H + 2 * x->x_pix_src_z[sel]) * x->x_pix_src_z[sel]; fsi *= fs; fsi /= 2*H2; - sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, sel, fsi); + char col[MAXPDSTRING]; + sprintf(col, "#%6.6x", x->x_col_src[sel]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + sel - 1, + x->x_pix_src_x[sel], + x->x_pix_src_y[sel], + fsi, + col + ); +// sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, sel, fsi); } else { @@ -367,22 +432,33 @@ static void room_sim_3d_motion(t_room_sim_3d *x, t_floatarg dx, t_floatarg dy) x->x_pix_src_x[sel] = x->x_pos_x; x->x_pix_src_y[sel] = x->x_pos_y; x->x_pix_src_z[sel] = x->x_pos_z; - if(x->x_pix_src_x[sel] < 0) + if (x->x_pix_src_x[sel] < 0) x->x_pix_src_x[sel] = 0; - if(x->x_pix_src_x[sel] > x->x_gui.x_w) + if (x->x_pix_src_x[sel] > x->x_gui.x_w) x->x_pix_src_x[sel] = x->x_gui.x_w; - if(x->x_pix_src_y[sel] < 0) + if (x->x_pix_src_y[sel] < 0) x->x_pix_src_y[sel] = 0; - if(x->x_pix_src_y[sel] > x->x_gui.x_h) + if (x->x_pix_src_y[sel] > x->x_gui.x_h) x->x_pix_src_y[sel] = x->x_gui.x_h; room_sim_3d_out_para(x); fsi = H2 + (H + 2 * x->x_pix_src_z[sel]) * x->x_pix_src_z[sel]; fsi *= fs; fsi /= 2*H2; - sys_vgui(".x%x.c coords %xSRC%d %d %d\n", - canvas, x, sel, xpos+x->x_pix_src_x[sel], ypos+x->x_pix_src_y[sel]); - sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, sel, fsi); + char col[MAXPDSTRING]; + sprintf(col, "#%6.6x", x->x_col_src[sel]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + sel - 1, + x->x_pix_src_x[sel], + x->x_pix_src_y[sel], + fsi, + col + ); + //sys_vgui(".x%x.c coords %xSRC%d %d %d\n", + // canvas, x, sel, xpos+x->x_pix_src_x[sel], ypos+x->x_pix_src_y[sel]); + //sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, sel, fsi); } } } @@ -468,18 +544,18 @@ static int room_sim_3d_newclick(t_gobj *z, struct _glist *glist, int xpix, int y { room_sim_3d_click( x, (t_floatarg)xpix, (t_floatarg)ypix, (t_floatarg)shift, 0, (t_floatarg)alt); if(alt) - x->x_gui.x_fsf.x_steady = 1; + x->x_steady = 1; else - x->x_gui.x_fsf.x_steady = 0; + x->x_steady = 0; if(shift) { - x->x_gui.x_fsf.x_finemoved = 1; + x->x_gui.x_finemoved = 1; room_sim_3d_out_rho(x); } else { - x->x_gui.x_fsf.x_finemoved = 0; + x->x_gui.x_finemoved = 0; room_sim_3d_out_para(x); } } @@ -511,7 +587,19 @@ static void room_sim_3d_src_font(t_room_sim_3d *x, t_floatarg ff) fsi = H2 + (H + 2 * x->x_pix_src_z[i]) * x->x_pix_src_z[i]; fsi *= fs; fsi /= 2*H2; - sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fsi); + + char col[MAXPDSTRING]; + sprintf(col, "#%6.6x", x->x_col_src[i]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + i - 1, + x->x_pix_src_x[i], + x->x_pix_src_y[i], + fsi, + col + ); + //sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fsi); } } @@ -577,9 +665,21 @@ static void room_sim_3d_set_src_xyz(t_room_sim_3d *x, t_symbol *s, int argc, t_a fsi = H2 + (H + 2 * x->x_pix_src_z[i]) * x->x_pix_src_z[i]; fsi *= fs; fsi /= 2*H2; - sys_vgui(".x%x.c coords %xSRC%d %d %d\n", - canvas, x, i, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i]); - sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fsi); + + char col[MAXPDSTRING]; + sprintf(col, "#%6.6x", x->x_col_src[i]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + i - 1, + x->x_pix_src_x[i], + x->x_pix_src_y[i], + fsi, + col + ); +// sys_vgui(".x%x.c coords %xSRC%d %d %d\n", +// canvas, x, i, xpos+x->x_pix_src_x[i], ypos+x->x_pix_src_y[i]); +// sys_vgui(".x%x.c itemconfigure %xSRC%d -font {times %d bold}\n", canvas, x, i, fsi); } } @@ -626,20 +726,30 @@ static void room_sim_3d_set_head_xyz(t_room_sim_3d *x, t_symbol *s, int argc, t_ x->x_pix_src_y[0] = x->x_gui.x_h/2 - (int)(x->x_cnvrt_roomlx2pixh * yh + 0.49999f); x->x_pix_src_z[0] = (int)(x->x_cnvrt_roomlx2pixh * zh + 0.49999f); - sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, - xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); + (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_UPDATE); + +// sys_vgui(".x%x.c coords %xHEAD %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-pixrad, ypos+x->x_pix_src_y[0]-pixrad, +// xpos+x->x_pix_src_x[0]+pixrad-1, ypos+x->x_pix_src_y[0]+pixrad-1); rad2 = H2 + (H + 2 * x->x_pix_src_z[0]) * x->x_pix_src_z[0]; rad2 *= pixrad; rad2 /= 8*H2; - sys_vgui(".x%x.c coords %xHEAD2 %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, - xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1); - ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); - ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); - sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", - canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], - xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); + gui_vmess("gui_room_sim_head2", "xxiiii", + canvas, + x, + x->x_pix_src_x[0] - rad2, + x->x_pix_src_y[0] - rad2, + x->x_pix_src_x[0] + rad2 - 1, + x->x_pix_src_y[0] + rad2 - 1 + ); +// sys_vgui(".x%x.c coords %xHEAD2 %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0]-rad2, ypos+x->x_pix_src_y[0]-rad2, +// xpos+x->x_pix_src_x[0]+rad2-1, ypos+x->x_pix_src_y[0]+rad2-1); +// ddx = -(int)((t_float)pixrad*(t_float)sin(x->x_rho_head*0.0174533f) + 0.49999f); +// ddy = -(int)((t_float)pixrad*(t_float)cos(x->x_rho_head*0.0174533f) + 0.49999f); +// sys_vgui(".x%x.c coords %xNOSE %d %d %d %d\n", +// canvas, x, xpos+x->x_pix_src_x[0], ypos+x->x_pix_src_y[0], +// xpos+x->x_pix_src_x[0]+ddx, ypos+x->x_pix_src_y[0]+ddy); } static void room_sim_3d_head_xyz(t_room_sim_3d *x, t_symbol *s, int argc, t_atom *argv) @@ -673,13 +783,13 @@ static void room_sim_3d_room_dim(t_room_sim_3d *x, t_symbol *s, int argc, t_atom x->x_height_z = (int)(x->x_cnvrt_roomlx2pixh * x->x_room_z + 0.49999f); x->x_pix_rad = (int)(x->x_cnvrt_roomlx2pixh * x->x_r_ambi + 0.49999f); - for(i=0; i<=n; i++) + for (i = 0; i <= n; i++) { - if(x->x_pix_src_x[i] > x->x_gui.x_w) + if (x->x_pix_src_x[i] > x->x_gui.x_w) x->x_pix_src_x[i] = x->x_gui.x_w; - if(x->x_pix_src_y[i] > x->x_gui.x_h) + if (x->x_pix_src_y[i] > x->x_gui.x_h) x->x_pix_src_y[i] = x->x_gui.x_h; - if(x->x_pix_src_z[i] > x->x_height_z) + if (x->x_pix_src_z[i] > x->x_height_z) x->x_pix_src_z[i] = x->x_height_z; } @@ -691,26 +801,38 @@ static void room_sim_3d_room_dim(t_room_sim_3d *x, t_symbol *s, int argc, t_atom static void room_sim_3d_room_col(t_room_sim_3d *x, t_floatarg fcol) { int col=(int)fcol; + char fgstring[MAXPDSTRING]; + char bgstring[MAXPDSTRING]; int i; t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); - if(col < 0) + if (col < 0) { i = -1 - col; x->x_gui.x_bcol = ((i & 0x3f000) << 6)|((i & 0xfc0) << 4)|((i & 0x3f) << 2); } else { - if(col > 29) + if (col > 29) col = 29; x->x_gui.x_bcol = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, x->x_gui.x_bcol); + sprintf(fgstring, "#%6.6x", x->x_gui.x_fcol); + sprintf(bgstring, "#%6.6x", x->x_gui.x_bcol); + gui_vmess("gui_room_sim_colors", "xxss", + canvas, + x, + fgstring, + bgstring + ); +// sys_vgui(".x%x.c itemconfigure %xBASE -fill #%6.6x\n", canvas, x, x->x_gui.x_bcol); } static void room_sim_3d_head_col(t_room_sim_3d *x, t_floatarg fcol) { int col=(int)fcol; + char fgstring[MAXPDSTRING]; + char bgstring[MAXPDSTRING]; int i; t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); @@ -725,35 +847,54 @@ static void room_sim_3d_head_col(t_room_sim_3d *x, t_floatarg fcol) col = 29; x->x_gui.x_fcol = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xHEAD -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); + sprintf(fgstring, "#%6.6x", x->x_gui.x_fcol); + sprintf(bgstring, "#%6.6x", x->x_gui.x_bcol); + gui_vmess("gui_room_sim_colors", "xxss", + canvas, + x, + fgstring, + bgstring + ); +// sys_vgui(".x%x.c itemconfigure %xHEAD -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); sys_vgui(".x%x.c itemconfigure %xHEAD2 -outline #%6.6x\n", canvas, x, x->x_gui.x_fcol); - sys_vgui(".x%x.c itemconfigure %xNOSE -fill #%6.6x\n", canvas, x, x->x_gui.x_fcol); +// sys_vgui(".x%x.c itemconfigure %xNOSE -fill #%6.6x\n", canvas, x, x->x_gui.x_fcol); } static void room_sim_3d_src_col(t_room_sim_3d *x, t_symbol *s, int argc, t_atom *argv) { int col; + char colstring[MAXPDSTRING]; int i, j, n=x->x_nr_src; t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist); - if((argc >= 2)&&IS_A_FLOAT(argv,0)&&IS_A_FLOAT(argv,1)) + if ((argc >= 2)&&IS_A_FLOAT(argv,0)&&IS_A_FLOAT(argv,1)) { j = (int)atom_getintarg(0, argc, argv); - if((j > 0)&&(j <= n)) + if ((j > 0)&&(j <= n)) { col = (int)atom_getintarg(1, argc, argv); - if(col < 0) + if (col < 0) { i = -1 - col; x->x_col_src[j] = ((i & 0x3f000) << 6)|((i & 0xfc0) << 4)|((i & 0x3f) << 2); } else { - if(col > 29) + if (col > 29) col = 29; x->x_col_src[j] = my_iemgui_color_hex[col]; } - sys_vgui(".x%x.c itemconfigure %xSRC%d -fill #%6.6x\n", canvas, x, j, x->x_col_src[j]); + sprintf(colstring, "#%6.6x", x->x_col_src[j]); + gui_vmess("gui_room_sim_update_src", "xxiiiis", + canvas, + x, + j, + x->x_pix_src_x[j], + x->x_pix_src_y[j], + x->x_fontsize, + colstring + ); +// sys_vgui(".x%x.c itemconfigure %xSRC%d -fill #%6.6x\n", canvas, x, j, x->x_col_src[j]); } } } @@ -805,8 +946,9 @@ static void room_sim_3d_nr_src(t_room_sim_3d *x, t_floatarg fnr_src) if(nr_src != x->x_nr_src) { - if(glist_isvisible(x->x_gui.x_glist)) - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); + if (glist_isvisible(x->x_gui.x_glist)) + room_sim_3d_draw_unmap(x, x->x_gui.x_glist); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_ERASE); old_nr_src = x->x_nr_src; x->x_nr_src = nr_src; @@ -824,12 +966,64 @@ static void room_sim_3d_nr_src(t_room_sim_3d *x, t_floatarg fnr_src) j %= 7; } - if(glist_isvisible(x->x_gui.x_glist)) - (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); + if (glist_isvisible(x->x_gui.x_glist)) + room_sim_3d_draw_map(x, x->x_gui.x_glist); +// (*x->x_gui.x_draw)(x, x->x_gui.x_glist, IEM_GUI_DRAW_MODE_NEW); } } -static void *room_sim_3d_new(t_symbol *s, int argc, t_atom *argv) +/* we may no longer need h_dragon... */ +static void room_sim_3d__clickhook(t_scalehandle *sh, int newstate) +{ + t_room_sim_3d *x = (t_room_sim_3d *)(sh->h_master); + if (newstate) + { + canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x); + if (!sh->h_scale) /* click on a label handle */ + scalehandle_click_label(sh); + } + /* We no longer need this "clickhook", as we can handle the dragging + either in the GUI (for the label handle) or or in canvas_doclick */ + //iemgui__clickhook3(sh,newstate); + sh->h_dragon = newstate; +} + +static void room_sim_3d__motionhook(t_scalehandle *sh, + t_floatarg mouse_x, t_floatarg mouse_y) +{ + if (sh->h_scale) + { + t_room_sim_3d *x = (t_room_sim_3d *)(sh->h_master); + int width = mouse_x - text_xpix(&x->x_gui.x_obj, x->x_gui.x_glist), + height = mouse_y - text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist), + minx = IEM_GUI_MINSIZE, + miny = IEM_GUI_MINSIZE; + x->x_gui.x_w = maxi(width, minx); + x->x_gui.x_h = maxi(height, miny); +// slider_check_length(x, x->x_orient ? x->x_gui.x_h : x->x_gui.x_w); + if (glist_isvisible(x->x_gui.x_glist)) + { + room_sim_3d_draw_unmap(x, x->x_gui.x_glist); + room_sim_3d_draw_map(x, x->x_gui.x_glist); + scalehandle_unclick_scale(sh); + } + + int properties = gfxstub_haveproperties((void *)x); + if (properties) + { + /* No properties for room_sim externals atm */ + //properties_set_field_int(properties,"width",new_w); + //properties_set_field_int(properties,"height",new_h); + } + } + scalehandle_dragon_label(sh,mouse_x, mouse_y); +} + +/* from the old header... */ +#define IEM_GUI_COLNR_GREEN 16 +#define IEM_GUI_COLNR_D_ORANGE 24 + +void *room_sim_3d_new(t_symbol *s, int argc, t_atom *argv) { t_room_sim_3d *x = (t_room_sim_3d *)pd_new(room_sim_3d_class); int i, j, n=1, c; @@ -888,7 +1082,7 @@ static void *room_sim_3d_new(t_symbol *s, int argc, t_atom *argv) x->x_pix_src_y[0] = 200; x->x_pix_src_z[0] = 42; j = 0; - for(i=1; i<=n; i++) + for (i = 1; i <= n; i++) { x->x_col_src[i] = simularca_color_hex[j]; if(i & 1) @@ -915,6 +1109,15 @@ static void *room_sim_3d_new(t_symbol *s, int argc, t_atom *argv) x->x_s_head_xyz = gensym("head_xyz"); x->x_s_src_xyz = gensym("src_xyz"); + + x->x_gui.x_lab = s_empty; + + x->x_gui.x_obj.te_iemgui = 1; + x->x_gui.x_handle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 1, room_sim_3d__clickhook, room_sim_3d__motionhook); + x->x_gui.x_lhandle = scalehandle_new((t_object *)x, + x->x_gui.x_glist, 0, room_sim_3d__clickhook, room_sim_3d__motionhook); + return (x); } @@ -949,6 +1152,7 @@ void room_sim_3d_setup(void) class_addmethod(room_sim_3d_class, (t_method)room_sim_3d_nr_src, gensym("nr_src"), A_DEFFLOAT, 0); room_sim_3d_widgetbehavior.w_getrectfn = room_sim_3d_getrect; + room_sim_3d_widgetbehavior.w_displacefnwtag = iemgui_displace_withtag; room_sim_3d_widgetbehavior.w_displacefn = iemgui_displace; room_sim_3d_widgetbehavior.w_selectfn = iemgui_select; room_sim_3d_widgetbehavior.w_activatefn = NULL; diff --git a/externals/moonlib/mknob.c b/externals/moonlib/mknob.c index bbba8d7ae3f5ba95e45c49f1f130b72f0a370536..f12edcf074f1e1726cf0b65bf2ab5c49113ce996 100644 --- a/externals/moonlib/mknob.c +++ b/externals/moonlib/mknob.c @@ -744,7 +744,7 @@ static void *mknob_new(t_symbol *s, int argc, t_atom *argv) x->x_thick = 0; iemgui_verify_snd_ne_rcv(&x->x_gui); outlet_new(&x->x_gui.x_obj, &s_float); - x->x_gui.x_obj.te_iemgui = 1; + x->x_gui.x_obj.te_iemgui = 1; x->x_gui.x_handle = scalehandle_new((t_object *)x, x->x_gui.x_glist, 1, mknob__clickhook, mknob__motionhook); diff --git a/packages/win32_inno/Makefile b/packages/win32_inno/Makefile index 348cc1a353d06656490e980151c0e26b884ee10c..2303725c378090363d6e67cb6cb45586b40a6755 100755 --- a/packages/win32_inno/Makefile +++ b/packages/win32_inno/Makefile @@ -108,7 +108,7 @@ ifneq ($(LIGHT),yes) install -p $(bin_src)/libsystre-0.dll $(DESTDIR)$(bindir)/ install -p $(bin_src)/libtre-5.dll $(DESTDIR)$(bindir)/ # For fluid~ - install -p $(bin_src)/libfluidsynth-1.dll $(DESTDIR)$(bindir)/ + install -p $(bin_src)/libfluidsynth-2.dll $(DESTDIR)$(bindir)/ # For other external libs install -p $(bin_src)/lua53.dll $(DESTDIR)$(bindir)/ #install -p $(bin_src)/pthreadGC2.dll $(DESTDIR)$(bindir)/ diff --git a/pd/doc/5.reference/unpost-help.pd b/pd/doc/5.reference/unpost-help.pd new file mode 100644 index 0000000000000000000000000000000000000000..bf9dea67e30eaa2917a6c2314f54a0bc73d7ef3f --- /dev/null +++ b/pd/doc/5.reference/unpost-help.pd @@ -0,0 +1,83 @@ +#N canvas 430 62 555 619 10; +#X obj 0 585 cnv 15 552 21 empty \$0-pddp.cnv.footer empty 20 12 0 +14 -228856 -66577 0; +#X obj 0 0 cnv 15 552 40 empty \$0-pddp.cnv.header unpost 3 12 0 18 +-204280 -1 0; +#X obj 0 279 cnv 3 550 3 empty \$0-pddp.cnv.inlets inlets 8 12 0 13 +-228856 -1 0; +#N canvas 497 325 482 332 META 0; +#X text 12 145 LIBRARY internal; +#X text 12 185 WEBSITE http://crca.ucsd.edu/~msp/; +#X text 12 45 LICENSE SIBSD; +#X text 12 165 AUTHOR Miller Puckette; +#X text 12 225 HELP_PATCH_AUTHORS Jonathan Wilkes revised the patch +to conform to the PDDP template for Pd version 0.42.; +#X text 12 25 KEYWORDS control storage; +#X text 12 205 RELEASE_DATE 1997; +#X text 12 65 DESCRIPTION reroute console messages to an outlet; +#X text 12 85 INLET_0 anything; +#X text 12 125 OUTLET_1 anything; +#X text 12 105 OUTLET_0 symbol; +#X restore 500 587 pd META; +#X obj 0 336 cnv 3 550 3 empty \$0-pddp.cnv.outlets outlets 8 12 0 +13 -228856 -1 0; +#X obj 0 503 cnv 3 550 3 empty \$0-pddp.cnv.argument arguments 8 12 +0 13 -228856 -1 0; +#X obj 0 553 cnv 3 550 3 empty \$0-pddp.cnv.more_info more_info 8 12 +0 13 -228856 -1 0; +#X obj 78 352 cnv 17 3 17 empty \$0-pddp.cnv.let.0 0 5 9 0 16 -228856 +-162280 0; +#N canvas 215 545 428 131 Related_objects 0; +#X obj 1 1 cnv 15 425 20 empty \$0-pddp.cnv.subheading empty 3 12 0 +14 -204280 -1 0; +#X text 7 1 [unpost] Related Objects; +#X obj 22 36 print; +#X restore 101 587 pd Related_objects; +#X obj 78 295 cnv 17 3 17 empty \$0-pddp.cnv.let.0 0 5 9 0 16 -228856 +-162280 0; +#X obj 4 587 pddp/pddplink all_about_help_patches.pd -text Usage Guide +; +#X text 11 20 reroute console messages to an outlet; +#X obj 502 12 unpost; +#X msg 58 60 bad_method; +#X obj 58 112 unpost; +#X symbolatom 58 194 0 0 0 0 - - -; +#X text 98 351 symbol; +#X text 168 351 - outputs the console messages. The messages triggered +by the right outlet are split at newlines. Then each resulting line +is converted to a symbol and output in sequence. (This means a single +message to the inlet of [unpost] may trigger multiple messages to this +outlet.; +#X text 99 295 anything; +#X text 169 295 - then input is sent to the right outlet; +#X obj 78 432 cnv 17 3 17 empty \$0-pddp.cnv.let.1 1 5 9 0 16 -228856 +-162280 0; +#X text 98 431 anything; +#X obj 91 163 print redirected; +#X msg 81 85 1; +#X obj 91 141 + 41; +#X msg 238 60 bad_method; +#X symbolatom 238 194 0 0 0 0 - - -; +#X obj 271 163 print redirected; +#X msg 261 85 1; +#X obj 271 141 + 41; +#X obj 238 112 unpost error; +#X text 319 112 <- only reroute errors; +#X text 168 431 - the message sent to the inlet is received to this +outlet. Any console messages which are triggered from the rest of object +chain connected below it will get forwarded to the left outlet. If +the "error" argument was given \, then only error messages will be +redirected to the left outlet.; +#X text 81 520 1) symbol; +#X text 171 520 - the symbol "error" will only redirect error messages. +All other console messages will get printed as usual.; +#X connect 13 0 14 0; +#X connect 14 0 15 0; +#X connect 14 1 24 0; +#X connect 23 0 14 0; +#X connect 24 0 22 0; +#X connect 25 0 30 0; +#X connect 28 0 30 0; +#X connect 29 0 27 0; +#X connect 30 0 26 0; +#X connect 30 1 29 0; diff --git a/pd/nw/pd_canvas.js b/pd/nw/pd_canvas.js index 570b20e99bb0d78b99a15a6e7cf6d33a20381804..db54bda82cc9a561b3f0fe2187b9920661862f20 100644 --- a/pd/nw/pd_canvas.js +++ b/pd/nw/pd_canvas.js @@ -11,102 +11,6 @@ pdgui.skin.apply(window); var l = pdgui.get_local_string; -console.log("my working dir is " + pdgui.get_pwd()); -document.getElementById("saveDialog") - .setAttribute("nwworkingdir", pdgui.get_pwd()); -document.getElementById("fileDialog") - .setAttribute("nwworkingdir", pdgui.get_pwd()); -document.getElementById("fileDialog").setAttribute("accept", - Object.keys(pdgui.pd_filetypes).toString()); - -function close_save_dialog() { - document.getElementById("save_before_quit").close(); -} - -function text_to_normalized_svg_path(text) { - text = text.slice(4).trim() // draw - .slice(4).trim() // path - .slice(1).trim() // d - .slice(1).trim(); // = - if (text.slice(0, 1) === '"') { - text = text.slice(1); - } - if (text.slice(-1) === '"') { - text = text.slice(0, -1); - } - text = pdgui.parse_svg_path(text); - return "draw path " + text.reduce(function (prev, curr) { - return prev.concat(curr) - }).join(" "); -} - -function text_to_fudi(text) { - text = text.trim(); - // special case for draw path d="arbitrary path string" ... - if (text.search(/^draw\s+path\s+d\s*=\s*"/) !== -1) { - text = text_to_normalized_svg_path(text); - } - text = text.replace(/(\$[0-9]+)/g, "\\$1"); // escape dollar signs - text = text.replace(/(\$@)/g, "\\$@"); // escape special $@ sign - text = text.replace(/(?!\\)(,|;)/g, " \\$1 "); // escape "," and ";" - text = text.replace(/\u0020+/g, " "); // filter consecutive ascii32 - return text; -} - -// Convert a string (FUDI message) to an array of strings small enough -// to fit in Pd's socketreceiver buffer of 4096 bytes -function string_to_array_of_chunks(msg) { - var chunk_max = 1024, - max_pd_string = 1000, - left, - in_array = [], - out_array = []; - if (msg.length <= chunk_max) { - out_array.push([msg]); - } else { - in_array = msg.split(/[\s\n]/); // split on newlines or spaces - while (in_array.length) { - left = in_array.slice(); // make a copy of in_array - if (left.toString().length > chunk_max) { - while (1) { - if (left.length < 2) { - pdgui.post("Warning: string truncated:"); - pdgui.post(left.toString(). - match(/............................../g).join("\n") - ); - break; - } - left = left.splice(0, left.length >> 1); - if (left.toString().length <= chunk_max) { - break; - } - } - } - // might need a check here for max_pd_string to warn - // user if a string is going to get truncated. (That's - // what max_pd_string is for above.) - out_array.push(left); - in_array = in_array.splice(left.length); - } - } - return out_array; -} - -// Super-simplistic guess at whether the string from the clipboard -// starts with Pd code. This is just meant as a convenience so that -// stuff in the copy buffer that obviously isn't Pd code doesn't get -// in the way when editing. -function might_be_a_pd_file(stuff_from_clipboard) { - var text = stuff_from_clipboard.trim(), - one = text.charAt(0), - two = text.charAt(1); - return (one === "#" && (two === "N" || two === "X")); -} - -function permission_to_paste_from_external_clipboard() { - return global.confirm(l("canvas.paste_clipboard_prompt")); -} - function nw_window_focus_callback(name) { pdgui.set_focused_patchwin(name); // on OSX, update the menu on focus @@ -132,28 +36,10 @@ function nw_window_zoom(name, delta) { } } -// These three functions need to be inside canvas_events closure -function canvas_find_whole_word(elem) { - canvas_events.match_words(elem.checked); -} - -function canvas_find_blur() { - canvas_events.normal(); -} - -function canvas_find_focus() { - var state = canvas_events.get_state(); - canvas_events.search(); -} - -function canvas_find_reset() { - canvas_events.find_reset(); -} - var canvas_events = (function() { var name, state, - scalar_draggables = {}, // elements of a scalar which have the "drag" event enabled + scalar_draggables = {}, // scalar child with the "drag" event enabled draggable_elem, // last scalar we dragged draggable_label, // kluge for handling [cnv] label/size anchors last_draggable_x, // last x coord for the element we're dragging @@ -188,6 +74,89 @@ var canvas_events = (function() { return 0; } }, + text_to_normalized_svg_path = function(text) { + text = text.slice(4).trim() // draw + .slice(4).trim() // path + .slice(1).trim() // d + .slice(1).trim(); // = + if (text.slice(0, 1) === '"') { + text = text.slice(1); + } + if (text.slice(-1) === '"') { + text = text.slice(0, -1); + } + text = pdgui.parse_svg_path(text); + return "draw path " + text.reduce(function (prev, curr) { + return prev.concat(curr) + }).join(" "); + }, + text_to_fudi = function(text) { + text = text.trim(); + // special case for draw path d="arbitrary path string" ... + if (text.search(/^draw\s+path\s+d\s*=\s*"/) !== -1) { + text = text_to_normalized_svg_path(text); + } + // escape dollar signs + text = text.replace(/(\$[0-9]+)/g, "\\$1"); + + // escape special $@ sign + text = text.replace(/(\$@)/g, "\\$@"); + + // escape "," and ";" + text = text.replace(/(?!\\)(,|;)/g, " \\$1 "); + + // filter consecutive ascii32 + text = text.replace(/\u0020+/g, " "); + return text; + }, + string_to_array_of_chunks = function(msg) { + // Convert a string (FUDI message) to an array of strings small enough + // to fit in Pd's socketreceiver buffer of 4096 bytes + var chunk_max = 1024, + max_pd_string = 1000, + left, + in_array = [], + out_array = []; + if (msg.length <= chunk_max) { + out_array.push([msg]); + } else { + in_array = msg.split(/[\s\n]/); // split on newlines or spaces + while (in_array.length) { + left = in_array.slice(); // make a copy of in_array + if (left.toString().length > chunk_max) { + while (1) { + if (left.length < 2) { + pdgui.post("Warning: string truncated:"); + pdgui.post(left.toString(). + match(/............................../g).join("\n") + ); + break; + } + left = left.splice(0, left.length >> 1); + if (left.toString().length <= chunk_max) { + break; + } + } + } + // might need a check here for max_pd_string to warn + // user if a string is going to get truncated. (That's + // what max_pd_string is for above.) + out_array.push(left); + in_array = in_array.splice(left.length); + } + } + return out_array; + }, + might_be_a_pd_file = function(stuff_from_clipboard) { + // Super-simplistic guess at whether the string from the clipboard + // starts with Pd code. This is just meant as a convenience so that + // stuff in the copy buffer that obviously isn't Pd code doesn't get + // in the way when editing. + var text = stuff_from_clipboard.trim(), + one = text.charAt(0), + two = text.charAt(1); + return (one === "#" && (two === "N" || two === "X")); + }, grow_svg_for_element= function(elem) { // See if an element overflows the svg bbox, and // enlarge the svg to accommodate it. @@ -683,175 +652,6 @@ var canvas_events = (function() { } ; - // Dialog events -- these are set elsewhere now because of a bug - // with nwworkingdir - document.querySelector("#saveDialog").addEventListener("change", - function(evt) { - pdgui.saveas_callback(name, evt.target.value, 0); - // reset value so that we can open the same file twice - evt.target.value = null; - console.log("tried to save something"); - }, false - ); - - // Whoa-- huge workaround! - // Right now we're getting the popup menu the way Pd Vanilla does it: - // 1) send a mouse(down) message to Pd - // 2) Pd checks whether it wants to send us a popup - // 3) Pd checks what popup menu items are available for this object/canvas - // 4) Pd sends GUI back a message with this info - // 5) GUI finally displays the popup - // 6) GUI keeps a _global_ _variable_ to remember where the popup coords - // 7) User clicks an option in the popup - // 8) GUI sends a message back to Pd with the popup index and coords - // 9) Pd walks the linked list of objects to look up the object - // 10) Pd asks that object if it reacts to popups, and if it reacts to the - // selected item in the popup - // 11) Pd sends a message to the relevant object for the item in question - // nw.js has a nice little "contextmenu" event handler, but it's too - // difficult to use when we're passing between GUI and Pd (twice). In the - // future we should just do all the popup menu event handling in the GUI, - // and only pass a message to Pd when the user has clicked an item. - // For now, however, we just turn off its default behavior and control - // it with a bunch of complicated callbacks. :( - document.addEventListener("contextmenu", function(evt) { - console.log("got a context menu evt..."); - evt.preventDefault(); - }); - - // Cut event - document.addEventListener("cut", function(evt) { - // This event doesn't currently get called because the - // nw menubar receives the event and doesn't propagate - // to the DOM. But if we add the ability to toggle menubar - // display, we might need to rely on this listener. - pdgui.pdsend(name, "cut"); - }); - - // Copy event - document.addEventListener("copy", function(evt) { - // On OSX, this event gets triggered when we're editing - // inside an object/message box. So we only forward the - // copy message to Pd if we're in a "normal" canvas state - if (canvas_events.get_state() === "normal") { - pdgui.pdsend(name, "copy"); - } - }); - - // Listen to paste event - // XXXTODO: Not sure whether this is even needed any more, as the - // paste-from-clipboard functionality has been moved to its own menu - // option. So this code may possibly be removed in the future. -ag - document.addEventListener("paste", function(evt) { - if (canvas_events.get_state() !== "normal") { - return; - } - // Send a canvas "paste" message to Pd - pdgui.pdsend(name, "paste"); - }); - - // MouseWheel event for zooming - document.addEventListener("wheel", function(evt) { - var d = { deltaX: 0, deltaY: 0, deltaZ: 0 }; - Object.keys(d).forEach(function(key) { - if (evt[key] < 0) { - d[key] = -1; - } else if (evt[key] > 0) { - d[key] = 1; - } else { - d[key] = 0; - } - }); - if (pdgui.cmd_or_ctrl_key(evt)) { - // scroll up for zoom-in, down for zoom-out - nw_window_zoom(name, -d.deltaY); - } - // Send a message on to Pd for the [mousewheel] legacy object - // (in the future we can refcount if we want to prevent forwarding - // these messages when there's no extant receiver) - pdgui.pdsend(name, "legacy_mousewheel", d.deltaX, d.deltaY, d.deltaZ); - }); - - // The following is commented out because we have to set the - // event listener inside nw_create_pd_window_menus due to a - // bug with nwworkingdir - - //document.querySelector("#fileDialog").addEventListener("change", - // function(evt) { - // var file_array = this.value; - // // reset value so that we can open the same file twice - // this.value = null; - // pdgui.menu_open(file_array); - // console.log("tried to open something\n\n\n\n\n\n\n\n"); - // }, false - //); - document.querySelector("#openpanel_dialog").addEventListener("change", - function(evt) { - var file_string = evt.target.value; - // reset value so that we can open the same file twice - evt.target.value = null; - pdgui.file_dialog_callback(file_string); - console.log("tried to openpanel something"); - }, false - ); - document.querySelector("#savepanel_dialog").addEventListener("change", - function(evt) { - var file_string = evt.target.value; - // reset value so that we can open the same file twice - evt.target.value = null; - pdgui.file_dialog_callback(file_string); - console.log("tried to savepanel something"); - }, false - ); - document.querySelector("#canvas_find_text").addEventListener("focusin", - canvas_find_focus, false - ); - - // disable drag and drop for the time being - window.addEventListener("dragover", function (evt) { - evt.preventDefault(); - }, false); - window.addEventListener("drop", function (evt) { - evt.preventDefault(); - }, false); - - // Add placeholder text... this all needs to be collected into an - // add_events function similiar to the one in index.js - document.querySelector("#canvas_find_text").placeholder = - l("canvas.find.placeholder"); - document.querySelector("#canvas_find_text").addEventListener("blur", - canvas_find_blur, false - ); - document.querySelector("#canvas_find_button").addEventListener("click", - events.find_click); - // We need to separate these into nw_window events and html5 DOM events - // closing the Window this isn't actually closing the window yet - gui.Window.get().on("close", function() { - pdgui.pdsend(name, "menuclose 0"); - }); - // update viewport size when window size changes - gui.Window.get().on("maximize", function() { - pdgui.gui_canvas_get_scroll(name); - }); - gui.Window.get().on("unmaximize", function() { - pdgui.gui_canvas_get_scroll(name); - }); - gui.Window.get().on("resize", function() { - pdgui.gui_canvas_get_scroll(name); - }); - gui.Window.get().on("focus", function() { - nw_window_focus_callback(name); - }); - gui.Window.get().on("blur", function() { - nw_window_blur_callback(name); - }); - gui.Window.get().on("move", function(x, y) { - var w = gui.Window.get(); - pdgui.pdsend(name, "setbounds", x, y, x + w.width, y + w.height); - }); - // set minimum window size - gui.Window.get().setMinimumSize(150, 100); - return { none: function() { var evt_name, prop; @@ -868,7 +668,7 @@ var canvas_events = (function() { } }, normal: function() { - this.none(); + canvas_events.none(); document.addEventListener("mousemove", events.mousemove, false); document.addEventListener("keydown", events.keydown, false); @@ -895,14 +695,14 @@ var canvas_events = (function() { // The exception is my_canvas, which is weird because the visible // rectangle extends past the bbox that it reports to Pd. // Unfortunately that means a lot of work to treat it separately. - this.none(); + canvas_events.none(); document.addEventListener("mousemove", events.iemgui_label_mousemove, false); document.addEventListener("mouseup", events.iemgui_label_mouseup, false); }, text: function() { - this.none(); + canvas_events.none(); document.addEventListener("mousemove", events.text_mousemove, false); document.addEventListener("keydown", events.text_keydown, false); @@ -915,8 +715,8 @@ var canvas_events = (function() { set_edit_menu_modals(false); }, floating_text: function() { - this.none(); - this.text(); + canvas_events.none(); + canvas_events.text(); document.removeEventListener("mousedown", events.text_mousedown, false); document.removeEventListener("mouseup", events.text_mouseup, false); document.removeEventListener("keypress", events.text_keypress, false); @@ -928,7 +728,7 @@ var canvas_events = (function() { set_edit_menu_modals(false); }, dropdown_menu: function() { - this.none(); + canvas_events.none(); document.addEventListener("mousedown", events.dropdown_menu_mousedown, false); document.addEventListener("mouseup", events.dropdown_menu_mouseup, false); document.addEventListener("mousemove", events.dropdown_menu_mousemove, false); @@ -938,7 +738,7 @@ var canvas_events = (function() { .addEventListener("wheel", events.dropdown_menu_wheel, false); }, search: function() { - this.none(); + canvas_events.none(); document.addEventListener("keydown", events.find_keydown, false); state = "search"; }, @@ -985,6 +785,189 @@ var canvas_events = (function() { close_without_saving: function(cid, force) { pdgui.pdsend(name, "dirty 0"); pdgui.pdsend(cid, "menuclose", force); + }, + close_save_dialog: function() { + document.getElementById("save_before_quit").close(); + }, + init: function() { + document.getElementById("saveDialog") + .setAttribute("nwworkingdir", pdgui.get_pwd()); + document.getElementById("fileDialog") + .setAttribute("nwworkingdir", pdgui.get_pwd()); + document.getElementById("fileDialog").setAttribute("accept", + Object.keys(pdgui.pd_filetypes).toString()); + // Dialog events -- these are set elsewhere now because of a bug + // with nwworkingdir + document.querySelector("#saveDialog").addEventListener("change", + function(evt) { + pdgui.saveas_callback(name, evt.target.value, 0); + // reset value so that we can open the same file twice + evt.target.value = null; + console.log("tried to save something"); + }, false + ); + // Whoa-- huge workaround! Right now we're getting + // the popup menu the way Pd Vanilla does it: + // 1) send a mouse(down) message to Pd + // 2) Pd checks whether it wants to send us a popup + // 3) Pd checks what popup menu items are available for obj/canvas + // 4) Pd sends GUI back a message with this info + // 5) GUI finally displays the popup + // 6) GUI keeps a _global_ _variable_ to remember the popup coords + // 7) User clicks an option in the popup + // 8) GUI sends a message back to Pd with the popup index and coords + // 9) Pd walks the linked list of objects to look up the object + // 10) Pd asks the object if it reacts to popups, and if it reacts + // to the selected item in the popup + // 11) Pd sends a message to the relevant object for the item in + // question + // nw.js has a nice little "contextmenu" event handler, but it's too + // difficult to use when passing between GUI and Pd (twice). In the + // future we should do all popup menu event handling in the GUI, + // and only pass a message to Pd when the user has clicked an item. + // For now, however, we just turn off its default behavior and + // control it with a bunch of complicated callbacks. + document.addEventListener("contextmenu", function(evt) { + console.log("got a context menu evt..."); + evt.preventDefault(); + }); + + // Cut event + document.addEventListener("cut", function(evt) { + // This event doesn't currently get called because the + // nw menubar receives the event and doesn't propagate + // to the DOM. But if we add the ability to toggle menubar + // display, we might need to rely on this listener. + pdgui.pdsend(name, "cut"); + }); + + // Copy event + document.addEventListener("copy", function(evt) { + // On OSX, this event gets triggered when we're editing + // inside an object/message box. So we only forward the + // copy message to Pd if we're in a "normal" canvas state + if (canvas_events.get_state() === "normal") { + pdgui.pdsend(name, "copy"); + } + }); + + // Listen to paste event + // XXXTODO: Not sure whether this is even needed any more, as the + // paste-from-clipboard functionality has been moved to its own menu + // option. So this code may possibly be removed in the future. -ag + document.addEventListener("paste", function(evt) { + if (canvas_events.get_state() !== "normal") { + return; + } + // Send a canvas "paste" message to Pd + pdgui.pdsend(name, "paste"); + }); + + // MouseWheel event for zooming + document.addEventListener("wheel", function(evt) { + var d = { deltaX: 0, deltaY: 0, deltaZ: 0 }; + Object.keys(d).forEach(function(key) { + if (evt[key] < 0) { + d[key] = -1; + } else if (evt[key] > 0) { + d[key] = 1; + } else { + d[key] = 0; + } + }); + if (pdgui.cmd_or_ctrl_key(evt)) { + // scroll up for zoom-in, down for zoom-out + nw_window_zoom(name, -d.deltaY); + } + // Send a message on to Pd for the [mousewheel] legacy object + // (in the future we can refcount to prevent forwarding + // these messages when there's no extant receiver) + pdgui.pdsend(name, "legacy_mousewheel", + d.deltaX, d.deltaY, d.deltaZ); + }); + + // The following is commented out because we have to set the + // event listener inside nw_create_pd_window_menus due to a + // bug with nwworkingdir + + //document.querySelector("#fileDialog").addEventListener("change", + // function(evt) { + // var file_array = this.value; + // // reset value so that we can open the same file twice + // this.value = null; + // pdgui.menu_open(file_array); + // console.log("tried to open something\n\n\n\n\n\n\n\n"); + // }, false + //); + document.querySelector("#openpanel_dialog") + .addEventListener("change", function(evt) { + var file_string = evt.target.value; + // reset value so that we can open the same file twice + evt.target.value = null; + pdgui.file_dialog_callback(file_string); + console.log("tried to openpanel something"); + }, false + ); + document.querySelector("#savepanel_dialog") + .addEventListener("change", function(evt) { + var file_string = evt.target.value; + // reset value so that we can open the same file twice + evt.target.value = null; + pdgui.file_dialog_callback(file_string); + console.log("tried to savepanel something"); + }, false + ); + document.querySelector("#canvas_find_text") + .addEventListener("focusin", canvas_events.search, false + ); + + // disable drag and drop for the time being + window.addEventListener("dragover", function (evt) { + evt.preventDefault(); + }, false); + window.addEventListener("drop", function (evt) { + evt.preventDefault(); + }, false); + + // Add placeholder text... this all needs to be collected into an + // add_events function similiar to the one in index.js + document.querySelector("#canvas_find_text").placeholder = + l("canvas.find.placeholder"); + document.querySelector("#canvas_find_text").addEventListener("blur", + canvas_events.normal, false + ); + document.querySelector("#canvas_find_button") + .addEventListener("click", events.find_click + ); + // We need to separate these into nw_window events and html5 DOM + // events closing the Window this isn't actually closing the window + // yet + gui.Window.get().on("close", function() { + pdgui.pdsend(name, "menuclose 0"); + }); + // update viewport size when window size changes + gui.Window.get().on("maximize", function() { + pdgui.gui_canvas_get_scroll(name); + }); + gui.Window.get().on("unmaximize", function() { + pdgui.gui_canvas_get_scroll(name); + }); + gui.Window.get().on("resize", function() { + pdgui.gui_canvas_get_scroll(name); + }); + gui.Window.get().on("focus", function() { + nw_window_focus_callback(name); + }); + gui.Window.get().on("blur", function() { + nw_window_blur_callback(name); + }); + gui.Window.get().on("move", function(x, y) { + var w = gui.Window.get(); + pdgui.pdsend(name, "setbounds", x, y, + x + w.width, y + w.height); + }); + // set minimum window size + gui.Window.get().setMinimumSize(150, 100); } }; }()); @@ -1022,6 +1005,8 @@ function register_window_id(cid, attr_array) { } create_popup_menu(cid); canvas_events.register(cid); + // Initialize global DOM state/events + canvas_events.init(document); translate_form(); // Trigger a "focus" event so that OSX updates the menu for this window nw_window_focus_callback(cid); @@ -1157,11 +1142,6 @@ function canvas_paste_from_clipboard(name, clipboard_data) return; } - // Maybe we want a warning prompt here? Then uncomment the line below. I - // disabled this for now, as the paste-from-clipboard command now has its - // own menu option, so presumably the user knows what he's doing. -ag - //if (!permission_to_paste_from_external_clipboard()) return; - // clear the buffer pdgui.pdsend(name, "copyfromexternalbuffer"); pd_message = ""; diff --git a/pd/nw/pdgui.js b/pd/nw/pdgui.js index e40a3072c64c9bfdede723e6f138b894c7714e40..97d0fa3a013323cee5624eadf55911722285334c 100644 --- a/pd/nw/pdgui.js +++ b/pd/nw/pdgui.js @@ -1995,7 +1995,12 @@ function add_gobj_to_svg(svg, gobj) { var gui = (function() { var c = {}; // object to hold references to all our canvas closures - var last_thing; // last thing we got + // We store the last "thing" we fetched from the window. This is either + // the window itself or a "gobj". Regular old DOM elements that aren't + // a "gobj" container don't count. This way we can do a "get_gobj" then + // gang multiple element queries after it that work within our last + // "gobj." (Same for window.) + var last_thing; var null_fn, null_canvas; var create_canvas = function(cid, w) { var get = function(parent, sel, arg, suffix) { @@ -4152,7 +4157,8 @@ function gui_grid_point(cid, tag, x, y) { } // mknob from moonlib -function gui_mknob_new(cid, tag, x, y, is_toplevel, show_in, show_out) { +function gui_mknob_new(cid, tag, x, y, is_toplevel, show_in, show_out, + is_footils_knob) { gui(cid).get_elem("patchsvg", function(svg_elem) { gui_gobj_new(cid, tag, "obj", x, y, is_toplevel); }); @@ -4162,50 +4168,239 @@ function gui_mknob_new(cid, tag, x, y, is_toplevel, show_in, show_out) { class: "border" // now we can inherit the css border styles }), circle = create_item(cid, "circle", { - class: "circle" + //class: "circle" }), line = create_item(cid, "line", { - class: "dial" + //class: "dial" }); frag.appendChild(border); frag.appendChild(circle); + /* An extra circle for footils/knob */ + if (!!is_footils_knob) { + frag.appendChild(create_item(cid, "circle", { + class: "dial_frag" + })); + } frag.appendChild(line); return frag; }); } -function gui_configure_mknob(cid, tag, size, bg_color, fg_color) { - gui(cid).get_gobj(tag) +function knob_dashes(d, len) { + var c = d * 3.14159; + return (c * len) + " " + (c * (1 - len)); +} + +function knob_offset(d) { + return d * 3.14 * -0.28; +} + +function gui_configure_mknob(cid, tag, size, bg_color, fg_color, + is_footils_knob) { + var w = size, + h = !!is_footils_knob ? size + 5 : size; + var g = gui(cid).get_gobj(tag) .q(".border", { - d: ["M", 0, 0, size, 0, - "M", 0, size, size, size, - "M", 0, 0, 0, size, - "M", size, 0, size, size + d: ["M", 0, 0, w, 0, + "M", 0, h, w, h, + "M", 0, 0, 0, h, + "M", w, 0, w, h ].join(" "), fill: "none", }) - .q(".circle", { + .q("circle", { cx: size / 2, cy: size / 2, r: size / 2, - fill: bg_color, + fill: !!is_footils_knob ? "none" : bg_color, stroke: "black", - "stroke-width": 1 + "stroke-width": !!is_footils_knob ? 3 : 1, + "stroke-dasharray": !!is_footils_knob ? + knob_dashes(size, 0.94) : "none", + "stroke-dashoffset": !!is_footils_knob ? knob_offset(size) : "0" }) - .q(".dial", { + .q("line", { // indicator "stroke-width": 2, stroke: fg_color }); + + if (!!is_footils_knob) { + g.q(".dial_frag", { + cx: size / 2, + cy: size / 2, + r: size / 2, + fill: "none", + stroke: bg_color, + "stroke-width": 3, + "stroke-dasharray": knob_dashes(size, 0.94), + "stroke-dashoffset": knob_offset(size) + }); + } } -function gui_turn_mknob(cid, tag, x1, y1, x2, y2) { - gui(cid).get_gobj(tag) - .q(".dial", { +function gui_turn_mknob(cid, tag, x1, y1, x2, y2, is_footils_knob, val) { + var g = gui(cid).get_gobj(tag) + .q("line", { // indicator x1: x1, y1: y1, x2: x2, y2: y2 }); + if (!!is_footils_knob) { + g.q(".dial_frag", { + "stroke-dasharray": knob_dashes(x1 * 2, val * 0.94) + }); + } +} + +// room_sim_2d and room_sim_3d objects from iemlib +function gui_room_sim_new(cid, tag, x, y, w, h, is_toplevel) { + gui(cid).get_elem("patchsvg", function(svg_elem) { + gui_gobj_new(cid, tag, "obj", x, y, is_toplevel); + }); + gui(cid).get_gobj(tag) + .append(function(frag) { +// frag.appendChild(line); + return frag; + }); +} + +function gui_room_sim_map(cid, tag, w, h, rad, head, xpix, ypix, fontsize, + fcol, bcol, src_array, r3d) { + gui(cid).get_gobj(tag, function(e) { + gui_text_draw_border(cid, tag, 0, 0, w, h); + // Set the style for the background directly... otherwise the + // default theme bgcolor will be used + e.querySelector(".border").style.fill = bcol; + }) + .append(function(frag) { + var x1 = xpix - rad, + x2 = xpix + rad - 1, + y1 = ypix - rad, + y2 = ypix + rad - 1, + dx = -((rad * Math.sin(head * 0.0174533) + 0.49999)|0), + dy = -((rad * Math.cos(head * 0.0174533) + 0.49999)|0), + i, + text; + for (i = 0; i < src_array.length; i++) { + text = create_item(cid, "text", { + x: src_array[i][0], + y: src_array[i][1], + fill: src_array[i][2], + "font-size": fontsize, + "dominant-baseline": "middle" + }); + text.textContent = (i + 1).toString(); + frag.appendChild(text); + } + var ellipse = create_item(cid, "ellipse", { + cx: (x2 - x1) * 0.5 + x1, + cy: (y2 - y1) * 0.5 + y1, + rx: (x2 - x1) * 0.5, + ry: (y2 - y1) * 0.5, + "stroke-width": 1, + "stroke": fcol, + "fill": "none" + }), + ellipse2 = create_item(cid, "ellipse", { + // for room_sim_3d + cx: r3d ? (r3d[2] - r3d[0]) * 0.5 + r3d[0] : 0, + cy: r3d ? (r3d[3] - r3d[1]) * 0.5 + r3d[1] : 0, + rx: r3d ? (r3d[2] - r3d[0]) * 0.5 : 0, + ry: r3d ? (r3d[3] - r3d[1]) * 0.5 : 0, + "stroke-width": 1, + stroke: fcol, + fill: "none" + }), + line = create_item(cid, "line", { + x1: xpix, + y1: ypix, + x2: xpix + dx, + y2: ypix + dy, + "stroke-width": 3, + stroke: fcol + }); + frag.appendChild(ellipse); + frag.appendChild(ellipse2); + frag.appendChild(line); + return frag; + }) +} + +function gui_room_sim_update_src(cid, tag, i, x, y, font_size, col) { + gui(cid).get_gobj(tag, function(e) { + var a = e.querySelectorAll("text"); + if (a.length && i < a.length) { + configure_item(a[i], { + x: x, + y: y, + "font-size": font_size, + fill: col + }); + } + }); +} + +function gui_room_sim_update(cid, tag, x0, y0, dx, dy, pixrad) { + gui(cid).get_gobj(tag) + .q("line", { + x1: x0, + y1: y0, + x2: x0 + dx, + y2: y0 + dy + }) + .q("ellipse", { + rx: ((x0 + pixrad - 1) - (x0 - pixrad)) * 0.5, + ry: ((y0 + pixrad - 1) - (y0 - pixrad)) * 0.5, + cx: ((x0 + pixrad - 1) - (x0 - pixrad)) * 0.5 + (x0 - pixrad), + cy: ((y0 + pixrad - 1) - (y0 - pixrad)) * 0.5 + (y0 - pixrad), + }); +} + +// for room_sim_3d +function gui_room_sim_head2(cid, tag, x1, y1, x2, y2) { + gui(cid).get_gobj(tag, function(e) { + configure_item(e.querySelectorAll("ellipse")[1], { + rx: (x2 - x1) * 0.5, + ry: (y2 - y1) * 0.5, + cx: (x2 - x1) * 0.5 + x1, + cy: (y2 - y1) * 0.5 + y1 + }); + }); +} + +function gui_room_sim_fontsize(cid, tag, i, size) { + gui(cid).get_gobj(tag, function(e) { + var i, a; + a = e.querySelectorAll("text"); + if (a.length) { + for (i = 0; i < a.length; i++) { + configure_item(a[i], { + "font-size": size + }); + } + } + }); +} + +// for the dial thingy +function gui_room_sim_colors(cid, tag, fg, bg) { + gui(cid).get_gobj(tag) + .q("ellipse", { + stroke: fg + }) + .q("line", { + stroke: fg + }) + .q(".border", function(e) { + e.style.fill = bg; + }); +} + +function gui_room_sim_erase(cid, tag) { + gui(cid).get_gobj(tag, function(e) { + e.innerHTML = ""; + }); } function add_popup(cid, popup) { diff --git a/pd/src/g_all_guis.c b/pd/src/g_all_guis.c index faaf731a41d081acede7293a0fd4f4d5642de81d..c2381e0f12f3d6d283868cf82e2c936143a31ef7 100644 --- a/pd/src/g_all_guis.c +++ b/pd/src/g_all_guis.c @@ -513,7 +513,6 @@ void iemgui_displace_withtag(t_gobj *z, t_glist *glist, int dx, int dy) canvas_fixlinesfor(glist_getcanvas(glist), (t_text *)z); } - void iemgui_select(t_gobj *z, t_glist *glist, int selected) { t_iemgui *x = (t_iemgui *)z; diff --git a/pd/src/s_print.c b/pd/src/s_print.c index 19d1760cb0b40e67c5bc9525af06cb763153e532..e00b814f3c25eadb37c3c08bb43e19f1a173c338 100644 --- a/pd/src/s_print.c +++ b/pd/src/s_print.c @@ -14,6 +14,7 @@ #endif t_printhook sys_printhook; +t_printhook sys_printhook_error; int sys_printtostderr; /* escape characters for tcl/tk */ @@ -51,11 +52,13 @@ static void doerror(const void *object, const char *s) char upbuf[MAXPDSTRING]; upbuf[MAXPDSTRING-1]=0; - // what about sys_printhook_error ? - if (sys_printhook) + if (sys_printhook || sys_printhook_error) { snprintf(upbuf, MAXPDSTRING-1, "error: %s", s); - (*sys_printhook)(upbuf); + if (sys_printhook_error) + (*sys_printhook_error)(upbuf); + if (sys_printhook) + (*sys_printhook)(upbuf); } else if (sys_printtostderr) fprintf(stderr, "error: %s", s); diff --git a/pd/src/x_interface.c b/pd/src/x_interface.c index 63223bb9b2d641c05b403e9ee9a3405d9e6e734c..34307dbeaa41713f71b1eca929e0668d6ecfc3e9 100644 --- a/pd/src/x_interface.c +++ b/pd/src/x_interface.c @@ -64,7 +64,7 @@ static void *print_new(t_symbol *sel, int argc, t_atom *argv) static void print_bang(t_print *x) { - if (sys_nogui) + if (sys_nogui || sys_printhook) post("%s%sbang", x->x_sym->s_name, (*x->x_sym->s_name ? ": " : "")); else { @@ -78,7 +78,7 @@ static void print_bang(t_print *x) static void print_pointer(t_print *x, t_gpointer *gp) { - if (sys_nogui) + if (sys_nogui || sys_printhook) post("%s%s(gpointer)", x->x_sym->s_name, (*x->x_sym->s_name ? ": " : "")); else @@ -93,7 +93,7 @@ static void print_pointer(t_print *x, t_gpointer *gp) static void print_float(t_print *x, t_floatarg f) { - if (sys_nogui) + if (sys_nogui || sys_printhook) post("%s%s" FLOAT_SPECIFIER, x->x_sym->s_name, (*x->x_sym->s_name ? ": " : ""), f); else @@ -108,7 +108,7 @@ static void print_float(t_print *x, t_floatarg f) static void print_symbol(t_print *x, t_symbol *s) { - if (sys_nogui) + if (sys_nogui || sys_printhook) post("%s%s%s", x->x_sym->s_name, (*x->x_sym->s_name ? ": " : ""), s->s_name); else @@ -126,7 +126,7 @@ static void print_anything(t_print *x, t_symbol *s, int argc, t_atom *argv) { char buf[MAXPDSTRING]; t_atom at; - if (sys_nogui) + if (sys_nogui || sys_printhook) { startpost("%s%s", x->x_sym->s_name, (*x->x_sym->s_name ? ":" : "")); if (s && (s != &s_list || (argc && argv->a_type != A_FLOAT))) @@ -164,6 +164,134 @@ static void print_setup(void) class_addanything(print_class, print_anything); } +/* -------------------------- unpost ------------------------------ */ + +#define NEWBUFSIZE MAXPDSTRING * 2 +static t_class *unpost_class; + +typedef struct _unpost { + t_object x_obj; + t_outlet *x_out0; + t_outlet *x_out1; + int x_all; +} t_unpost; + +typedef struct _unpost_frame { + t_unpost *u_self; + char *u_buf; + unsigned int u_bufsize; + unsigned int u_index; +} t_unpost_frame; + +static t_unpost_frame *current_unpost; + +static void *unpost_frame_new(t_unpost *x) +{ + t_unpost_frame *u = (t_unpost_frame *)t_getbytes(sizeof(t_unpost_frame)); + u->u_self = x; + u->u_buf = (char *)t_getbytes(sizeof(char) * NEWBUFSIZE); + u->u_buf[0] = '\0'; + u->u_bufsize = NEWBUFSIZE; + u->u_index = 0; + return u; +} + +static void unpost_frame_resize(t_unpost_frame *u, int newsize) +{ + u->u_buf = (char *)t_resizebytes(u->u_buf, sizeof(*u->u_buf) * u->u_bufsize, + sizeof(*u->u_buf) * newsize); + u->u_bufsize = newsize; +} + +static void unpost_frame_free(t_unpost_frame* u) +{ + t_freebytes(u->u_buf, sizeof(*u->u_buf)); + u->u_buf = NULL; +} + +static void *unpost_new(t_symbol *s) +{ + t_unpost *x = (t_unpost *)pd_new(unpost_class); + x->x_all = s != gensym("error"); + x->x_out0 = outlet_new(&x->x_obj, &s_symbol); + x->x_out1 = outlet_new(&x->x_obj, &s_symbol); + return x; +} + +extern t_printhook sys_printhook; +extern t_printhook sys_printhook_error; +static void unpost_printhook(const char *s) { + t_unpost *x = current_unpost->u_self; + /* does strlen include terminator? can't remember... */ + unsigned int needbytes = strlen(current_unpost->u_buf) + strlen(s) + 1; + if (needbytes >= current_unpost->u_bufsize) + { + /* Raise the bufsize to NEWBUFSIZE * n where n is a power of two that + will fit the number of bytes we need) */ + unsigned int n = ((needbytes / NEWBUFSIZE) >> 1) + + (needbytes % NEWBUFSIZE != 0); + unpost_frame_resize(current_unpost, + current_unpost->u_bufsize * (1 << n)); + } + + else if (strlen(current_unpost->u_buf) + strlen(s) < + current_unpost->u_bufsize / 4) + { + /* Pretty sure I prematurely optimized here. There's no good reason to + shrink the buffer in the midst of receiving some enormous number of + posts from an object chain. Besides, we free the buffer once the + object chain downstream from the [unpost] outlet finishes computing. + Anyhow, here we are. */ + unpost_frame_resize(current_unpost, + (current_unpost->u_bufsize / 4 < NEWBUFSIZE) ? + NEWBUFSIZE : + current_unpost->u_bufsize / 4); + } + strcat(current_unpost->u_buf, s); + const char *p; + const char *d = current_unpost->u_buf, *dd = d; + for (;;) { + p = strchr(d, '\n'); + if (!p) break; + /* To avoid infinite recursion, let's reset our hooks before we + send to the right outlet. */ + t_printhook print_backup = sys_printhook, + error_backup = sys_printhook_error; + sys_printhook = NULL; + sys_printhook_error = NULL; + outlet_symbol(x->x_out0, gensym(d)); + sys_printhook = print_backup; + sys_printhook_error = error_backup; + d = p + 1; + } + if (d != dd) + { + char *q = strdup(d); + strcpy(current_unpost->u_buf, q); + t_freebytes(q, sizeof(*q) * strlen(q)); + } +} + +static void unpost_anything(t_unpost *x, t_symbol *s, int argc, t_atom *argv) +{ + t_printhook *myhook = x->x_all ? &sys_printhook : &sys_printhook_error; + t_printhook backup1 = *myhook; + t_unpost_frame *backup2 = current_unpost; + *myhook = unpost_printhook; + current_unpost = unpost_frame_new(x); + outlet_anything(x->x_out1, s, argc, argv); + *myhook = backup1; + unpost_frame_free(current_unpost); + current_unpost = backup2; +} + +static void unpost_setup(void) +{ + unpost_class = class_new(gensym("unpost"), (t_newmethod)unpost_new, 0, + sizeof(t_unpost), 0, A_DEFSYM, 0); + class_addanything(unpost_class, unpost_anything); +} + /* --------------pdinfo, canvasinfo, classinfo, objectinfo --------------- */ static t_class *canvasinfo_class; @@ -1497,6 +1625,7 @@ void objectinfo_setup(void) void x_interface_setup(void) { print_setup(); + unpost_setup(); canvasinfo_setup(); pdinfo_setup(); classinfo_setup(); diff --git a/scripts/regression_tests.pd b/scripts/regression_tests.pd index 6b7af932edfb94f49b3c6f47709cdb7a20d3cb9d..d691cc83e49212a6574072dc3a9b7d4fa4f4e2bf 100644 --- a/scripts/regression_tests.pd +++ b/scripts/regression_tests.pd @@ -34,10 +34,13 @@ is handy for some binbuf tests.; #X text 536 150 <- we have to escape the arg; #X text 556 190 escape it in a comment.; #X text 556 170 in bash but we can't; -#X obj 521 374 spigot; -#X obj 537 440 route 1; -#X obj 537 495 print success; -#X obj 145 353 rtest makefilename_percent_parsing; +#X obj 391 374 spigot; +#X obj 407 440 route 1; +#X obj 407 495 print success; +#X obj 145 358 rtest unpost_sanity; +#X obj 145 414 rtest unpost_error; +#X obj 145 465 rtest unpost_print; +#X obj 145 516 rtest unpost_long_message; #X connect 0 0 1 0; #X connect 1 0 7 0; #X connect 1 1 29 0; @@ -64,3 +67,6 @@ is handy for some binbuf tests.; #X connect 29 0 7 0; #X connect 30 0 31 0; #X connect 30 1 2 0; +#X connect 32 0 33 0; +#X connect 33 0 34 0; +#X connect 34 0 35 0; diff --git a/scripts/regression_tests/msg_click.pd b/scripts/regression_tests/msg_click.pd index 4e715320f4466cf6f1c6da41afdcf1db8b072c22..49e54fefb47eede644b59277bcdf5f50ab70eb93 100644 --- a/scripts/regression_tests/msg_click.pd +++ b/scripts/regression_tests/msg_click.pd @@ -1,4 +1,4 @@ -#N canvas 34 75 582 396 12; +#N canvas 37 104 582 396 12; #X obj 36 25 inlet; #X msg 152 97 click 0 0 0 0 0; #X msg 152 122 \$1; diff --git a/scripts/regression_tests/unpost_error.pd b/scripts/regression_tests/unpost_error.pd new file mode 100644 index 0000000000000000000000000000000000000000..f799db61968cd04572aa60617382a28e69aa0211 --- /dev/null +++ b/scripts/regression_tests/unpost_error.pd @@ -0,0 +1,29 @@ +#N canvas 170 138 650 396 12; +#X obj 36 25 inlet; +#X obj 36 349 outlet; +#X msg 36 63 unknown_method; +#X obj 61 126 unpost error; +#X obj 36 226 list; +#X obj 36 251 route bang; +#X text 155 16 Unpost takes an argument "error" to set it to only redirect +error messages. Here we make sure it redirects an error message.; +#X obj 36 88 trigger bang anything bang; +#X obj 142 181 float; +#X text 189 181 <- trigger an error; +#X obj 36 276 f 0; +#X msg 103 276 1; +#X obj 36 309 list append unpost with "error" argument should redirect +an error message; +#X connect 0 0 2 0; +#X connect 2 0 7 0; +#X connect 3 0 4 1; +#X connect 3 1 8 0; +#X connect 4 0 5 0; +#X connect 5 0 10 0; +#X connect 5 1 11 0; +#X connect 7 0 4 0; +#X connect 7 1 3 0; +#X connect 7 2 4 1; +#X connect 10 0 12 0; +#X connect 11 0 12 0; +#X connect 12 0 1 0; diff --git a/scripts/regression_tests/unpost_error2.pd b/scripts/regression_tests/unpost_error2.pd new file mode 100644 index 0000000000000000000000000000000000000000..a6aed2efc7b35231111a3b3ea89393493285fa59 --- /dev/null +++ b/scripts/regression_tests/unpost_error2.pd @@ -0,0 +1,31 @@ +#N canvas 170 138 650 396 12; +#X obj 36 25 inlet; +#X obj 36 349 outlet; +#X msg 36 63 unknown_method; +#X text 155 16 Unpost takes an argument "error" to set it to only redirect +error messages. Here we make sure it doesn't allow non-error messages +through; +#X obj 61 126 unpost error; +#X obj 142 211 print -n; +#X obj 36 226 list; +#X obj 36 88 trigger bang bang bang; +#X obj 36 251 route bang; +#X obj 36 309 list append unpost with "error" argument should not redirect +non-error messages; +#X obj 36 276 f 1; +#X msg 103 276 0; +#X msg 142 186 unpost_error test message; +#X connect 0 0 2 0; +#X connect 2 0 7 0; +#X connect 4 0 6 1; +#X connect 4 1 12 0; +#X connect 6 0 8 0; +#X connect 7 0 6 0; +#X connect 7 1 4 0; +#X connect 7 2 6 1; +#X connect 8 0 10 0; +#X connect 8 1 11 0; +#X connect 9 0 1 0; +#X connect 10 0 9 0; +#X connect 11 0 9 0; +#X connect 12 0 5 0; diff --git a/scripts/regression_tests/unpost_long_message.pd b/scripts/regression_tests/unpost_long_message.pd new file mode 100644 index 0000000000000000000000000000000000000000..e3738436dc55768a3a906ddf2984c63325fdeaa3 --- /dev/null +++ b/scripts/regression_tests/unpost_long_message.pd @@ -0,0 +1,22 @@ +#N canvas 86 182 696 396 12; +#X obj 36 25 inlet; +#X obj 36 349 outlet; +#X obj 36 266 f 1; +#X text 155 16 Unpost should be able to handle long messages. Here +we trigger one by sending a list of all the class names to the left +inlet of unpost.; +#X msg 36 63 classlist; +#X obj 36 88 unpost; +#X obj 75 116 pdinfo; +#X obj 75 148 print long_message; +#X obj 36 113 b; +#X obj 36 299 list append unpost should print long messages without +crashing; +#X connect 0 0 4 0; +#X connect 2 0 9 0; +#X connect 4 0 5 0; +#X connect 5 0 8 0; +#X connect 5 1 6 0; +#X connect 6 0 7 0; +#X connect 8 0 2 0; +#X connect 9 0 1 0; diff --git a/scripts/regression_tests/unpost_print.pd b/scripts/regression_tests/unpost_print.pd new file mode 100644 index 0000000000000000000000000000000000000000..7dda0dabec1bf919d9c7ba1c2d1a027b64cbfb7d --- /dev/null +++ b/scripts/regression_tests/unpost_print.pd @@ -0,0 +1,44 @@ +#N canvas 35 102 696 396 12; +#X obj 36 25 inlet; +#X obj 36 349 outlet; +#X msg 36 63 unknown_method; +#X obj 36 266 f 1; +#X obj 528 228 print unpost_print; +#X obj 378 146 unpost; +#X msg 528 146 testing unpost...; +#X obj 198 228 print unpost_print; +#X msg 198 146 testing unpost error...; +#X obj 58 146 unpost error; +#X obj 139 171 float; +#X obj 58 171 b; +#X msg 58 201 test message; +#X obj 36 88 trigger bang anything bang bang bang; +#X text 155 16 Unpost allows printing from the left outlet *without* +redirecting back to the outlet. This prevents infinite loops. Without +this protection this test would crash.; +#X obj 36 299 list append when unpost's left-outlet messages are printed +to the console an infinite loop should not occur; +#X obj 58 228 print -n; +#X obj 378 228 print -n; +#X obj 378 171 b; +#X msg 378 196 test message; +#X obj 417 171 print test; +#X connect 0 0 2 0; +#X connect 2 0 13 0; +#X connect 3 0 15 0; +#X connect 5 0 18 0; +#X connect 5 1 20 0; +#X connect 6 0 4 0; +#X connect 8 0 7 0; +#X connect 9 0 11 0; +#X connect 9 1 10 0; +#X connect 11 0 12 0; +#X connect 12 0 16 0; +#X connect 13 0 3 0; +#X connect 13 1 9 0; +#X connect 13 2 8 0; +#X connect 13 3 5 0; +#X connect 13 4 6 0; +#X connect 15 0 1 0; +#X connect 18 0 19 0; +#X connect 19 0 17 0; diff --git a/scripts/regression_tests/unpost_sanity.pd b/scripts/regression_tests/unpost_sanity.pd new file mode 100644 index 0000000000000000000000000000000000000000..d9fc5715abe85412cd5b510d5a3e2b958c1591e3 --- /dev/null +++ b/scripts/regression_tests/unpost_sanity.pd @@ -0,0 +1,28 @@ +#N canvas 164 80 650 396 12; +#X obj 36 25 inlet; +#X obj 36 309 outlet; +#X text 155 16 Unpost is used to redirect messages from the console. +We need it for some of the following tests \, so we make sure it operates +correctly here first.; +#X obj 61 136 unpost; +#X msg 36 63 unknown_method; +#X obj 100 166 float; +#X obj 36 166 list; +#X obj 36 88 trigger bang anything bang; +#X obj 36 191 route bang; +#X obj 36 269 list append unpost should redirect an error message; +#X msg 103 225 1; +#X obj 36 226 f 0; +#X connect 0 0 4 0; +#X connect 3 0 6 1; +#X connect 3 1 5 0; +#X connect 4 0 7 0; +#X connect 6 0 8 0; +#X connect 7 0 6 0; +#X connect 7 1 3 0; +#X connect 7 2 6 1; +#X connect 8 0 11 0; +#X connect 8 1 10 0; +#X connect 9 0 1 0; +#X connect 10 0 9 0; +#X connect 11 0 9 0;