first import
1
sites/all/modules/imagecache_actions/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
||||
*.patch
|
59
sites/all/modules/imagecache_actions/CHANGELOG.txt
Normal file
@@ -0,0 +1,59 @@
|
||||
Imagecache Actions
|
||||
------------------
|
||||
|
||||
Imagecache Actions 7.x-1.x-dev
|
||||
------------------------------
|
||||
- [#1591198]: Image dimensions callbacks should handle unknown (NULL) dimensions
|
||||
as valid input.
|
||||
- [#464092]: Aspect Switcher -- Need to Flush Presets. Side-effect: now allows
|
||||
to define no image style for either landscape or portrait.
|
||||
- [#1676924]: Autorotate: check for exif extension and test dimensions callback.
|
||||
- [#1849020]: Cant get text test examples to work.
|
||||
- [#1702716]: System Stream Wrappers Conflict.
|
||||
- [#1844298]: Remove dependency from entity module.
|
||||
- [#1778594]: Negative image effect does not work.
|
||||
- [#1778214]: Remove version from image_effects_text_test.info.
|
||||
- [#1752664]: Error on importing image style.
|
||||
- [#1676924]: Autorotate: check for exif extension.
|
||||
- [#1666912] by Dubs: Call to undefined function.
|
||||
imagecache_actions_get_image_context() in imagecache_customactions.module.
|
||||
- [#1630194]: Custom actions broken after upgrading from dev to prod.
|
||||
- [#1591484] by epieddy: Underlay Center or Right parameter not working.
|
||||
- Moved the image styles administrative features to its own module.
|
||||
- Augment the duplicate with an export and import. This will allow users to
|
||||
duplicate between sites or to post failing styles in the issue queue (without
|
||||
using the features module).
|
||||
- Backport of patch from geoffreyr for [#1403962]: Add action "duplicate" to
|
||||
image style admin screen. This will allow us to easily create large sets of
|
||||
test/showcase styles and to fine tune this request before it lands in D8 core.
|
||||
|
||||
|
||||
Imagecache Actions 7.x-1.0 2012-05-19
|
||||
-------------------------------------
|
||||
Incompatibilities:
|
||||
- Effects that operate on the transparency layer, do not automatically change
|
||||
the format anymore to png. You will have to add this as a separate effect.
|
||||
- File handling: the way that files (e.g. fonts, watermarks, backgrounds) are
|
||||
searched for has been aligned over all effects that work with an additional
|
||||
file. This may cause current styles to not being able to find specified files
|
||||
anymore.
|
||||
- Custom actions: custom snippets are now executed using the PHP filter module,
|
||||
meaning that the image syle editor must have the 'Use PHP for settings'
|
||||
permission to be able to edit the custom action snippet.
|
||||
- Custom actions: information and variables that are available in your custom
|
||||
snippet have changed. See the README.txt of the custom actions module.
|
||||
|
||||
No changelog exists from before this 7.x-1.0 version. So the changelog starts
|
||||
from here. 7.x-1.0 is the first reasonably well working D7 version. The 7.x-0.0
|
||||
version was based on posted patches in the D7 port request issue and has never
|
||||
been tested well. This 7.x-1.0 version is also not extensively tested by the
|
||||
maintainers, but has been used in many (live) sites - currently (may 2012) over
|
||||
5.000 reported 7.x-1.x-dev installs - so has received fairly good "test
|
||||
coverage".
|
||||
|
||||
|
||||
Current and past maintainers for Imagecache Actions
|
||||
---------------------------------------------------
|
||||
- dman (http://drupal.org/user/33240)
|
||||
- sidneyshan (http://drupal.org/user/652426)
|
||||
- fietserwin (http://drupal.org/user/750928)
|
339
sites/all/modules/imagecache_actions/LICENSE.txt
Normal file
@@ -0,0 +1,339 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Lesser General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
this service if you wish), that you receive source code or can get it
|
||||
if you want it, that you can change the software or use pieces of it
|
||||
in new free programs; and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute verbatim copies of the Program's
|
||||
source code as you receive it, in any medium, provided that you
|
||||
conspicuously and appropriately publish on each copy an appropriate
|
||||
copyright notice and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the Program
|
||||
specifies a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||
REPAIR OR CORRECTION.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
convey the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License along
|
||||
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License.
|
196
sites/all/modules/imagecache_actions/README.txt
Normal file
@@ -0,0 +1,196 @@
|
||||
README for the Imagecache Actions Drupal module
|
||||
-----------------------------------------------
|
||||
|
||||
Project page: http://drupal.org/project/imagecache_actions
|
||||
|
||||
Current and past maintainers for Imagecache Actions:
|
||||
- dman (http://drupal.org/user/33240)
|
||||
- sidneyshan (http://drupal.org/user/652426)
|
||||
- fietserwin (http://drupal.org/user/750928)
|
||||
|
||||
Release notes for 7.x-1.1
|
||||
-------------------------
|
||||
- Clear the cache after updating.
|
||||
|
||||
|
||||
Release notes for 7.x-1.0
|
||||
-------------------------
|
||||
- Clear the (registry) cache after installing or updating to 7.x-1.0.
|
||||
- If you use custom actions, run update.php.
|
||||
- If you use custom actions, be sure to enable the 'PHP filter' module and give
|
||||
image style editors that may create custom actions the 'use PHP for settings'
|
||||
permission. The module must also be enabled on image creation.
|
||||
- If you use custom actions, please read the README.txt from that sub-module to
|
||||
find out about how information and resources are available to you. You will
|
||||
probably have to change your code snippets.
|
||||
- If you use effects that use files (mask, overlays, underlays, text fonts),
|
||||
check the way they are specified. From 7.x-1.0 on, you have to specify the
|
||||
location using one of the schemes private://, public://, module:// or
|
||||
temporary://. If no scheme is specified, the file is searched for as is, thus
|
||||
relative to the current directory or as an absolute path.
|
||||
- Effects that use the transparency layer (e.g. mask, rounded corners) do not
|
||||
automatically convert to PNG anymore. Use the "Change file format" for that.
|
||||
- There's no upgrade from D6. You will have to recreate your styles manually.
|
||||
|
||||
Warning:
|
||||
Ongoing development in the area of e.g. making the effects more consistent,
|
||||
adding and/or removing parameters or redefining their meaning, might cause
|
||||
backward incompatibilities between future versions and the current version.
|
||||
Thus, we cannot and do not guarantee backwards compatibility or automatic
|
||||
upgrade paths for future versions.
|
||||
|
||||
|
||||
Introduction
|
||||
------------
|
||||
The Imagecache Actions module provides a suite of additional image effects that
|
||||
can be added to image styles. Image styles let you create derivations of images
|
||||
by applying (a series of) effect(s) to it. Think of resizing, desaturating,
|
||||
masking, etc.
|
||||
|
||||
Furthermore, imagecache_actions extends the administrative interface for image
|
||||
styles by providing additional features. It does so in the "Image styles admin"
|
||||
sub module.
|
||||
|
||||
The additional effects that Imagecache Actions provides include:
|
||||
- Watermark: place a image with transparency anywhere over a source picture.
|
||||
- Overlay: add photo-corners etc to the image
|
||||
- Text overlay: add e.g. a copyright notice to your image.
|
||||
- Color-shifting: colorize images.
|
||||
- Brighten/Darken.
|
||||
- Alpha blending: use a grayscale image to define the transparency layer of an
|
||||
image.
|
||||
- Canvas manipulation: resize the canvas and add a backgroundcolor or image.
|
||||
- File Format switcher: if you need tranparency in JPGs, make them PNG. If your
|
||||
PNG thumbnails are 30K each, save them as JPGs.
|
||||
- Rounded corners.
|
||||
- TODO: complete list, check short descrptions
|
||||
|
||||
These effects are grouped in submodules. Just enable the ones you want to use.
|
||||
TODO: list submodules and their sets of effects.
|
||||
|
||||
Imagecache Actions supports both the GD toolkit from Drupal core and the
|
||||
Imagemagick toolkit. However, please note that Imagemagick support is not yet
|
||||
complete. Please file an issue if you encounter problems in using Imagemagick.
|
||||
|
||||
What is imagecache_action not?
|
||||
------------------------------
|
||||
Imagecache Actions does not provide a new UI or new menu items. It hooks into
|
||||
the already existing image styles system (from Drupal core). See
|
||||
http://drupal.org/documentation/modules/image for more information about working
|
||||
with images.
|
||||
|
||||
|
||||
A note about the name of this module
|
||||
------------------------------------
|
||||
Image styles are part of Drupal 7 core and are the successor of the Drupal 6
|
||||
imagecache module. In Drupal 6 image styles were called (imagecache) presets and
|
||||
the separate effects that made up a style were called (imagecache) actions. In
|
||||
porting to D7, that name has not been changed (yet).
|
||||
|
||||
|
||||
Which toolkit to use?
|
||||
---------------------
|
||||
Personally, I (fieterwin) prefer the imagemagick toolkit:
|
||||
- It is better in anti-aliasing. Try to rotate an image using both toolkits and
|
||||
you will see what I mean.
|
||||
- It does not execute in the PHP memory space, so is not restricted by the
|
||||
memory_limit PHP setting.
|
||||
- The GD toolkit will, at least on my Windows configuration, keep the font file
|
||||
open after a text operation, so you cannot delete, move or rename it anymore.
|
||||
|
||||
On the other hand: the GD toolkit is always available (in the correct version),
|
||||
whereas imagemagick is not always present on shared hosting or may be present in
|
||||
an antique version that might give problems.
|
||||
|
||||
Please also note that effects may give different results depending on the
|
||||
toolkit used.
|
||||
|
||||
|
||||
Hard Dependencies
|
||||
-----------------
|
||||
- Drupal 7.x
|
||||
- Image module from Drupal core
|
||||
|
||||
At least 1 of the available image toolkits:
|
||||
- GD toolkit from Drupal core
|
||||
- Imagemagick toolkit: http://drupal.org/project/imagemagick
|
||||
|
||||
Soft Dependencies
|
||||
-----------------
|
||||
- System stream wrapper (http://drupal.org/project/system_stream_wrapper)
|
||||
- Remote stream wrapper (http://drupal.org/project/remote_stream_wrapper)
|
||||
These modules provide additional stream wrappers. Especially the system stream
|
||||
wrapper is very handy as it provides, among others, a module:// and theme://
|
||||
wrapper.
|
||||
|
||||
Installing
|
||||
----------
|
||||
As usual.
|
||||
|
||||
|
||||
Usage
|
||||
-----
|
||||
After enabling the module:
|
||||
- Assure that the Image module from core is enabled.
|
||||
- Configure your toolkit and its settings at admin/config/media/image-toolkit.
|
||||
- Define image styles at admin/config/media/image-styles and add 1 or more
|
||||
effects as defined by this module
|
||||
- Use the image styles via e.g. the formatters of image fields.
|
||||
|
||||
|
||||
Upgrading from D6
|
||||
-----------------
|
||||
There's no upgrade path defined for sites upgrading from D6 to D7. This means
|
||||
that you will have to manually redefine your D6 imagecache presets as D7 image
|
||||
styles. Note that actually an upgrade path would have to be defined by the
|
||||
imagecache module, not this imagecache actions module. However, as there is no
|
||||
D7 version of imagecache that provides an upgrade, users may post an upgrade
|
||||
function to the issue queue and we will incorporate it.
|
||||
|
||||
|
||||
Backwards compatibility
|
||||
-----------------------
|
||||
Future releases will not be guaranteed to be backwards compatible. Implementing
|
||||
Imagemagick support e.g. might give unforeseen problems that can only be solved
|
||||
by changing the details of what an effect does. We will document these kind of
|
||||
incompatibilities in the changelog and the release notes.
|
||||
|
||||
|
||||
File form fields
|
||||
----------------
|
||||
A number of effects have a file form field where the editor can define a file
|
||||
name to use. This can be e.g. for overlays, masks or fonts. The file name should
|
||||
be defined using either:
|
||||
- 1 of the (enabled) scheme's:
|
||||
* public://
|
||||
* private:// Preferred for site specific masks, overlays, etc, that do not
|
||||
need to be shared publicly.
|
||||
* temporary:// Unlikely to be useful, but supported anyway as all schemes are
|
||||
supported.
|
||||
* module:// Introduced by the system stream wrapper module and preferred for
|
||||
module provided resources.
|
||||
* theme:// idem.
|
||||
* profile:// idem.
|
||||
* library:// idem.
|
||||
- A relative (to the current directory, probably Drupal root) or absolute path.
|
||||
|
||||
|
||||
|
||||
Support
|
||||
-------
|
||||
Via the issue queue of this project at Drupal.org.
|
||||
|
||||
|
||||
Known problems
|
||||
--------------
|
||||
These are better documented in the issue queue, but might be listed here (as
|
||||
well).
|
||||
|
||||
- Underlay does not work in imagemagick if the dimensions of both images are not
|
||||
equal. As a workaround first add a canvas effect with a fully transparent
|
||||
background.
|
||||
- Underlay/overlay: keywords in the x and y offset fields do not work.
|
||||
- Underlay does still display a message about Imagemagick not being supported.
|
||||
- Brightness values outside the -250 .. 250 range are accepted.
|
||||
- Check color fields that allow a transparency component or allow to be empty to
|
||||
specify fully transparent.
|
90
sites/all/modules/imagecache_actions/ROADMAP.txt
Normal file
@@ -0,0 +1,90 @@
|
||||
Imagecache Actions roadmap
|
||||
--------------------------
|
||||
|
||||
This is a non-contractual roadmap for the D7 branch of the imagecache actions
|
||||
module. Actual release dates and completed features will largely depend on
|
||||
available time. So support is always welcome. Furthermore, critical bugs may
|
||||
make us release more often, but that should only change the release number in
|
||||
which certain features are planned, not the timeline.
|
||||
|
||||
|
||||
Imagecache Actions 7.x-1.0
|
||||
--------------------------
|
||||
Targeted release date: may 2012
|
||||
|
||||
- (DONE) Clean up D7 issue queue
|
||||
- (DONE) Commit posted patches from the issue queue
|
||||
- (DONE) Solve easy to solve bug reports
|
||||
- Do some basic testing
|
||||
- (DONE) Reintroduce the text action, currently living in a sandbox project at
|
||||
http://drupal.org/node/1090312
|
||||
- (DONE) Update README.txt
|
||||
- (DONE) Introduce CHANGELOG.txt
|
||||
- (DONE) Add this ROADMAP.txt
|
||||
|
||||
|
||||
Imagecache Actions 7.x-1.1
|
||||
--------------------------
|
||||
Targeted release date: end 2012
|
||||
|
||||
Mainly a bug fix release
|
||||
- (DONE) Solve remaining outstanding bug reports
|
||||
- (ALMOST DONE) Keep D7 issue queue clean
|
||||
|
||||
|
||||
Imagecache Actions 7.x-1.2
|
||||
--------------------------
|
||||
Targeted release date: spring 2013
|
||||
|
||||
- Check help and documentation. a.o: hook_help, effect descriptions.
|
||||
- Continue to keep the D7 issue queue clean.
|
||||
- Improve Imagemagick support and/or document what effects are not working for
|
||||
which toolkit (version).
|
||||
- Check that all effects implement all of the effects API (especially the
|
||||
dimensions callback).
|
||||
- Add testing. The idea is to create a set of image styles that cover all
|
||||
effects and can be used to visually check for regressions.
|
||||
|
||||
Automated testing would be nice, but I am not sure that we can guarantee that
|
||||
jpeg or png files are binary identical on each run across different computers.
|
||||
But if the number of false positives is small, it would certainly be a useful
|
||||
addition.
|
||||
|
||||
|
||||
Imagecache Actions 7.x-1.3
|
||||
--------------------------
|
||||
Targeted release date: ...
|
||||
|
||||
- Look at outstanding feature requests
|
||||
- Continue to keep the D7 issue queue clean
|
||||
- Keep improving Imagemagick support, a.o:
|
||||
- Try to get insight into what version of imagemagick is required by what
|
||||
effect.
|
||||
- Document which effects might produce different results.
|
||||
- Refactor code
|
||||
- Extract common form fields
|
||||
- Extract common error handling
|
||||
- (STARTED) Clean up comments, todo's, etc.
|
||||
- (STARTED) Doxygen code documentation according to latest standards
|
||||
- Increase the amount of lazy loaded code
|
||||
|
||||
|
||||
Imagecache Actions 7.x-2.x
|
||||
--------------------------
|
||||
Targeted release date: ...
|
||||
|
||||
- Can we refactor effects into auto loading class based plugins? What does
|
||||
CTools offer in these? How do D8 core effects do this?
|
||||
- If so, will it make implementing new effects much simpler? Because that is
|
||||
what we should strive for.
|
||||
- Should we rename the module to image_effects?
|
||||
- If we manage to provide an easy to use plugin system, we should strive to
|
||||
merge with other modules that provide effects:
|
||||
- FiltersIE module (http://drupal.org/project/filtersie)
|
||||
- Imagecache Effects: http://drupal.org/project/imagecache_effects (D6 only,
|
||||
but are their effects more advanced then our counterparts or do they add new
|
||||
effects?)
|
||||
- Image watermark: http://drupal.org/project/watermark (D5 only, but is their
|
||||
watermark effect more advanced?)
|
||||
- ImageCache Scale-9 Actions: http://drupal.org/project/imagecache_scale9actions
|
||||
- etc.
|
@@ -0,0 +1,16 @@
|
||||
name = Imagecache Autorotate
|
||||
description = Autorotate image based on EXIF Orientation.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = image
|
||||
|
||||
files[] = imagecache_autorotate.install
|
||||
files[] = imagecache_autorotate.module
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,13 @@
|
||||
<?php
|
||||
/**
|
||||
* @file (un)install and (dis/en)able hooks for imagecache autorotate module.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_enable().
|
||||
*/
|
||||
function imagecache_autorotate_enable() {
|
||||
if (!function_exists('exif_read_data')) {
|
||||
drupal_set_message(t('The exif_read_data() function is not available in this PHP installation. You probably will have to enable the exif extension.'), 'warning');
|
||||
}
|
||||
}
|
@@ -0,0 +1,148 @@
|
||||
<?php
|
||||
/**
|
||||
* @file
|
||||
* Autorotate image based on EXIF Orientation tag.
|
||||
* http://sylvana.net/jpegcrop/exif_orientation.html
|
||||
*
|
||||
* This mini-module contributed by jonathan_hunt http://drupal.org/user/28976
|
||||
* September 1, 2009
|
||||
*
|
||||
* Tweaked by dman to add documentation
|
||||
*/
|
||||
|
||||
/* In Adobe PNGs and TIFF, this information MAY be present in the XMP
|
||||
* metadata like so:
|
||||
|
||||
<x:xmpmeta xmlns:x="adobe:ns:meta/">
|
||||
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||
<rdf:Description rdf:about="" xmlns:tiff="http://ns.adobe.com/tiff/1.0/">
|
||||
<tiff:Orientation>6</tiff:Orientation>
|
||||
</rdf:Description>
|
||||
</rdf:RDF>
|
||||
</x:xmpmeta>
|
||||
|
||||
*/
|
||||
|
||||
function imagecache_autorotate_image_effect_info() {
|
||||
$effects = array();
|
||||
|
||||
$effects['imagecache_autorotate'] = array(
|
||||
'label' => t('Autorotate'),
|
||||
'help' => t('Add autorotate image based on EXIF Orientation.'),
|
||||
'effect callback' => 'imagecache_autorotate_image',
|
||||
'dimensions callback' => 'imagecache_autorotate_dimensions',
|
||||
);
|
||||
|
||||
return $effects;
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo: This form is no longer needed nor defined in the hook above. If this
|
||||
* information still needs to be displayed it should probably be moved to help.
|
||||
*/
|
||||
function imagecache_autorotate_form() {
|
||||
$form = array();
|
||||
$form['help'] = array(
|
||||
'#type' => 'markup',
|
||||
'#value' => "<p>
|
||||
<b>There are no user-configurable options for this process.</b>
|
||||
</p><p>
|
||||
Certain cameras can embed <em>orientation</em> information into image
|
||||
files when they save them. This information is embedded in an EXIF tag
|
||||
and can be used to rotate images to their correct position for display.
|
||||
</p><p>
|
||||
<em>Not all cameras or images contain this information.</em>
|
||||
This process is only useful for those that do.
|
||||
</p><p>
|
||||
The expected/supported values are
|
||||
<br/><strong>Tag</strong>: <code>0x0112 Orientation</code>
|
||||
</p>
|
||||
<ul>
|
||||
<li>1 = Horizontal (normal)</li>
|
||||
<li>3 = Rotate 180</li>
|
||||
<li>6 = Rotate 90 CW</li>
|
||||
<li>8 = Rotate 270 CW</li>
|
||||
</ul>
|
||||
<p>
|
||||
<a href='http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/EXIF.html'>
|
||||
EXIF Reference</a>
|
||||
</p>
|
||||
",
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Autorotate image based on EXIF Orientation tag.
|
||||
*
|
||||
* See code at
|
||||
* http://sylvana.net/jpegcrop/exif_orientation.html
|
||||
*
|
||||
* and reference at
|
||||
* http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/EXIF.html
|
||||
*
|
||||
* @todo:
|
||||
* Add horizontal and vertical flips etc.
|
||||
* Need to create sample set for tests.
|
||||
*/
|
||||
function imagecache_autorotate_image($image, $data) {
|
||||
// Test to see if EXIF is supported for image type (e.g. not PNG).
|
||||
if ($image->info['mime_type'] == 'image/jpeg') {
|
||||
if (!function_exists('exif_read_data')) {
|
||||
watchdog('image', 'The image %file could not be auto-rotated because the exif_read_data() function is not available in this PHP installation. You probably will have to enable the exif extension.', array('%file' => $image->source));
|
||||
return FALSE;
|
||||
}
|
||||
$exif = exif_read_data(drupal_realpath($image->source));
|
||||
if (isset($exif['Orientation'])) {
|
||||
switch ($exif['Orientation']) {
|
||||
case 3:
|
||||
$degrees = 180;
|
||||
break;
|
||||
case 6:
|
||||
$degrees = 90;
|
||||
break;
|
||||
case 8:
|
||||
$degrees = 270;
|
||||
break;
|
||||
default:
|
||||
$degrees = 0;
|
||||
}
|
||||
if ($degrees != 0) {
|
||||
$org_width = $image->info['width'];
|
||||
$org_height = $image->info['height'];
|
||||
image_rotate($image, $degrees);
|
||||
if (($degrees === 90 || $degrees === 270) && $image->info['width'] === $org_width) {
|
||||
// The toolkit failed to alter the dimensions (imagemagick currently
|
||||
// fails to do so). So we do it ourselves.
|
||||
$image->info['width'] = $org_height;
|
||||
$image->info['height'] = $org_width;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else if ($image->source === 'modules/image/sample.png' && !function_exists('exif_read_data')) {
|
||||
// Issue a warning if we are in the admin screen and the exif extension is
|
||||
// not enabled.
|
||||
drupal_set_message(t('The exif_read_data() function is not available in this PHP installation. You probably will have to enable the exif extension.'), 'warning');
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Image dimensions callback; Auto-rotate.
|
||||
*
|
||||
* @param array $dimensions
|
||||
* An array with the dimensions (in pixels) to be modified.
|
||||
* @param array $data
|
||||
* An array of parameters for the autorotate effect (empty for this effect).
|
||||
*/
|
||||
function imagecache_autorotate_dimensions(array &$dimensions, array $data) {
|
||||
// We can only know the resulting dimensions if both dimensions are equal.
|
||||
// Otherwise we need to inspect the image itself, which is not passed in here.
|
||||
// (this callback was introduced to enhance performance by not accessing the
|
||||
// image file when rendering the width and height attributes of the html img
|
||||
// tag).
|
||||
if ($dimensions['width'] !== $dimensions['height']) {
|
||||
$dimensions['width'] = $dimensions['height'] = NULL;
|
||||
}
|
||||
}
|
@@ -0,0 +1,852 @@
|
||||
<?php
|
||||
/**
|
||||
* @file Helper functions for the text2canvas action for imagecache
|
||||
*
|
||||
* @author Dan Morrison http://coders.co.nz
|
||||
*
|
||||
* Individually configurable rounded corners logic contributed by canaryMason
|
||||
* 2009 03 http://drupal.org/node/402112
|
||||
*
|
||||
* Better algorithm for trimming rounded corners from donquixote
|
||||
* 2009 09 http://drupal.org/node/564036
|
||||
*
|
||||
*/
|
||||
|
||||
if (!function_exists('image_overlay')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'image_overlay');
|
||||
}
|
||||
if (!function_exists('imagecache_actions_pos_form')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'utility-form');
|
||||
}
|
||||
if (!function_exists('imagecache_actions_keyword_filter')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'utility');
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////
|
||||
// IMAGEMASK
|
||||
|
||||
/**
|
||||
* Implements the form callback for the image mask effect.
|
||||
*
|
||||
* @param array $data
|
||||
* array of settings for this action.
|
||||
* @return array
|
||||
* A form definition.
|
||||
*/
|
||||
function canvasactions_imagemask_form($data) {
|
||||
// @todo: add offset/positioning/scaling support - currently the mask is applied to the supplied image without resizing and positioned at (0,0)
|
||||
$form = array();
|
||||
$form['effect_help_text'] = array(
|
||||
'#type' => 'item',
|
||||
'#title' => t('Image mask'),
|
||||
'#description' => t('This effect will add (or replace) a transparency channel to your image. The mask file should be a grayscale image where black is fully transparent and white is fully opaque. The referenced mask will be applied to the top left of the image.'),
|
||||
);
|
||||
$form['path'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Mask file name'),
|
||||
'#default_value' => isset($data['path']) ? $data['path'] : '',
|
||||
'#description' => imagecache_actions_file_field_description(),
|
||||
'#element_validate' => array('imagecache_actions_validate_file'),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements the summary theme function for the image mask effect.
|
||||
*/
|
||||
function theme_canvasactions_imagemask_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
return 'file: ' . $data['path'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Implement the effect callback for the image mask effect.
|
||||
*
|
||||
* @param object $image
|
||||
* @param array $data
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function canvasactions_imagemask_image(&$image, $data = array()) {
|
||||
$mask = imagecache_actions_image_load($data['path'], $image->toolkit);
|
||||
if ($mask) {
|
||||
// @todo: (sydneyshan) Consider best way to add offset support - I assume we
|
||||
// would position the mask somewhere (top/left/offset px etc) and choose if
|
||||
// the surrounding area is white or black (opaque or transparent) using an
|
||||
// extra form element (radio). Assess existing positioning code first to
|
||||
// reduce duplication of code. Pass the results to the following function as
|
||||
// array($mask, $data). Perhaps add a 'scale mask to fit image'/'scale image
|
||||
// to fit mask'/'no scale' radio group?
|
||||
return image_toolkit_invoke('imagemask', $image, array($mask));
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements the image mask effect using the GD toolkit.
|
||||
*
|
||||
* $image object
|
||||
* Image object containing the GD image resource to operate on.
|
||||
* $mask object
|
||||
* An image object containing the image to use as mask.
|
||||
*/
|
||||
function image_gd_imagemask($image, $mask) {
|
||||
$newPicture = imagecreatetruecolor($image->info['width'], $image->info['height']);
|
||||
imagesavealpha($newPicture, TRUE);
|
||||
imagealphablending($newPicture, TRUE);
|
||||
$transparent = imagecolorallocatealpha($newPicture, 0, 0, 0, 127);
|
||||
imagefill($newPicture, 0, 0, $transparent);
|
||||
|
||||
// Perform pixel-based alpha map application
|
||||
for ($x = 0; $x < $image->info['width']; $x++) {
|
||||
for ($y = 0; $y < $image->info['height']; $y++) {
|
||||
// Deal with images with mismatched sizes
|
||||
if ($x >= $mask->info['width'] || $y >= $mask->info['height']) {
|
||||
imagesetpixel($newPicture, $x, $y, $transparent);
|
||||
}
|
||||
else {
|
||||
$alpha = imagecolorsforindex($mask->resource, imagecolorat($mask->resource, $x, $y));
|
||||
$alpha = 127 - floor($alpha['red'] / 2);
|
||||
$color = imagecolorsforindex($image->resource, imagecolorat($image->resource, $x, $y));
|
||||
imagesetpixel($newPicture, $x, $y, imagecolorallocatealpha($newPicture, $color['red'], $color['green'], $color['blue'], $alpha));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Copy back to original picture
|
||||
imagedestroy($image->resource);
|
||||
$image->resource = $newPicture;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implements the image mask effect using the imagemagick toolkit.
|
||||
*
|
||||
* $image object
|
||||
* Image object containing a.o. the image to operate on.
|
||||
* $mask object
|
||||
* An image object containing the image to use as mask.
|
||||
*/
|
||||
function image_imagemagick_imagemask($image, $mask) {
|
||||
$image->ops[] = escapeshellarg($mask->source) . ' -alpha Off -compose CopyOpacity -composite';
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* Settings for preparing a canvas.
|
||||
*
|
||||
* @param $data array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function canvasactions_definecanvas_form($data) {
|
||||
module_load_include('inc', 'imagecache_actions', 'utility-color');
|
||||
$defaults = array(
|
||||
'RGB' => array(
|
||||
'HEX' => '#333333',
|
||||
),
|
||||
'under' => TRUE,
|
||||
'exact' => array(
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' => array(
|
||||
'leftdiff' => '',
|
||||
'rightdiff' => '',
|
||||
'topdiff' => '',
|
||||
'bottomdiff' => '',
|
||||
),
|
||||
);
|
||||
$data = array_merge($defaults, (array) $data);
|
||||
|
||||
$form = array(
|
||||
'RGB' => imagecache_rgb_form($data['RGB']),
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t('Enter no color value for transparent. This will have the effect of adding clear margins around the image.'),
|
||||
'#prefix' => '<p>',
|
||||
'#suffix' => '</p>',
|
||||
),
|
||||
'under' => array(
|
||||
'#type' => 'checkbox',
|
||||
'#title' => t('Resize canvas <em>under</em> image (possibly cropping)'),
|
||||
'#default_value' => $data['under'],
|
||||
'#description' => t('If <em>not</em> set, this will create a solid flat layer, probably totally obscuring the source image'),
|
||||
),
|
||||
);
|
||||
$form['info'] = array('#value' => t('Enter values in ONLY ONE of the below options. Either exact or relative. Most values are optional - you can adjust only one dimension as needed. If no useful values are set, the current base image size will be used.'));
|
||||
$form['exact'] = array(
|
||||
'#type' => 'fieldset',
|
||||
'#collapsible' => TRUE,
|
||||
'#title' => 'Exact size',
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t('Set the canvas to a precise size, possibly cropping the image. Use to start with a known size.'),
|
||||
'#prefix' => '<p>',
|
||||
'#suffix' => '</p>',
|
||||
),
|
||||
'width' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Width'),
|
||||
'#default_value' => $data['exact']['width'],
|
||||
'#description' => t('Enter a value in pixels or percent'),
|
||||
'#size' => 5,
|
||||
),
|
||||
'height' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Height'),
|
||||
'#default_value' => $data['exact']['height'],
|
||||
'#description' => t('Enter a value in pixels or percent'),
|
||||
'#size' => 5,
|
||||
),
|
||||
);
|
||||
$form['exact'] = array_merge($form['exact'], imagecache_actions_pos_form($data['exact']));
|
||||
if (!$data['exact']['width'] && !$data['exact']['height']) {
|
||||
$form['exact']['#collapsed'] = TRUE;
|
||||
}
|
||||
|
||||
$form['relative'] = array(
|
||||
'#type' => 'fieldset',
|
||||
'#collapsible' => TRUE,
|
||||
'#title' => t('Relative size'),
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => '<p>' . t('Set the canvas to a relative size, based on the current image dimensions. Use to add simple borders or expand by a fixed amount. Negative values may crop the image.') . '</p>',
|
||||
),
|
||||
'leftdiff' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('left difference'),
|
||||
'#default_value' => $data['relative']['leftdiff'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Enter an offset in pixels.'),
|
||||
),
|
||||
'rightdiff' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('right difference'),
|
||||
'#default_value' => $data['relative']['rightdiff'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Enter an offset in pixels.'),
|
||||
),
|
||||
'topdiff' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('top difference'),
|
||||
'#default_value' => $data['relative']['topdiff'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Enter an offset in pixels.'),
|
||||
),
|
||||
'bottomdiff' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('bottom difference'),
|
||||
'#default_value' => $data['relative']['bottomdiff'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Enter an offset in pixels.'),
|
||||
),
|
||||
);
|
||||
if (!$data['relative']['leftdiff'] && !$data['relative']['rightdiff'] && !$data['relative']['topdiff'] && !$data['relative']['bottomdiff']) {
|
||||
$form['relative']['#collapsed'] = TRUE;
|
||||
}
|
||||
|
||||
$form['#submit'][] = 'canvasactions_definecanvas_form_submit';
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_canvasactions_definecanvas_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
if ($data['exact']['width'] || $data['exact']['height']) {
|
||||
$w = !empty($data['exact']['width']) ? $data['exact']['width'] : '100%';
|
||||
$h = !empty($data['exact']['height']) ? $data['exact']['height'] : '100%';
|
||||
$x = !empty($data['exact']['xpos']) ? $data['exact']['xpos'] : '0';
|
||||
$y = !empty($data['exact']['ypos']) ? $data['exact']['ypos'] : '0';
|
||||
$output = "{$w}x{$h} ($x, $y)";
|
||||
}
|
||||
else {
|
||||
$output = ' left:' . $data['relative']['leftdiff'];
|
||||
$output .= ' right:' . $data['relative']['rightdiff'];
|
||||
$output .= ' top:' . $data['relative']['topdiff'];
|
||||
$output .= ' bottom:' . $data['relative']['bottomdiff'];
|
||||
}
|
||||
$output .= theme('imagecacheactions_rgb', array('RGB' => $data['RGB']));
|
||||
$output .= ($data['under']) ? t(" <b>under</b> image ") : t(" <b>over</b> image ");
|
||||
return $output;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Creates a solid background canvas
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*
|
||||
* @param $image
|
||||
* array defining an image file, including :
|
||||
*
|
||||
* $image- >source as the filename,
|
||||
*
|
||||
* $image->info array
|
||||
*
|
||||
* $image->resource handle on the image object
|
||||
*/
|
||||
function canvasactions_definecanvas_effect($image, $data) {
|
||||
// May be given either exact or relative dimensions.
|
||||
if ($data['exact']['width'] || $data['exact']['height']) {
|
||||
// Allows only one dimension to be used if the other is unset.
|
||||
if (!$data['exact']['width']) {
|
||||
$data['exact']['width'] = $image->info['width'];
|
||||
}
|
||||
if (!$data['exact']['height']) {
|
||||
$data['exact']['height'] = $image->info['height'];
|
||||
}
|
||||
|
||||
$targetsize['width'] = imagecache_actions_percent_filter($data['exact']['width'], $image->info['width']);
|
||||
$targetsize['height'] = imagecache_actions_percent_filter($data['exact']['height'], $image->info['height']);
|
||||
|
||||
$targetsize['left'] = image_filter_keyword($data['exact']['xpos'], $targetsize['width'], $image->info['width']);
|
||||
$targetsize['top'] = image_filter_keyword($data['exact']['ypos'], $targetsize['height'], $image->info['height']);
|
||||
|
||||
}
|
||||
else {
|
||||
// calculate relative size
|
||||
$targetsize['width'] = $image->info['width'] + $data['relative']['leftdiff'] + $data['relative']['rightdiff'];
|
||||
$targetsize['height'] = $image->info['height'] + $data['relative']['topdiff'] + $data['relative']['bottomdiff'];
|
||||
$targetsize['left'] = $data['relative']['leftdiff'];
|
||||
$targetsize['top'] = $data['relative']['topdiff'];
|
||||
}
|
||||
|
||||
// convert from hex (as it is stored in the UI)
|
||||
if ($data['RGB']['HEX'] && $deduced = imagecache_actions_hex2rgba($data['RGB']['HEX'])) {
|
||||
$data['RGB'] = array_merge($data['RGB'], $deduced);
|
||||
}
|
||||
|
||||
// All the maths is done, now defer to the api toolkits;
|
||||
$data['targetsize'] = $targetsize;
|
||||
|
||||
$success = image_toolkit_invoke('definecanvas', $image, array($data));
|
||||
if ($success) {
|
||||
$image->info['width'] = $targetsize['width'];
|
||||
$image->info['height'] = $targetsize['height'];
|
||||
}
|
||||
return $success;
|
||||
}
|
||||
|
||||
function canvasactions_definecanvas_dimensions(array &$dimensions, array $data) {
|
||||
// May be given either exact or relative dimensions.
|
||||
if ($data['exact']['width'] || $data['exact']['height']) {
|
||||
// Allows only one dimension to be used if the other is unset.
|
||||
if (!$data['exact']['width']) {
|
||||
$data['exact']['width'] = $dimensions['width'];
|
||||
}
|
||||
if (!$data['exact']['height']) {
|
||||
$data['exact']['height'] = $dimensions['height'];
|
||||
}
|
||||
|
||||
$dimensions['width'] = imagecache_actions_percent_filter($data['exact']['width'], $dimensions['width']);
|
||||
$dimensions['height'] = imagecache_actions_percent_filter($data['exact']['height'], $dimensions['height']);
|
||||
}
|
||||
else {
|
||||
// calculate relative size
|
||||
$dimensions['width'] = $dimensions['width'] + $data['relative']['leftdiff'] + $data['relative']['rightdiff'];
|
||||
$dimensions['height'] = $dimensions['height'] + $data['relative']['topdiff'] + $data['relative']['bottomdiff'];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw a color (or transparency) behind an image
|
||||
*
|
||||
* $targetsize is an array expected to contain a width,height and a left,top
|
||||
* offset.
|
||||
*/
|
||||
function image_gd_definecanvas($image, $data = array()) {
|
||||
$targetsize = $data['targetsize'];
|
||||
$RGB = $data['RGB'];
|
||||
|
||||
$newcanvas = imagecreatetruecolor($targetsize['width'], $targetsize['height']);
|
||||
imagesavealpha($newcanvas, TRUE);
|
||||
imagealphablending($newcanvas, FALSE);
|
||||
imagesavealpha($image->resource, TRUE);
|
||||
if ($RGB['HEX']) {
|
||||
// Set color, allow it to define transparency, or assume opaque.
|
||||
$background = imagecolorallocatealpha($newcanvas, $RGB['red'], $RGB['green'], $RGB['blue'], $RGB['alpha']);
|
||||
}
|
||||
else {
|
||||
// No color, attempt transparency, assume white
|
||||
$background = imagecolorallocatealpha($newcanvas, 255, 255, 255, 127);
|
||||
}
|
||||
imagefilledrectangle($newcanvas, 0, 0, $targetsize['width'], $targetsize['height'], $background);
|
||||
# imagealphablending($newcanvas, TRUE);
|
||||
|
||||
if ($data['under']) {
|
||||
$canvas_object = (object) array(
|
||||
'resource' => $newcanvas,
|
||||
'info' => array(
|
||||
'width' => $targetsize['width'],
|
||||
'height' => $targetsize['height'],
|
||||
'mime_type' => $image->info['mime_type'],
|
||||
'extension' => $image->info['extension'],
|
||||
),
|
||||
'toolkit' => $image->toolkit,
|
||||
);
|
||||
image_overlay($image, $canvas_object, $targetsize['left'], $targetsize['top'], 100, TRUE);
|
||||
}
|
||||
else {
|
||||
$image->resource = $newcanvas;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw a color (or transparency) behind an image
|
||||
* $targetsize is an array expected to contain a width,height and a left,top
|
||||
* offset.
|
||||
*
|
||||
* See http://www.imagemagick.org/script/command-line-options.php#extent
|
||||
* @todo: reset gravity?
|
||||
*/
|
||||
function image_imagemagick_definecanvas($image, $data = array()) {
|
||||
$backgroundcolor = $data['RGB']['HEX'] != '' ? '#' . $data['RGB']['HEX'] : 'None';
|
||||
$image->ops[] = '-background ' . escapeshellarg($backgroundcolor);
|
||||
|
||||
$compose_operator = $data['under'] ? 'src-over' : 'dst-over';
|
||||
$image->ops[] = "-compose $compose_operator";
|
||||
|
||||
$targetsize = $data['targetsize'];
|
||||
$geometry = sprintf('%dx%d', $targetsize['width'], $targetsize['height']);
|
||||
if ($targetsize['left'] || $targetsize['top']) {
|
||||
$geometry .= sprintf('%+d%+d', -$targetsize['left'], -$targetsize['top']);
|
||||
}
|
||||
$image->ops[] = "-extent $geometry";
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////
|
||||
|
||||
/**
|
||||
* Place a given image under the current canvas
|
||||
*
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* @param $data array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function canvasactions_canvas2file_form($data) {
|
||||
// if (image_get_toolkit() != 'gd') {
|
||||
// drupal_set_message('Overlays are not currently supported by using imagemagick. This effect requires GD image toolkit only.', 'warning');
|
||||
// }
|
||||
|
||||
$defaults = array(
|
||||
'xpos' => '0',
|
||||
'ypos' => '0',
|
||||
'alpha' => '100',
|
||||
'path' => '',
|
||||
'dimensions' => 'original',
|
||||
);
|
||||
$data = array_merge($defaults, (array) $data);
|
||||
|
||||
$form = imagecache_actions_pos_form($data);
|
||||
$form['alpha'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('opacity'),
|
||||
'#default_value' => $data['alpha'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Opacity: 0-100. Be aware that values other than 100% may be slow to process.'),
|
||||
);
|
||||
$form['path'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('file name'),
|
||||
'#default_value' => $data['path'],
|
||||
'#description' => imagecache_actions_file_field_description(),
|
||||
'#element_validate' => array('imagecache_actions_validate_file'),
|
||||
);
|
||||
$form['dimensions'] = array(
|
||||
'#type' => 'radios',
|
||||
'#title' => t('final dimensions'),
|
||||
'#default_value' => $data['dimensions'],
|
||||
'#options' => array(
|
||||
'original' => 'original (dimensions are retained)',
|
||||
'background' => 'background (image will be forced to match the size of the background)',
|
||||
'minimum' => 'minimum (image may be cropped)',
|
||||
'maximum' => 'maximum (image may end up with gaps)',
|
||||
),
|
||||
'#description' => t('What to do when the background image is a different size from the source image. Backgrounds are not tiled, but may be arbitrarily large.'),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_canvasactions_canvas2file_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
$file = $data['path'];
|
||||
return "xpos:{$data['xpos']} , ypos:{$data['ypos']} alpha:{$data['alpha']}%. file: $file, dimensions:{$data['dimensions']}";
|
||||
}
|
||||
|
||||
/**
|
||||
* Place the source image on the current background
|
||||
*
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Note - this is currently incompatable with imagemagick, due to the way it
|
||||
* addresses $image->resource directly - a gd only thing.
|
||||
*
|
||||
* @param $image
|
||||
* @param $data
|
||||
*/
|
||||
function canvasactions_canvas2file_image(&$image, $data = array()) {
|
||||
$underlay = imagecache_actions_image_load($data['path'], $image->toolkit);
|
||||
if ($underlay) {
|
||||
// To handle odd sizes, we will resize/crop the background image to the
|
||||
// desired dimensions before starting the merge. The built-in
|
||||
// imagecopymerge, and the watermark library both do not allow overlays to
|
||||
// be bigger than the target.
|
||||
|
||||
// Adjust size
|
||||
$crop_rules = array(
|
||||
'xoffset' => 0,
|
||||
'yoffset' => 0,
|
||||
);
|
||||
if (empty($data['dimensions'])) {
|
||||
$data['dimensions'] = 'original';
|
||||
}
|
||||
switch ($data['dimensions']) {
|
||||
case 'original':
|
||||
// If the underlay is smaller than the target size,
|
||||
// then when preparing the underlay by cropping it,
|
||||
// the offsets may need to be negative
|
||||
// which will produce a 'cropped' image larger than the original.
|
||||
// In this case, we need to calculate the position of the bg image
|
||||
// in relation to the space it will occupy under the top layer
|
||||
#$crop_rules['xoffset'] = $underlay->info['width'] - $image->info['width'] ;
|
||||
|
||||
$crop_rules['width'] = $image->info['width'];
|
||||
$crop_rules['height'] = $image->info['height'];
|
||||
break;
|
||||
case 'background':
|
||||
$crop_rules['width'] = $underlay->info['width'];
|
||||
$crop_rules['height'] = $underlay->info['height'];
|
||||
break;
|
||||
case 'minimum':
|
||||
$crop_rules['width'] = min($underlay->info['width'], $image->info['width']);
|
||||
$crop_rules['height'] = min($underlay->info['height'], $image->info['height']);
|
||||
break;
|
||||
case 'maximum':
|
||||
$crop_rules['width'] = max($underlay->info['width'], $image->info['width']);
|
||||
$crop_rules['height'] = max($underlay->info['height'], $image->info['height']);
|
||||
break;
|
||||
}
|
||||
// imageapi crop assumes upsize is legal.
|
||||
|
||||
// Crop both before processing to avoid unwanted processing.
|
||||
image_crop_effect($underlay, $crop_rules);
|
||||
# BUG - this doesn't position either
|
||||
// Actually this fails because imagecache_crop fills it with solid color when 'cropping' to a larger size.
|
||||
#imagecache_crop_image($image, $crop_rules);
|
||||
#dpm(get_defined_vars());
|
||||
// This func modifies the underlay image by ref, placing the current canvas on it
|
||||
if (image_overlay($image, $underlay, $data['xpos'], $data['ypos'], $data['alpha'], TRUE)) {
|
||||
#$image->resource = $underlay->resource;
|
||||
$image = $underlay;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Image dimensions callback; canvas2file (underlay/background).
|
||||
*
|
||||
* @param array $dimensions
|
||||
* Dimensions to be modified - an associative array containing the items
|
||||
* 'width' and 'height' (in pixels).
|
||||
* @param $data
|
||||
* An associative array containing the effect data.
|
||||
*/
|
||||
function canvasactions_canvas2file_dimensions(array &$dimensions, array $data) {
|
||||
if ($data['dimensions'] !== 'original') {
|
||||
$underlay = imagecache_actions_image_load($data['path']);
|
||||
if ($underlay) {
|
||||
// If the new dimensions are taken from the background, we don't need to
|
||||
// know the original dimensions, we can just set the new dimensions to the
|
||||
// dimensions of the background. Otherwise, we need to know the old
|
||||
// dimensions. If unknown we have to leave them unknown.
|
||||
switch ($data['dimensions']) {
|
||||
case 'background':
|
||||
$dimensions['width'] = $underlay->info['width'];
|
||||
$dimensions['height'] = $underlay->info['height'];
|
||||
break;
|
||||
case 'minimum':
|
||||
$dimensions['width'] = isset($dimensions['width']) ? min($underlay->info['width'], $dimensions['width']) : NULL;
|
||||
$dimensions['height'] = isset($dimensions['height']) ? min($underlay->info['height'], $dimensions['height']) : NULL;
|
||||
break;
|
||||
case 'maximum':
|
||||
$dimensions['width'] = isset($dimensions['width']) ? max($underlay->info['width'], $dimensions['width']) : NULL;
|
||||
$dimensions['height'] = isset($dimensions['height']) ? max($underlay->info['height'], $dimensions['height']) : NULL;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Place a given image on top of the current canvas
|
||||
*
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* @param $data array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function canvasactions_file2canvas_form($data) {
|
||||
|
||||
$defaults = array(
|
||||
'xpos' => '',
|
||||
'ypos' => '',
|
||||
'alpha' => '100',
|
||||
'path' => '',
|
||||
);
|
||||
$data = array_merge($defaults, (array) $data);
|
||||
|
||||
$form = array(
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t('Note that using a transparent overlay that is larger than the source image may result in unwanted results - a solid background.'),
|
||||
),
|
||||
);
|
||||
$form += imagecache_actions_pos_form($data);
|
||||
$form['alpha'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('opacity'),
|
||||
'#default_value' => $data['alpha'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Opacity: 0-100. <b>Warning:</b> Due to a limitation in the GD toolkit, using an opacity other than 100% requires the system to use an algorithm that\'s much slower than the built-in functions. If you want partial transparency, you are better to use an already-transparent png as the overlay source image.'),
|
||||
);
|
||||
$form['path'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('file name'),
|
||||
'#default_value' => $data['path'],
|
||||
'#description' => imagecache_actions_file_field_description(),
|
||||
'#element_validate' => array('imagecache_actions_validate_file'),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_canvasactions_file2canvas_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
return '<strong>' . $data['path'] . '</strong> x:' . $data['xpos'] . ', y:' . $data['ypos'] . ' alpha:' . (@$data['alpha'] ? $data['alpha'] : 100) . '%';
|
||||
}
|
||||
|
||||
/**
|
||||
* Place the source image on the current background
|
||||
*
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
*
|
||||
* @param $image
|
||||
* @param $data
|
||||
*/
|
||||
function canvasactions_file2canvas_image($image, $data = array()) {
|
||||
$overlay = imagecache_actions_image_load($data['path']);
|
||||
if ($overlay) {
|
||||
if (!isset($data['alpha'])) {
|
||||
$data['alpha'] = 100;
|
||||
}
|
||||
return image_overlay($image, $overlay, $data['xpos'], $data['ypos'], $data['alpha']);
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////
|
||||
|
||||
/**
|
||||
* Place the source image on top of the current canvas
|
||||
*
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
*
|
||||
*
|
||||
* @param $data array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function canvasactions_source2canvas_form($data) {
|
||||
$defaults = array(
|
||||
'xpos' => '',
|
||||
'ypos' => '',
|
||||
'alpha' => '100',
|
||||
'path' => '',
|
||||
);
|
||||
$data = array_merge($defaults, (array) $data);
|
||||
|
||||
$form = imagecache_actions_pos_form($data);
|
||||
$form['alpha'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('opacity'),
|
||||
'#default_value' => $data['alpha'],
|
||||
'#size' => 6,
|
||||
'#description' => t('Opacity: 0-100.'),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_canvasactions_source2canvas_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
return 'xpos:' . $data['xpos'] . ', ypos:' . $data['ypos'] . ' alpha:' . $data['alpha'] . '%';
|
||||
}
|
||||
|
||||
/**
|
||||
* Place the source image on the current background
|
||||
*
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
*
|
||||
* @param $image
|
||||
* @param $data
|
||||
*/
|
||||
function canvasactions_source2canvas_image($image, $data = array()) {
|
||||
$overlay = image_load($image->source, $image->toolkit);
|
||||
return image_overlay($image, $overlay, $data['xpos'], $data['ypos'], $data['alpha']);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements the image effect form callback for the aspect switcher effect.
|
||||
*
|
||||
* @param array $data
|
||||
* Array of settings for this action
|
||||
* @return array
|
||||
* The form definition
|
||||
*/
|
||||
function canvasactions_aspect_form($data) {
|
||||
$defaults = array(
|
||||
'ratio_adjustment' => 1,
|
||||
'portrait' => NULL,
|
||||
'landscape' => NULL,
|
||||
);
|
||||
$data = array_merge($defaults, (array) $data);
|
||||
|
||||
$form = array(
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t('You must create the two presets to use <em>before</em> enabling this process.'),
|
||||
)
|
||||
);
|
||||
|
||||
$styles = image_style_options(TRUE);
|
||||
// @todo: remove the current style to prevent (immediate) recursion?
|
||||
|
||||
$form['portrait'] = array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Style to use if the image is portrait (vertical)'),
|
||||
'#default_value' => $data['portrait'],
|
||||
'#options' => $styles,
|
||||
'#description' => t('If you choose none, no extra processing will be done.'),
|
||||
);
|
||||
$form['landscape'] = array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Style to use if the image is landscape (horizontal)'),
|
||||
'#default_value' => $data['landscape'],
|
||||
'#options' => $styles,
|
||||
'#description' => t('If you choose none, no extra processing will be done.'),
|
||||
);
|
||||
|
||||
$form['ratio_adjustment'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Ratio Adjustment (advanced)'),
|
||||
'#size' => 3,
|
||||
'#default_value' => $data['ratio_adjustment'],
|
||||
'#description' => t("
|
||||
This allows you to bend the rules for how different the proportions need to be to trigger the switch.
|
||||
<br/>If the (width/height)*n is greater than 1, use 'landscape', otherwise use 'portrait'.
|
||||
<br/>When n = 1 (the default) it will switch between portrait and landscape modes.
|
||||
<br/>If n > 1, images that are slightly wide will still be treated as portraits.
|
||||
If n < 1 then blunt portraits will be treated as landscape.
|
||||
"),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implements the summary theme callback for the aspect switcher effect.
|
||||
*
|
||||
* @param array $variables
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function theme_canvasactions_aspect_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
$ratio_adjustment = '';
|
||||
if ($data['ratio_adjustment'] != 1) {
|
||||
$ratio_adjustment = " (switch at 1:{$data['ratio_adjustment']})";
|
||||
}
|
||||
return 'Portrait size: <strong>' . $data['portrait'] . '</strong>. Landscape size: <strong>' . $data['landscape'] . '</strong>' . $ratio_adjustment;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements the image effect callback for the aspect switcher effect.
|
||||
*
|
||||
* @param object $image
|
||||
* @param array $data
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function canvasactions_aspect_image($image, $data = array()) {
|
||||
$ratio_adjustment = isset($data['ratio_adjustment']) ? floatval( $data['ratio_adjustment']) : 1;
|
||||
$aspect = $image->info['width'] / $image->info['height'];
|
||||
|
||||
// width / height * adjustment. If > 1, it's wide.
|
||||
$style_name = (($aspect * $ratio_adjustment) > 1) ? $data['landscape'] : $data['portrait'];
|
||||
|
||||
if (empty($style_name)) {
|
||||
// Do nothing. just return what we've got.
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
$style = image_style_load($style_name);
|
||||
|
||||
if (empty($style)) {
|
||||
// Required preset has gone missing?
|
||||
watchdog('imagecache_canvasactions', "When running 'aspect' action, I was unable to load sub-action %style_name. Either it's been deleted or the DB needs an update", array('%style_name' => $style_name), WATCHDOG_ERROR);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Run the preset actions ourself.
|
||||
foreach ($style['effects'] as $sub_effect) {
|
||||
image_effect_apply($image, $sub_effect);
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements the dimension callback for the aspect switcher effect.
|
||||
*
|
||||
* @param array $dimensions
|
||||
* Dimensions to be modified - an associative array containing the items
|
||||
* 'width' and 'height' (in pixels).
|
||||
* @param array $data
|
||||
* An associative array containing the effect data.
|
||||
*/
|
||||
function canvasactions_aspect_dimensions(array &$dimensions, array $data) {
|
||||
if (empty($dimensions['width']) || empty($dimensions['height'])) {
|
||||
return;
|
||||
}
|
||||
$ratio_adjustment = isset($data['ratio_adjustment']) ? floatval( $data['ratio_adjustment']) : 1;
|
||||
$aspect = $dimensions['width'] / $dimensions['height'];
|
||||
$style_name = (($aspect * $ratio_adjustment) > 1) ? $data['landscape'] : $data['portrait'];
|
||||
image_style_transform_dimensions($style_name, $dimensions);
|
||||
}
|
@@ -0,0 +1,31 @@
|
||||
name = Imagecache Canvas Actions
|
||||
description = Actions for manipulating image canvases layers, including watermark and background effect. Also an aspect switcher (portrait/landscape)
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = imagecache_actions
|
||||
dependencies[] = image
|
||||
|
||||
files[] = canvasactions.inc
|
||||
files[] = imagecache_canvasactions.install
|
||||
files[] = imagecache_canvasactions.module
|
||||
files[] = rounded_corners.inc
|
||||
files[] = tests/cheap_dropshadow.imagecache_preset.inc
|
||||
files[] = tests/keyword_positioning.imagecache_preset.inc
|
||||
files[] = tests/positioned_underlay.imagecache_preset.inc
|
||||
files[] = tests/rotate_alpha.imagecache_preset.inc
|
||||
files[] = tests/rotate_alpha_gif.imagecache_preset.inc
|
||||
files[] = tests/rotate_scale.imagecache_preset.inc
|
||||
files[] = tests/rotate_scale_alpha.imagecache_preset.inc
|
||||
files[] = tests/rounded.imagecache_preset.inc
|
||||
files[] = tests/rounded_bl.imagecache_preset.inc
|
||||
files[] = tests/rounded_flattened.imagecache_preset.inc
|
||||
files[] = tests/watermark_100.imagecache_preset.inc
|
||||
files[] = tests/watermark_50.imagecache_preset.inc
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Set up new canvas actions. Tell imagecache.module about them
|
||||
*/
|
||||
|
||||
/**
|
||||
* Need to flush the cache when this module is enabled or disabled
|
||||
*/
|
||||
function imagecache_canvasactions_install() {
|
||||
if (function_exists('imagecache_action_definitions') ) {
|
||||
imagecache_action_definitions(TRUE);
|
||||
}
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
drupal_set_message(t('Additional image style actions should now be available in the presets !settings_link', array('!settings_link' => l(t('settings'), 'admin/config/media/image-styles'))));
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo Please document this function.
|
||||
* @see http://drupal.org/node/1354
|
||||
*/
|
||||
function imagecache_canvasactions_uninstall() {
|
||||
if (function_exists('imagecache_action_definitions') ) {
|
||||
imagecache_action_definitions(TRUE);
|
||||
}
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
}
|
@@ -0,0 +1,188 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file A collection of canvas (layer) type manipulations for imagecache -
|
||||
* including "Watermark"
|
||||
*
|
||||
* Based on first draft of the code by Dimm (imagecache.module 5--1)
|
||||
* http://drupal.org/node/184816
|
||||
*
|
||||
* Rewritten and ported to Imagecache actions API (imagecache.module 5--2) by
|
||||
* dman http://coders.co.nz/
|
||||
*
|
||||
*
|
||||
* Notes about imagecache action extensions. For each action:
|
||||
*
|
||||
* 1: Implement imagecache_HOOK_form($formdata) to define the config form.
|
||||
*
|
||||
* 1a: Implement theme_imagecache_HOOK_form if needed - optional
|
||||
*
|
||||
* 2: Implement imagecache_HOOK_image($image, $data) to DO the process
|
||||
*
|
||||
* 3: Implement theme_imagecache_HOOK($element) to return a text description of
|
||||
* the setting
|
||||
*
|
||||
* 4: Declare the action in HOOK_imagecache_actions()
|
||||
*
|
||||
*
|
||||
* API ref for hook_image()
|
||||
*
|
||||
* @param $image array defining an image file, including :
|
||||
*
|
||||
* $image- >source as the filename,
|
||||
*
|
||||
* $image->info array
|
||||
*
|
||||
* $image->resource handle on the image object
|
||||
*
|
||||
* @param $action array of settings as defined in your form.
|
||||
*
|
||||
*/
|
||||
|
||||
// During devel, caching is pointless. Flush it
|
||||
// imagecache_action_definitions(TRUE);
|
||||
|
||||
if (! function_exists('imagecache_actions_calculate_relative_position') ) {
|
||||
module_load_include('inc', 'imagecache_canvasactions', 'utility');
|
||||
}
|
||||
|
||||
// @todo There doesn't seem to be a way to specify a file in hook_image_effect_info
|
||||
// so placing this here for the time being.
|
||||
module_load_include('inc', 'imagecache_canvasactions', 'canvasactions');
|
||||
module_load_include('inc', 'imagecache_canvasactions', 'rounded_corners');
|
||||
// imageapi extensions
|
||||
module_load_include('inc', 'imagcache_actions', 'image_overlay.inc');
|
||||
|
||||
|
||||
function imagecache_canvasactions_image_effect_info() {
|
||||
$effects = array();
|
||||
|
||||
$effects['canvasactions_definecanvas'] = array(
|
||||
'label' => t('Define canvas'),
|
||||
'help' => t('Define the size of the working canvas and background color, this controls the dimensions of the output image.'),
|
||||
'effect callback' => 'canvasactions_definecanvas_effect',
|
||||
'dimensions callback' => 'canvasactions_definecanvas_dimensions',
|
||||
'form callback' => 'canvasactions_definecanvas_form',
|
||||
'summary theme' => 'canvasactions_definecanvas_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_imagemask'] = array(
|
||||
'label' => t('Image mask'),
|
||||
'help' => t(' Choose the file image you wish to use as a mask, and apply it to the canvas.'),
|
||||
'effect callback' => 'canvasactions_imagemask_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_imagemask_form',
|
||||
'summary theme' => 'canvasactions_imagemask_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_file2canvas'] = array(
|
||||
'label' => t('Overlay (watermark)'),
|
||||
'help' => t('Choose the file image you wish to use as an overlay, and position it in a layer on top of the canvas.'),
|
||||
'effect callback' => 'canvasactions_file2canvas_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_file2canvas_form',
|
||||
'summary theme' => 'canvasactions_file2canvas_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_canvas2file'] = array(
|
||||
'label' => t('Underlay (background)'),
|
||||
'help' => t('Choose the file image you wish to use as an background, and position the processed image on it.'),
|
||||
'effect callback' => 'canvasactions_canvas2file_image',
|
||||
'dimensions callback' => 'canvasactions_canvas2file_dimensions',
|
||||
'form callback' => 'canvasactions_canvas2file_form',
|
||||
'summary theme' => 'canvasactions_canvas2file_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_source2canvas'] = array(
|
||||
'label' => t('Overlay: source image to canvas'),
|
||||
'help' => t('Places the source image onto the canvas for compositing.'),
|
||||
'effect callback' => 'canvasactions_source2canvas_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_source2canvas_form',
|
||||
'summary theme' => 'canvasactions_source2canvas_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_roundedcorners'] = array(
|
||||
'label' => t('Rounded Corners'),
|
||||
'help' => t('This is true cropping, not overlays, so the result <em>can</em> be transparent.'),
|
||||
'effect callback' => 'canvasactions_roundedcorners_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_roundedcorners_form',
|
||||
'summary theme' => 'canvasactions_roundedcorners_summary',
|
||||
);
|
||||
|
||||
$effects['canvasactions_aspect'] = array(
|
||||
'label' => t('Aspect switcher'),
|
||||
'help' => t('Use different effects depending on whether the image is landscape of portrait shaped. This re-uses other preset definitions, and just chooses between them based on the rule.'),
|
||||
'effect callback' => 'canvasactions_aspect_image',
|
||||
'dimensions callback' => 'canvasactions_aspect_dimensions',
|
||||
'form callback' => 'canvasactions_aspect_form',
|
||||
'summary theme' => 'canvasactions_aspect_summary',
|
||||
);
|
||||
|
||||
return $effects;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Need to register the theme functions we expect to use
|
||||
*/
|
||||
function imagecache_canvasactions_theme() {
|
||||
$util_dir = drupal_get_path('module', 'imagecache_actions');
|
||||
return array(
|
||||
'canvasactions_definecanvas_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'canvasactions_imagemask_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'arguments' => array('element' => NULL),
|
||||
),
|
||||
'canvasactions_file2canvas_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'canvasactions_source2canvas_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'canvasactions_canvas2file_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'canvasactions_roundedcorners_summary' => array(
|
||||
'file' => 'rounded_corners.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'canvasactions_aspect_summary' => array(
|
||||
'file' => 'canvasactions.inc',
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_image_style_flush.
|
||||
*
|
||||
* This hook checks if the image style that is being flushed is used in an
|
||||
* aspect switcher effect. If so, the style that contains the aspect switcher
|
||||
* effect, should be flushed as well as the flushed style was probably changed.
|
||||
*
|
||||
* @param array $flushed_style
|
||||
* The image style that is being flushed.
|
||||
*/
|
||||
function imagecache_canvasactions_image_style_flush($flushed_style) {
|
||||
$styles = image_styles();
|
||||
foreach ($styles as $style) {
|
||||
if ($style['name'] !== $flushed_style['name']) {
|
||||
foreach ($style['effects'] as $effect) {
|
||||
if ($effect['name'] === 'canvasactions_aspect') {
|
||||
if ( (isset($effect['data']['portrait']) && $effect['data']['portrait'] === $flushed_style['name'])
|
||||
|| (isset($effect['data']['landscape']) && $effect['data']['landscape'] === $flushed_style['name'])) {
|
||||
image_style_flush($style);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,325 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Routines for rounded corners
|
||||
*/
|
||||
|
||||
/**
|
||||
* Set radius for corner rounding
|
||||
*
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function canvasactions_roundedcorners_form($action) {
|
||||
if (image_get_toolkit() != 'gd') {
|
||||
drupal_set_message('Rounded corners are not currently supported on all versions of imagemagick. This effect works best with GD image toolkit only.', 'warning');
|
||||
}
|
||||
drupal_add_js(drupal_get_path('module', 'imagecache_actions') . '/imagecache_actions.jquery.js');
|
||||
$defaults = array(
|
||||
'radius' => '16',
|
||||
#'antialias' => TRUE,
|
||||
'independent_corners_set' => array(
|
||||
'independent_corners' => FALSE,
|
||||
'radii' => array(
|
||||
'tl' => 0,
|
||||
'tr' => 0,
|
||||
'bl' => 0,
|
||||
'br' => 0,
|
||||
),
|
||||
),
|
||||
);
|
||||
$action = array_merge($defaults, (array) $action);
|
||||
|
||||
$form['radius'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('radius'),
|
||||
'#default_value' => $action['radius'],
|
||||
'#size' => 2,
|
||||
);
|
||||
|
||||
$form['independent_corners_set'] = array(
|
||||
'#type' => 'fieldset',
|
||||
'#title' => t('Individual Corner Values'),
|
||||
'#collapsible' => TRUE,
|
||||
'#collapsed' => (! $action['independent_corners_set']['independent_corners']),
|
||||
);
|
||||
$form['independent_corners_set']['independent_corners'] = array(
|
||||
'#type' => 'checkbox',
|
||||
'#title' => t('Set Corners Independently'),
|
||||
'#default_value' => $action['independent_corners_set']['independent_corners'],
|
||||
);
|
||||
$corners = array(
|
||||
'tl' => t("Top Left Radius"),
|
||||
'tr' => t("Top Right Radius"),
|
||||
'bl' => t("Bottom Left Radius"),
|
||||
'br' => t("Bottom Right Radius"),
|
||||
);
|
||||
// Loop over the four corners and create field elements for them.
|
||||
$form['independent_corners_set']['radii'] = array(
|
||||
'#type' => 'item',
|
||||
'#id' => 'independent-corners-set',
|
||||
);
|
||||
foreach ($corners as $attribute => $label) {
|
||||
$form['independent_corners_set']['radii'][$attribute] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => $label,
|
||||
'#default_value' => 0 + $action['independent_corners_set']['radii'][$attribute],
|
||||
'#size' => 2,
|
||||
);
|
||||
}
|
||||
/*
|
||||
$form['antialias'] = array(
|
||||
'#type' => 'checkbox',
|
||||
'#title' => t('antialias'),
|
||||
'#return_value' => TRUE,
|
||||
'#default_value' => $action['antialias'],
|
||||
'#description' => t('Attempt antialias smoothing when drawing the corners'),
|
||||
);
|
||||
*/
|
||||
$form['notes'] = array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t('
|
||||
Note: the rounded corners effect uses true alpha transparency masking.
|
||||
This means that this effect <b>will fail to be saved</b> on jpegs
|
||||
<em>unless</em> you either <ul>
|
||||
<li>convert the image to PNG (using the coloractions filter for that),</li>
|
||||
<li>define a canvas underneath it (using canvasactions-define-canvas) or</li>
|
||||
<li>underlay a solid color (using coloractions-alpha-flatten) or</li>
|
||||
<li>underlay a background image (canvasactions-underlay)</li>
|
||||
</ul>
|
||||
as a later part of this imagecache pipeline.
|
||||
<br/>
|
||||
'),
|
||||
);
|
||||
|
||||
return $form;
|
||||
}
|
||||
|
||||
function canvasactions_roundedcorners_image($image, $action = array()) {
|
||||
$independent_corners = !empty($action['independent_corners_set']['independent_corners']);
|
||||
if (!$independent_corners) {
|
||||
// set the independant corners to all be the same.
|
||||
$corners = array('tl', 'tr', 'bl', 'br');
|
||||
foreach ($corners as $key) {
|
||||
// Use the all-the-same radius setting.
|
||||
$action['independent_corners_set']['radii'][$key] = $action['radius'];
|
||||
}
|
||||
}
|
||||
|
||||
return image_toolkit_invoke('roundedcorners', $image, array($action));
|
||||
}
|
||||
|
||||
/**
|
||||
* Trim rounded corners off an image, using an anti-aliasing algorithm.
|
||||
*
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Note, this is not image toolkit-agnostic yet! It just assumes GD.
|
||||
* We can abstract it out once we have something else to abstract to.
|
||||
* In the meantime just don't.
|
||||
*
|
||||
* 'handcoded' rounded corners logic contributed by donquixote 2009-08-31
|
||||
*
|
||||
* @param $image
|
||||
* @param $action
|
||||
*/
|
||||
function image_gd_roundedcorners($image, $action = array()) {
|
||||
// Read settings.
|
||||
$width = $image->info['width'];
|
||||
$height = $image->info['height'];
|
||||
$radius = $action['radius'];
|
||||
$independent_corners = !empty($action['independent_corners_set']['independent_corners']);
|
||||
$corners = array('tl', 'tr', 'bl', 'br');
|
||||
|
||||
$im = &$image->resource;
|
||||
// Prepare drawing on the alpha channel.
|
||||
imagesavealpha($im, TRUE);
|
||||
imagealphablending($im, FALSE);
|
||||
|
||||
foreach ($corners as $key) {
|
||||
if ($independent_corners && isset($action['independent_corners_set']['radii'][$key])) {
|
||||
$r = $action['independent_corners_set']['radii'][$key];
|
||||
}
|
||||
else {
|
||||
// Use the all-the-same radius setting.
|
||||
$r = $radius;
|
||||
}
|
||||
|
||||
// key can be 'tl', 'tr', 'bl', 'br'.
|
||||
$is_bottom = ($key{0} == 'b');
|
||||
$is_right = ($key{1} == 'r');
|
||||
|
||||
// dx and dy are in "continuous coordinates",
|
||||
// and mark the distance of the pixel middle to the image border.
|
||||
for ($dx = .5; $dx < $r; ++$dx) {
|
||||
for ($dy = .5; $dy < $r; ++$dy) {
|
||||
|
||||
// ix and iy are in discrete pixel indices,
|
||||
// counting from the top left
|
||||
$ix = floor($is_right ? $width -$dx : $dx);
|
||||
$iy = floor($is_bottom ? $height -$dy : $dy);
|
||||
|
||||
// Color lookup at ($ix, $iy).
|
||||
$color_ix = imagecolorat($im, $ix, $iy);
|
||||
$color = imagecolorsforindex($im, $color_ix);
|
||||
|
||||
|
||||
// Do not process opacity if transparency is 100%. Just jump...
|
||||
// Opacity is always 0 on a transparent source pixel.
|
||||
if ($color['alpha'] != 127) {
|
||||
$opacity = _canvasactions_roundedcorners_pixel_opacity($dx, $dy, $r);
|
||||
if ($opacity >= 1) {
|
||||
// we can finish this row,
|
||||
// all following pixels will be fully opaque.
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
if (isset($color['alpha'])) {
|
||||
$color['alpha'] = 127 - round($opacity * (127 - $color['alpha']));
|
||||
}
|
||||
else {
|
||||
$color['alpha'] = 127 - round($opacity * 127);
|
||||
}
|
||||
// Value should not be more than 127, and not less than 0.
|
||||
$color['alpha'] = ($color['alpha'] > 127) ? 127 : (($color['alpha'] < 0) ? 0 : $color['alpha']);
|
||||
}
|
||||
|
||||
$color_ix = imagecolorallocatealpha($im, $color['red'], $color['green'], $color['blue'], $color['alpha']);
|
||||
imagesetpixel($im, $ix, $iy, $color_ix);
|
||||
}
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculate the transparency value for a rounded corner pixel
|
||||
*
|
||||
* @param $x
|
||||
* distance from pixel center to image border (left or right)
|
||||
* should be an integer + 0.5
|
||||
*
|
||||
* @param $y
|
||||
* distance from pixel center to image border (top or bottom)
|
||||
* should be an integer + 0.5
|
||||
*
|
||||
* @param $r
|
||||
* radius of the rounded corner
|
||||
* should be an integer
|
||||
*
|
||||
* @return float
|
||||
* opacity value between 0 (fully transparent) and 1 (fully opaque).
|
||||
*
|
||||
* OPTIMIZE HERE! This is a really tight loop, potentially getting called
|
||||
* thousands of times
|
||||
*/
|
||||
function _canvasactions_roundedcorners_pixel_opacity($x, $y, $r) {
|
||||
if ($x < 0 || $y < 0) {
|
||||
return 0;
|
||||
}
|
||||
else if ($x > $r || $y > $r) {
|
||||
return 1;
|
||||
}
|
||||
$dist_2 = ($r -$x) * ($r -$x) + ($r -$y) * ($r -$y);
|
||||
$r_2 = $r * $r;
|
||||
if ($dist_2 > ($r + 0.8) * ($r + 0.8)) {
|
||||
return 0;
|
||||
}
|
||||
else if ($dist_2 < ($r -0.8) * ($r -0.8)) {
|
||||
return 1;
|
||||
}
|
||||
else {
|
||||
// this pixel needs special analysis.
|
||||
// thanks to a quite efficient algorithm, we can afford 10x antialiasing :)
|
||||
$opacity = 0.5;
|
||||
if ($x > $y) {
|
||||
// cut the pixel into 10 vertical "stripes"
|
||||
for ($dx = -0.45; $dx < 0.5; $dx += 0.1) {
|
||||
// find out where the rounded corner edge intersects with the stripe
|
||||
// this is plain triangle geometry.
|
||||
$dy = $r - $y - sqrt($r_2 - ($r -$x -$dx) * ($r -$x -$dx));
|
||||
$dy = ($dy > 0.5) ? 0.5 : (($dy < -0.5) ? -0.5 : $dy);
|
||||
// count the opaque part of the stripe.
|
||||
$opacity -= 0.1 * $dy;
|
||||
}
|
||||
}
|
||||
else {
|
||||
// cut the pixel into 10 horizontal "stripes"
|
||||
for ($dy = -0.45; $dy < 0.5; $dy += 0.1) {
|
||||
// this is the math:
|
||||
// ($r-$x-$dx)^2 + ($r-$y-$dy)^2 = $r^2
|
||||
// $dx = $r - $x - sqrt($r^2 - ($r-$y-$dy)^2)
|
||||
$dx = $r - $x - sqrt($r_2 - ($r -$y -$dy) * ($r -$y -$dy));
|
||||
$dx = ($dx > 0.5) ? 0.5 : (($dx < -0.5) ? -0.5 : $dx);
|
||||
$opacity -= 0.1 * $dx;
|
||||
}
|
||||
}
|
||||
return ($opacity < 0) ? 0 : (($opacity > 1) ? 1 : $opacity);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* imageapi_roundedcorners
|
||||
*/
|
||||
function image_imagemagick_roundedcorners($image, $action = array()) {
|
||||
// Based on the imagemagick documentation.
|
||||
// http://www.imagemagick.org/Usage/thumbnails/#rounded
|
||||
// Create arc cut-outs, then mask them.
|
||||
// Draw black triangles and white circles.
|
||||
// draw circle is center: x,y, and a point on the perimeter
|
||||
$corners = array('tl', 'tr', 'bl', 'br');
|
||||
$radii = $action['independent_corners_set']['radii'];
|
||||
$width = $image->info['width'];
|
||||
$height = $image->info['height'];
|
||||
$tl = $radii['tl'];
|
||||
$tr = $radii['tr'];
|
||||
$bl = $radii['bl'];
|
||||
$br = $radii['br'];
|
||||
|
||||
$drawmask = '';
|
||||
if ($tl) {
|
||||
$drawmask .= " fill black polygon 0,0 0,{$tl} {$tl},0";
|
||||
$drawmask .= " fill white circle {$tl},{$tl} {$tl},0";
|
||||
}
|
||||
if ($tr) {
|
||||
$right = $width -$tr;
|
||||
$drawmask .= " fill black polygon {$right},0 {$width},0 {$width},{$tr}";
|
||||
$drawmask .= " fill white circle {$right},{$tr} {$right},0";
|
||||
}
|
||||
if ($bl) {
|
||||
$bottom = $height -$bl;
|
||||
$drawmask .= " fill black polygon 0,{$bottom} 0,{$height} {$bl},{$height}";
|
||||
$drawmask .= " fill white circle {$bl},{$bottom} 0,{$bottom}";
|
||||
}
|
||||
if ($br) {
|
||||
$bottom = $height -$br;
|
||||
$right = $width -$br;
|
||||
$drawmask .= " fill black polygon {$right},{$height} {$width},{$bottom} {$width},{$height}";
|
||||
$drawmask .= " fill white circle {$right},{$bottom} {$width},{$bottom}";
|
||||
}
|
||||
$draw = ' -draw ' . escapeshellarg($drawmask);
|
||||
|
||||
$compose = ' ' . escapeshellcmd('(') . " +clone -threshold -1 $draw " . escapeshellcmd(')') . ' +matte -compose CopyOpacity -composite ';
|
||||
$image->ops[] = $compose;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_canvasactions_roundedcorners($variables) {
|
||||
$element = $variables['element'];
|
||||
$data = $element['#value'];
|
||||
if (!empty($data['independent_corners_set']['independent_corners'])) {
|
||||
$dimens = join('px | ', $data['independent_corners_set']['radii']) . 'px';
|
||||
}
|
||||
else {
|
||||
$dimens = "Radius: {$data['radius']}px";
|
||||
}
|
||||
return $dimens;
|
||||
}
|
@@ -0,0 +1,114 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['cheap_dropshadow'] = array (
|
||||
'name' => 'cheap_dropshadow',
|
||||
'#weight' => '3.3',
|
||||
'effects' =>
|
||||
array (
|
||||
-1 => array (
|
||||
'weight' => '-1',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
0 =>
|
||||
array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_definecanvas',
|
||||
'data' =>
|
||||
array (
|
||||
'RGB' =>
|
||||
array (
|
||||
'HEX' => '999999',
|
||||
),
|
||||
'under' => 0,
|
||||
'exact' =>
|
||||
array (
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' =>
|
||||
array (
|
||||
'leftdiff' => '0',
|
||||
'rightdiff' => '0',
|
||||
'topdiff' => '0',
|
||||
'bottomdiff' => '0',
|
||||
),
|
||||
),
|
||||
),
|
||||
1 =>
|
||||
array (
|
||||
'weight' => '1',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_definecanvas',
|
||||
'data' =>
|
||||
array (
|
||||
'RGB' =>
|
||||
array (
|
||||
'HEX' => '',
|
||||
),
|
||||
'under' => 1,
|
||||
'exact' =>
|
||||
array (
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' =>
|
||||
array (
|
||||
'leftdiff' => '20',
|
||||
'rightdiff' => '0',
|
||||
'topdiff' => '20',
|
||||
'bottomdiff' => '0',
|
||||
),
|
||||
),
|
||||
),
|
||||
2 =>
|
||||
array (
|
||||
'weight' => '2',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_source2canvas',
|
||||
'data' =>
|
||||
array (
|
||||
'xpos' => 0,
|
||||
'ypos' => 0,
|
||||
'alpha' => '100',
|
||||
),
|
||||
),
|
||||
3 =>
|
||||
array (
|
||||
'weight' => '3',
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'data' =>
|
||||
array (
|
||||
'width' => '200',
|
||||
'height' => '100%',
|
||||
'upscale' => 0,
|
||||
),
|
||||
),
|
||||
4 => array (
|
||||
'weight' => '10',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 3.4 KiB |
After Width: | Height: | Size: 24 KiB |
@@ -0,0 +1,62 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['keyword_positioning'] = array (
|
||||
'name' => 'keyword_positioning',
|
||||
'#weight' => 4.2,
|
||||
'effects' => array (
|
||||
array (
|
||||
'weight' => '-1',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'right',
|
||||
'ypos' => 'top',
|
||||
'alpha' => '100',
|
||||
'path' => drupal_get_path('module', 'imagecache_testsuite') . "/grid-240x160.png",
|
||||
),
|
||||
),
|
||||
|
||||
array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => '25%',
|
||||
'ypos' => 'bottom-10%',
|
||||
'path' => 'misc/druplicon.png',
|
||||
),
|
||||
),
|
||||
|
||||
array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => '0%',
|
||||
'ypos' => 'top+10%',
|
||||
'path' => 'misc/druplicon.png',
|
||||
),
|
||||
),
|
||||
|
||||
array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'right+50',
|
||||
'ypos' => '50%',
|
||||
'path' => 'misc/druplicon.png',
|
||||
),
|
||||
),
|
||||
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 9.8 KiB |
@@ -0,0 +1,47 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['positioned_underlay'] = array (
|
||||
'name' => 'positioned_underlay',
|
||||
'#weight' => 4.4,
|
||||
'effects' => array (
|
||||
0 => array (
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'weight' => '0',
|
||||
'data' => array (
|
||||
'width' => '200',
|
||||
'height' => '',
|
||||
'upscale' => 0,
|
||||
),
|
||||
),
|
||||
1 => array (
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_canvas2file',
|
||||
'weight' => '1',
|
||||
'data' => array (
|
||||
'xpos' => '50',
|
||||
'ypos' => 'bottom+50',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/shiny-bg.png",
|
||||
'dimensions' => 'background',
|
||||
),
|
||||
),
|
||||
4 => array (
|
||||
'weight' => '10',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 52 KiB |
@@ -0,0 +1,35 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rotate_alpha'] = array (
|
||||
'name' => 'rotate_alpha',
|
||||
'#weight' => 1.4,
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '1',
|
||||
'module' => 'image',
|
||||
'name' => 'image_rotate',
|
||||
'data' => array (
|
||||
'degrees' => '15',
|
||||
'random' => 0,
|
||||
'bgcolor' => '',
|
||||
),
|
||||
),
|
||||
3 => array (
|
||||
'weight' => '3',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 76 KiB |
@@ -0,0 +1,36 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rotate_alpha_gif'] = array (
|
||||
'name' => 'rotate_alpha_gif',
|
||||
'#weight' => 1.5,
|
||||
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '1',
|
||||
'module' => 'image',
|
||||
'name' => 'image_rotate',
|
||||
'data' => array (
|
||||
'degrees' => '15',
|
||||
'random' => 0,
|
||||
'bgcolor' => '',
|
||||
),
|
||||
),
|
||||
3 => array (
|
||||
'weight' => '3',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/gif',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
@@ -0,0 +1,38 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rotate_scale'] = array (
|
||||
'name' => 'rotate_scale',
|
||||
'#weight' => 1.2,
|
||||
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '1',
|
||||
'module' => 'image',
|
||||
'name' => 'image_rotate',
|
||||
'data' => array (
|
||||
'degrees' => '15',
|
||||
'random' => 0,
|
||||
'bgcolor' => '',
|
||||
),
|
||||
),
|
||||
2 => array (
|
||||
'weight' => '2',
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'data' => array (
|
||||
'width' => '',
|
||||
'height' => '150',
|
||||
'upscale' => TRUE,
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 4.8 KiB |
@@ -0,0 +1,59 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rotate_scale_alpha'] = array (
|
||||
'name' => 'rotate_scale_alpha',
|
||||
'#weight' => 1.6,
|
||||
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '1',
|
||||
'module' => 'image',
|
||||
'name' => 'image_rotate',
|
||||
'data' => array (
|
||||
'degrees' => '65',
|
||||
'random' => 0,
|
||||
'bgcolor' => '',
|
||||
),
|
||||
),
|
||||
/*
|
||||
* imageapi resize is NOT alpha-safe. This test case proves the bug.
|
||||
* A work-around is to change format before resizing.
|
||||
2 => array (
|
||||
'weight' => '2',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
*/
|
||||
3 => array (
|
||||
'weight' => '3',
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'data' => array (
|
||||
'width' => '',
|
||||
'height' => '150',
|
||||
'upscale' => TRUE,
|
||||
),
|
||||
),
|
||||
4 => array (
|
||||
'weight' => '4',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' => array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 28 KiB |
@@ -0,0 +1,34 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rounded'] = array (
|
||||
'name' => 'rounded',
|
||||
'#weight' => 3.0,
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_roundedcorners',
|
||||
'data' => array (
|
||||
'radius' => '25',
|
||||
'antialias' => true,
|
||||
),
|
||||
),
|
||||
2 => array (
|
||||
'weight' => '3',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' =>array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 69 KiB |
@@ -0,0 +1,43 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rounded_bl'] = array (
|
||||
'name' => 'rounded_bl',
|
||||
'#weight' => 3.1,
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_roundedcorners',
|
||||
'data' => array (
|
||||
'radius' => '50',
|
||||
'independent_corners_set' => array (
|
||||
'independent_corners' => 1,
|
||||
'radii' => array (
|
||||
'tl' => '10',
|
||||
'tr' => '0',
|
||||
'bl' => '50',
|
||||
'br' => '10',
|
||||
),
|
||||
),
|
||||
'antialias' => true,
|
||||
),
|
||||
),
|
||||
2 => array (
|
||||
'weight' => '3',
|
||||
'module' => 'imagecache_coloractions',
|
||||
'name' => 'coloractions_convert',
|
||||
'data' =>array (
|
||||
'format' => 'image/png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 69 KiB |
@@ -0,0 +1,59 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['rounded_flattened'] = array (
|
||||
'name' => 'rounded_flattened',
|
||||
'#weight' => 3.3,
|
||||
|
||||
'effects' => array (
|
||||
1 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_roundedcorners',
|
||||
'data' => array (
|
||||
'radius' => '50',
|
||||
'independent_corners_set' => array (
|
||||
'independent_corners' => 1,
|
||||
'radii' => array (
|
||||
'tr' => '100',
|
||||
'br' => '0',
|
||||
'tl' => '0',
|
||||
'bl' => '0',
|
||||
),
|
||||
),
|
||||
'antialias' => true,
|
||||
),
|
||||
),
|
||||
2 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_definecanvas',
|
||||
'data' => array (
|
||||
'RGB' => array (
|
||||
'HEX' => '333333',
|
||||
),
|
||||
'under' => 1,
|
||||
'exact' => array (
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' => array (
|
||||
'leftdiff' => '2',
|
||||
'rightdiff' => '2',
|
||||
'topdiff' => '2',
|
||||
'bottomdiff' => '2',
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 69 KiB |
After Width: | Height: | Size: 2.3 KiB |
@@ -0,0 +1,28 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['watermark_100'] = array (
|
||||
'name' => 'watermark_100',
|
||||
'#weight' => 4.0,
|
||||
'effects' => array (
|
||||
0 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => '10',
|
||||
'ypos' => '5',
|
||||
'alpha' => '100',
|
||||
'path' => 'misc/druplicon.png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 8.0 KiB |
@@ -0,0 +1,28 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['watermark_50'] = array (
|
||||
'name' => 'watermark_50',
|
||||
'#weight' => 4.1,
|
||||
'effects' => array (
|
||||
0 => array (
|
||||
'weight' => '0',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'right+20',
|
||||
'ypos' => 'bottom',
|
||||
'alpha' => '50',
|
||||
'path' => 'misc/druplicon.png',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
After Width: | Height: | Size: 8.0 KiB |
@@ -0,0 +1,19 @@
|
||||
name = Imagecache Color Actions
|
||||
description = Additional ImageCache actions, providing color-shifting, brightness and alpha transparency effects.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = imagecache_actions
|
||||
dependencies[] = image
|
||||
|
||||
files[] = imagecache_coloractions.install
|
||||
files[] = imagecache_coloractions.module
|
||||
files[] = transparency.inc
|
||||
files[] = tests/green.imagecache_preset.inc
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Set up new color actions. Tell imagecache.module about them
|
||||
*/
|
||||
|
||||
/**
|
||||
* Need to flush the cache when this module is enabled or disabled
|
||||
*/
|
||||
function imagecache_coloractions_install() {
|
||||
if (function_exists('imagecache_action_definitions') ) {
|
||||
imagecache_action_definitions(TRUE);
|
||||
}
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
drupal_set_message(t('Additional image style actions should now be available in the presets !settings_link', array('!settings_link' => l(t('settings'), 'admin/config/media/image-styles'))));
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo Please document this function.
|
||||
* @see http://drupal.org/node/1354
|
||||
*/
|
||||
function imagecache_coloractions_uninstall() {
|
||||
if (function_exists('imagecache_action_definitions') ) {
|
||||
imagecache_action_definitions(TRUE);
|
||||
}
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
}
|
@@ -0,0 +1,501 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file
|
||||
* Additional actions for imagecache processing.
|
||||
*
|
||||
* Exposes some of the simpler PHP 'imagefilter' actions (colorshift,
|
||||
* brightness, negative)
|
||||
* - A transparency masker for merging with backgrounds.
|
||||
* - A pseudo - file conversion feature.
|
||||
*
|
||||
* Compatible with the 2008 revision (imagecache 2)
|
||||
*
|
||||
* @author dan http://coders.co.nz
|
||||
* @author sydneyshan http://enigmadigital.net.au
|
||||
*/
|
||||
|
||||
// During devel, caching is pointless. Flush it
|
||||
//imagecache_action_definitions(TRUE);
|
||||
|
||||
if (! function_exists('imagecache_actions_calculate_relative_position') ) {
|
||||
module_load_include('inc', 'imagecache_actions', 'utility');
|
||||
}
|
||||
module_load_include('inc', 'imagecache_actions', 'utility-color');
|
||||
|
||||
|
||||
// @todo There doesn't seem to be a way to specify a file in hook_image_effect_info
|
||||
// so placing this here for the time being.
|
||||
module_load_include('inc', 'imagecache_coloractions', 'transparency');
|
||||
|
||||
/**
|
||||
* hook_image_effect_info()
|
||||
*
|
||||
* Return the descriptions for the supported actions.
|
||||
*/
|
||||
function imagecache_coloractions_image_effect_info() {
|
||||
$effects = array();
|
||||
|
||||
$effects['coloractions_colorshift'] = array(
|
||||
'label' => t('Color Shift'),
|
||||
'help' => t('Adjust image colors.'),
|
||||
'effect callback' => 'coloractions_colorshift_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'coloractions_colorshift_form',
|
||||
'summary theme' => 'coloractions_colorshift_summary',
|
||||
);
|
||||
|
||||
$effects['imagecache_coloroverlay'] = array(
|
||||
'label' => t('Color Overlay'),
|
||||
'help' => t('Apply a color tint to an image (retaining blacks and whites).'),
|
||||
'effect callback' => 'coloractions_coloroverlay_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'coloractions_coloroverlay_form',
|
||||
'summary theme' => 'coloractions_coloroverlay_summary',
|
||||
);
|
||||
|
||||
$effects['coloractions_brightness'] = array(
|
||||
'label' => t('Brightness'),
|
||||
'help' => t('Adjust image brightness.'),
|
||||
'effect callback' => 'coloractions_brightness_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'coloractions_brightness_form',
|
||||
'summary theme' => 'coloractions_brightness_summary',
|
||||
);
|
||||
|
||||
$effects['coloractions_inverse'] = array(
|
||||
'label' => t('Negative Image'),
|
||||
'help' => t('Invert colors and brightness.'),
|
||||
'effect callback' => 'coloractions_inverse_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
);
|
||||
|
||||
// @todo Convert may need a little more work.
|
||||
$effects['coloractions_convert'] = array(
|
||||
'label' => t('Change file format'),
|
||||
'help' => t('Choose to save the image as a different filetype.'),
|
||||
'effect callback' => 'coloractions_convert_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'coloractions_convert_form',
|
||||
'summary theme' => 'coloractions_convert_summary',
|
||||
);
|
||||
|
||||
$effects['imagecache_alpha'] = array(
|
||||
'label' => t('Alpha Transparency'),
|
||||
'help' => t('Adjust transparency.'),
|
||||
'effect callback' => 'imagecache_alpha_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'imagecache_alpha_form',
|
||||
'summary theme' => 'coloractions_alpha_summary',
|
||||
);
|
||||
|
||||
return $effects;
|
||||
}
|
||||
|
||||
/**
|
||||
* hook_theme()
|
||||
*/
|
||||
function imagecache_coloractions_theme() {
|
||||
return array(
|
||||
'coloractions_colorshift_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'coloractions_coloroverlay_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'coloractions_alpha_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'coloractions_brightness_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
'coloractions_convert_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* Settings for colorshift actions.
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function coloractions_colorshift_form($action) {
|
||||
$defaults = array(
|
||||
'RGB' => array(
|
||||
'HEX' => '#FF0000',
|
||||
),
|
||||
);
|
||||
$action = array_merge($defaults, (array) $action);
|
||||
$form = array('#theme' => 'imagecache_rgb_form');
|
||||
$form['RGB'] = imagecache_rgb_form($action['RGB']);
|
||||
$form['note'] = array('#value' => t("<p>
|
||||
Note that colorshift is a mathematical filter that doesn't always
|
||||
have the expected result.
|
||||
To shift an image precisely TO a target color,
|
||||
desaturate (greyscale) it before colorizing.
|
||||
The hue (color wheel) is the <em>direction</em> the
|
||||
existing colors are shifted. The tone (inner box) is the amount.
|
||||
Keep the tone half-way up the left site of the color box
|
||||
for best results.
|
||||
</p>"));
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_coloractions_colorshift_summary($variables) {
|
||||
return theme_imagecacheactions_rgb($variables['data']);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*
|
||||
* Just converts and passes the vals to the all-purpose 'filter' action
|
||||
*/
|
||||
function coloractions_colorshift_image($image, $data = array()) {
|
||||
// convert color from hex (as it is stored in the UI)
|
||||
if ($data['RGB']['HEX'] && $deduced = imagecache_actions_hex2rgba($data['RGB']['HEX'])) {
|
||||
$data['RGB'] = array_merge($data['RGB'], $deduced);
|
||||
}
|
||||
return image_toolkit_invoke('colorshift', $image, array($data));
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_gd_colorshift($image, $data = array()) {
|
||||
$RGB = $data['RGB'];
|
||||
if (!function_exists('imagefilter')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'imagefilter');
|
||||
}
|
||||
return imagefilter($image->resource, 4, $RGB['red'], $RGB['green'], $RGB['blue']);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_imagemagick_colorshift($image, $data = array()) {
|
||||
$RGB = $data['RGB'];
|
||||
$image->ops[] = "-fill rgb" . escapeshellcmd('(') . "{$RGB['red']},{$RGB['green']},{$RGB['blue']}" . escapeshellcmd(')') . " -colorize 50" . escapeshellcmd('%');
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* Settings for coloroverlay actions.
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function coloractions_coloroverlay_form($action) {
|
||||
$defaults = array(
|
||||
'RGB' => array(
|
||||
'HEX' => '#E2DB6A',
|
||||
),
|
||||
);
|
||||
$action = array_merge($defaults, (array) $action);
|
||||
$form = array('#theme' => 'imagecache_rgb_form');
|
||||
$form['RGB'] = imagecache_rgb_form($action['RGB']);
|
||||
$form['note'] = array('#value' => t("<p>
|
||||
Note that color overlay is a mathematical filter that doesn't always
|
||||
have the expected result.
|
||||
To shift an image precisely TO a target color,
|
||||
desaturate (greyscale) it before colorizing.
|
||||
The hue (color wheel) is the <em>direction</em> the
|
||||
existing colors are shifted. The tone (inner box) is the amount.
|
||||
Keep the tone half-way up the left site of the color box
|
||||
for best results.
|
||||
</p>"));
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_coloractions_coloroverlay_summary($variables) {
|
||||
return theme_imagecacheactions_rgb($variables['data']);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*
|
||||
* Just converts and passes the vals to the all-purpose 'filter' action
|
||||
*/
|
||||
function coloractions_coloroverlay_image($image, $data = array()) {
|
||||
// convert color from hex (as it is stored in the UI)
|
||||
if ($data['RGB']['HEX'] && $deduced = imagecache_actions_hex2rgba($data['RGB']['HEX'])) {
|
||||
$data['RGB'] = array_merge($data['RGB'], $deduced);
|
||||
}
|
||||
return image_toolkit_invoke('coloroverlay', $image, array($data));
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_gd_coloroverlay($image, $data = array()) {
|
||||
$RGB = $data['RGB'];
|
||||
|
||||
$w = $image->info['width'];
|
||||
$h = $image->info['height'];
|
||||
|
||||
for($y=0;$y<$h;$y++) {
|
||||
for($x=0;$x<$w;$x++) {
|
||||
$rgb = imagecolorat($image->resource, $x, $y);
|
||||
$source = imagecolorsforindex($image->resource, $rgb);
|
||||
|
||||
if($source['red'] <= 128){
|
||||
$final_r = (2 * $source['red'] * $RGB['red'])/256;
|
||||
}else{
|
||||
$final_r = 255 - (((255 - (2 * ($source['red'] - 128))) * (255 - $RGB['red']))/256);
|
||||
}
|
||||
if($source['green'] <= 128){
|
||||
$final_g = (2 * $source['green'] * $RGB['green'])/256;
|
||||
}else{
|
||||
$final_g = 255 - (((255 - (2 * ($source['green'] - 128))) * (255 - $RGB['green']))/256);
|
||||
}
|
||||
if($source['blue'] <= 128){
|
||||
$final_b = (2 * $source['blue'] * $RGB['blue'])/256;
|
||||
}else{
|
||||
$final_b = 255 - (((255 - (2 * ($source['blue'] - 128))) * (255 - $RGB['blue']))/256);
|
||||
}
|
||||
$final_colour = imagecolorallocatealpha($image->resource, $final_r, $final_g, $final_b, $source['alpha']);
|
||||
imagesetpixel($image->resource, $x, $y, $final_colour);
|
||||
}
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_imagemagick_coloroverlay($image, $data = array()) {
|
||||
$RGB = $data['RGB'];
|
||||
$image->ops[] = escapeshellcmd('(') . " +clone +matte -fill rgb" . escapeshellcmd('(') . "{$RGB['red']},{$RGB['green']},{$RGB['blue']}" . escapeshellcmd(')') . " -colorize 100" . escapeshellcmd('%') . " +clone +swap -compose overlay -composite " . escapeshellcmd(')') . " -compose SrcIn -composite";
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* Settings for colorshift actions.
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function coloractions_brightness_form($action) {
|
||||
$default = array('filter_arg1' => '100');
|
||||
$action = array_merge($default, (array) $action);
|
||||
$form = array();
|
||||
$form['help'] = array('#value' => t("The brightness effect seldom looks good on its own, but can be useful to wash out an image before making it transparent - eg for a watermark."));
|
||||
$form['filter_arg1'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Brightness'),
|
||||
'#description' => t('-255 - +255'),
|
||||
'#default_value' => $action['filter_arg1'],
|
||||
'#size' => 3,
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*/
|
||||
function coloractions_brightness_image($image, $data = array()) {
|
||||
return image_toolkit_invoke('brightness', $image, array($data));
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_gd_brightness($image, $data = array()) {
|
||||
if (!function_exists('imagefilter')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'imagefilter'); }
|
||||
return imagefilter($image->resource, 2, $data['filter_arg1']);
|
||||
}
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_imagemagick_brightness($image, $data = array()) {
|
||||
$image->ops[] = "-modulate " . (int)(100 + ( $data['filter_arg1'] / 128 * 100 ));
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_coloractions_brightness_summary($variables) {
|
||||
return t("Adjust") . " : " . $variables['data']['filter_arg1'];
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* No settings.
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function coloractions_inverse_form($action) {
|
||||
$form = array();
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*/
|
||||
function coloractions_inverse_image($image, $data = array()) {
|
||||
return image_toolkit_invoke('inverse', $image, array($data));
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_gd_inverse($image, $data = array()) {
|
||||
if (!function_exists('imagefilter')) {
|
||||
module_load_include('inc', 'imagecache_actions', 'imagefilter');
|
||||
}
|
||||
return imagefilter($image->resource, 0);
|
||||
}
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*/
|
||||
function image_imagemagick_inverse(&$image, $data = array()) {
|
||||
// TODO
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function coloractions_convert_form($action) {
|
||||
$form = array(
|
||||
'help' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t("If you've been using transparencies in the process, the result may get saved as a PNG (as the image was treated as a one in in-between processes). If this is not desired (file sizes may get too big) you should use this process to force a flatten action before saving. "),
|
||||
),
|
||||
'help2' => array(
|
||||
'#type' => 'markup',
|
||||
'#value' => t("For technical reasons, changing the file format within imagecache does <em>not</em> change the filename suffix. A png may be saved as a *.jpg or vice versa. This may confuse some browsers and image software, but most of them have no trouble. "),
|
||||
),
|
||||
'format' => array(
|
||||
'#title' => t("File format"),
|
||||
'#type' => 'select',
|
||||
'#default_value' => isset($action['format']) ? $action['format'] : 'image/png',
|
||||
'#options' => coloractions_file_formats(),
|
||||
),
|
||||
'quality' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Quality'),
|
||||
'#description' => t('Override the default image quality. Works for Imagemagick only. Ranges from 0 to 100. For jpg, higher values mean better image quality but bigger files. For png it is a combination of compression and filter'),
|
||||
'#size' => 10,
|
||||
'#maxlength' => 3,
|
||||
'#default_value' => isset($action['quality']) ? $action['quality'] : '75',
|
||||
'#field_suffix' => '%',
|
||||
),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_coloractions_convert_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
$formats = coloractions_file_formats();
|
||||
if ($formats[$data['format']] == 'jpg') {
|
||||
return t('Convert to: @format, quality: @quality%', array(
|
||||
'@format' => $formats[$data['format']],
|
||||
'@quality' => $data['quality']
|
||||
));
|
||||
}
|
||||
else {
|
||||
return t("Convert to") .": ". $formats[$data['format']];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* Process the imagecache action on the passed image
|
||||
*/
|
||||
function coloractions_convert_image($image, $data = array()) {
|
||||
$formats = coloractions_file_formats();
|
||||
$image->info['mime_type'] = $data['format'];
|
||||
$image->info['extension'] = $formats[$data['format']];
|
||||
image_toolkit_invoke('convert_image', $image, array($data));
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of hook_{toolkit}_{effect}()
|
||||
*
|
||||
* image_toolkit_invoke will exit with an error when no implementation is
|
||||
* provided for the active toolkit so provide an empty operation for the GD
|
||||
* tookit
|
||||
*/
|
||||
function image_gd_convert_image($image, $data) {
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_{toolkit}_{effect}().
|
||||
*
|
||||
* Converting the image format with imagemagick is done by prepending the output
|
||||
* format to the target file separated by a colon (:). This is done with
|
||||
* hook_imagemagick_arguments_alter(), see below.
|
||||
*/
|
||||
function image_imagemagick_convert_image($image, $data) {
|
||||
$image->ops['output_format'] = $image->info['extension'];
|
||||
$image->ops['custom_quality_value'] = (int) $data['quality'];
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_imagemagick_arguments_alter.
|
||||
*
|
||||
* This hook moves a change in output format from the args (action list) to the
|
||||
* destination format setting within the context.
|
||||
*/
|
||||
function imagecache_coloractions_imagemagick_arguments_alter(&$args, &$context) {
|
||||
if (isset($args['output_format'])) {
|
||||
$context['destination_format'] = $args['output_format'];
|
||||
unset($args['output_format']);
|
||||
}
|
||||
if (isset($args['custom_quality_value'])) {
|
||||
$args['quality'] = sprintf('-quality %d', $args['custom_quality_value']);
|
||||
unset($args['custom_quality_value']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Mini mime-type list
|
||||
*/
|
||||
function coloractions_file_formats() {
|
||||
return array('image/jpeg' => 'jpg', 'image/gif' => 'gif', 'image/png' => 'png');
|
||||
}
|
||||
|
@@ -0,0 +1,239 @@
|
||||
<?php
|
||||
/**
|
||||
* @file Helper functions for the alpha action for imagecache
|
||||
*
|
||||
* @author dan http://coders.co.nz
|
||||
*/
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* Settings for alpha actions.
|
||||
*
|
||||
* @param $action array of settings for this action
|
||||
* @return a form definition
|
||||
*/
|
||||
function imagecache_alpha_form($action) {
|
||||
$defaults = array(
|
||||
'flatten' => FALSE,
|
||||
'RGB' => array('HEX' => '#000000'),
|
||||
'opacity' => 0.5,
|
||||
);
|
||||
$action = array_merge($defaults, (array) $action);
|
||||
|
||||
$form = array();
|
||||
$form['help'] = array(
|
||||
'#markup' => t("
|
||||
<p>You can <em>either</em> set the alpha values of the image to a fixed
|
||||
amount by defining opacity, <em>or</em> choose a color and let the
|
||||
darkness of the image pixels define an opacity.
|
||||
These are different effects. Don't do both
|
||||
or you will just get a plain block of color of a certain opacity.
|
||||
")
|
||||
);
|
||||
|
||||
$form['opacity'] = array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Opacity'),
|
||||
'#default_value' => $action['opacity'],
|
||||
'#size' => 3,
|
||||
'#description' => t("
|
||||
A decimal between 0 and 1.
|
||||
You can define the amount of transparency to apply, eg 0.8 (80%) opacity
|
||||
will make the image slightly transparent.
|
||||
Use this with <em>no</em> fill color defined for normal results.
|
||||
If you follow up by flattening the image onto white or grey,
|
||||
this will have the effect of partial desaturation.
|
||||
"),
|
||||
);
|
||||
|
||||
$form['description'] = array('#value' => t(
|
||||
"<p>Alpha toning is an advanced method of greyscaling or colorizing.
|
||||
It works using transparency, not colour matching.
|
||||
The results of this filter are excellent for using as watermarks,
|
||||
and for 'sepia' type imprints on coloured or textured backgrounds.
|
||||
It converts dark areas of the image to opaque, light to transparent.</p>
|
||||
<p>Note that if you are working with JPEGs, this alpha effect will not last into the final image
|
||||
<em>unless</em> you either <strong>flatten</strong> this image against a background color
|
||||
or image in a later process or <strong>convert</strong> it to a PNG before saving
|
||||
using available imagecache actions.</p>"
|
||||
));
|
||||
|
||||
$form['RGB'] = imagecache_rgb_form($action['RGB']);
|
||||
$form['RGB']['#type'] = 'fieldset';
|
||||
$form['RGB']['#title'] = t('Fill Color');
|
||||
$form['RGB']['HEX']['#description'] = t("
|
||||
Although this image will end up as an alpha transparency mask,
|
||||
it still has to have some colour to be visible.
|
||||
Black is safe. Dark Sepia #704214 is good too.
|
||||
Set it to nothing to not perform any color shift.
|
||||
");
|
||||
|
||||
|
||||
$form['flatten'] = array(
|
||||
'#type' => 'checkbox',
|
||||
'#title' => t('Flatten Transparency'),
|
||||
'#default_value' => $action['flatten'],
|
||||
'#return_value' => TRUE,
|
||||
'#description' => t("The opposite of adding alpha transparency, 'flatten' will place the given colour solidly behind the image. Use this if you can't trust IE, or you really do want the image filled in with a solid colour."),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_imagecache_alpha($variables) {
|
||||
$element = $variables['element'];
|
||||
return ($element['#value']['flatten'] ? t("Flatten") : t("Transparent"))
|
||||
. ($element['#value']['opacity'] ? " : " . ($element['#value']['opacity'] * 100) . '%' : '')
|
||||
. " : " . theme_imagecacheactions_rgb($element['#value']['RGB']);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Given an image, manipulate the transparancy behaviour.
|
||||
*
|
||||
* implementation of hook_image()
|
||||
*
|
||||
* Either convert light parts of an image to see-through, or place a solid
|
||||
* colour behind areas that would otherwise be see-though
|
||||
*
|
||||
* An imagecache_action_hook() . Handle a pipelined image transformation.
|
||||
*
|
||||
* To save a partially transparent image, the image resource must be switched to PNG.
|
||||
* REMEMBER TO SWITCH IT BACK if needed
|
||||
*
|
||||
* @param $image handle on the image definition, including a gd image resource
|
||||
* to act upon
|
||||
* @param $data settings for this process.
|
||||
* @return bool success
|
||||
*/
|
||||
function imagecache_alpha_image($image, $data = array()) {
|
||||
if (! $data['flatten']) {
|
||||
// given an image, convert dark areas to opaque,
|
||||
// light to transparent,
|
||||
return png_color2alpha($image, $data['RGB']['HEX'], $data['opacity']);
|
||||
}
|
||||
else {
|
||||
// Do the opposite, flatten the transparency ONTO the given colour
|
||||
$info = $image->info;
|
||||
|
||||
if (!$info) {
|
||||
watchdog("imagecache", "Problem converting image to fill behind. Source image returned no info");
|
||||
#dsm($source);
|
||||
return; // error
|
||||
}
|
||||
|
||||
$base_image = imagecreatetruecolor($info['width'], $info['height']);
|
||||
imagesavealpha($base_image, TRUE);
|
||||
imagealphablending($base_image, FALSE);
|
||||
|
||||
// Start with a solid colour
|
||||
$background_rgb = imagecache_actions_hex2rgba($data['RGB']['HEX']);
|
||||
|
||||
// Setting the background colour here solid is what flattens the image
|
||||
$background_color = @imagecolorallocatealpha($base_image, $background_rgb['red'], $background_rgb['green'], $background_rgb['blue'], 0);
|
||||
|
||||
// But what I really want to do is set it
|
||||
// coloured rgb AND 100% transparent, in the hope that
|
||||
// a failure to render transparency would instead render
|
||||
// THAT colour.
|
||||
$background_color = @imagecolorallocatealpha($base_image, $background_rgb['red'], $background_rgb['green'], $background_rgb['blue'], 0);
|
||||
// But that still doesn't work.
|
||||
// Yet somehow I've seen transparent images that fallback to white, not silver.
|
||||
|
||||
imagefill( $base_image, 0, 0, $background_color );
|
||||
|
||||
// And set the overlay behaviour back again
|
||||
imagealphablending($base_image, TRUE);
|
||||
|
||||
// Place the current image over it
|
||||
$foreground = $image->resource;
|
||||
$success = imagecopy($base_image, $image->resource, 0, 0, 0, 0, $info['width'], $info['height']);
|
||||
|
||||
$image->resource = $base_image;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This achives a tonal effect by converting the images combined tone and
|
||||
* existing transparency into one shade value. This is then used as the ALPHA
|
||||
* transparency for that pixel, while the whole thing is coloured the same
|
||||
* shade. Images 'greytoned' in this manner should sit smoothly on any
|
||||
* background.
|
||||
*
|
||||
* With no color set, use the existing hue.
|
||||
*
|
||||
* To save a partially transparent image, the image resource must be switched to PNG.
|
||||
* ... or maybe not. Just flatten it yourself, or switch the format yourself.
|
||||
* This hack would produce side effects otherwise.
|
||||
*
|
||||
* This algorithm runs maths per-pixel, and therefore is incredibly much more
|
||||
* inefficient than any native routine. Will kill the server on large images.
|
||||
*
|
||||
* @param $opacity between 0 transparent and 1 solid.
|
||||
*/
|
||||
function png_color2alpha($image, $color, $opacity = NULL) {
|
||||
$info = $image->info;
|
||||
if (!$info) {
|
||||
return FALSE;
|
||||
}
|
||||
$im1 = $image->resource;
|
||||
|
||||
imagesavealpha($im1, TRUE);
|
||||
imagealphablending($im1, FALSE);
|
||||
|
||||
if ($color) {
|
||||
$background = imagecache_actions_hex2rgba($color);
|
||||
}
|
||||
$width = imagesx($im1);
|
||||
$height = imagesy($im1);
|
||||
|
||||
if (($width * $height) > (1200 * 1200)) {
|
||||
watchdog('imagecache_actions', __FUNCTION__ . " on {$image->source}. Image is TOO BIG to run the per-pixel algorithm. Aborting.");
|
||||
return TRUE;
|
||||
}
|
||||
for ($i = 0; $i < $height; $i++) {
|
||||
//this loop traverses each row in the image
|
||||
for ($j = 0; $j < $width; $j++) {
|
||||
//this loop traverses each pixel of each row
|
||||
// Get the color & alpha info of the current pixel.
|
||||
$retrieved_color = imagecolorat($im1, $j, $i); // an index
|
||||
$rgba_array = imagecolorsforindex($im1, $retrieved_color);
|
||||
$alpha = 127;
|
||||
|
||||
// Calculate the total shade value of this pixel.
|
||||
|
||||
// If the rule sets a color, then the darkness of the existing
|
||||
// pixel will define the desired alpha value.
|
||||
if ($color) {
|
||||
$lightness = ( $rgba_array['red'] + $rgba_array['green'] + $rgba_array['blue'] ) / 3;
|
||||
// Need to flip the numbers around before doing maths.
|
||||
#$opacity = 1-($rgba_array['alpha']/127);
|
||||
#$darkness = 1-($lightness/256); // 0 is white, 1 is black
|
||||
#$visibility = $darkness * $opacity;
|
||||
#$alpha = (1-$visibility) * 127;
|
||||
$alpha = (1 - ((1 -($lightness / 256)) * (1 -($rgba_array['alpha'] / 127)))) * 127;
|
||||
}
|
||||
// If color is NOT set, then the existing color is passed though, only
|
||||
// made somewhat transparent.
|
||||
if (!$color) {
|
||||
$background = $rgba_array;
|
||||
}
|
||||
if ($opacity) {
|
||||
// It's a user-defined alpha value
|
||||
$alpha = $alpha * $opacity;
|
||||
}
|
||||
|
||||
// Paint the pixel.
|
||||
$color_to_paint = imagecolorallocatealpha($image->resource, $background['red'], $background['green'], $background['blue'], $alpha);
|
||||
imagesetpixel($image->resource, $j, $i, $color_to_paint);
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
146
sites/all/modules/imagecache_actions/customactions/README.txt
Normal file
@@ -0,0 +1,146 @@
|
||||
README
|
||||
------
|
||||
README for the custom actions effect module.
|
||||
|
||||
|
||||
Dependencies
|
||||
------------
|
||||
Hard dependencies:
|
||||
- Imagecache actions.
|
||||
- Image (Drupal core).
|
||||
|
||||
Soft dependencies/recommended modules:
|
||||
- Imagemagick (preferred toolkit).
|
||||
- PHP filter (Drupal core).
|
||||
|
||||
|
||||
Which toolkit?
|
||||
--------------
|
||||
Personally, I prefer the imagemagick toolkit:
|
||||
- It is better in anti-aliasing, try to rotate an image using both toolkits and
|
||||
you will see what I mean.
|
||||
- It does not execute in the PHP memory space, so is not restricted by the
|
||||
memory_limit PHP setting.
|
||||
- The GD toolkit will, at least on my Windows configuration, keep the font file
|
||||
open after a text operation, so you cannot delete, move or rename it anymore.
|
||||
- This module does a better job with Imagemagick (see below).
|
||||
|
||||
|
||||
Installing
|
||||
----------
|
||||
As usual. After enabling the module you can add custom actions to images.
|
||||
|
||||
|
||||
Custom action PHP snippets
|
||||
--------------------------
|
||||
Given the correct permission, the custom action effect allows you to write your
|
||||
own PHP snippet that does the requested processing on the image. How it can do
|
||||
so, depends on the toolkit.
|
||||
|
||||
For all toolkits, the snippet should return true to indicate success and false
|
||||
to indicate failure.
|
||||
|
||||
GD
|
||||
--
|
||||
The GD image resource is available in $image->resource. You can call the GD
|
||||
functions on this resource. This effect will query the width and height after
|
||||
your processing, so you don't have to change that yourself.
|
||||
|
||||
Imagemagick
|
||||
-----------
|
||||
All real image processing is done at the end, if all effects have added their
|
||||
command line arguments to the $image->ops array. So your custom action should
|
||||
add the imagemagick commands and its parameters by adding new string entries to
|
||||
the end of that array.
|
||||
|
||||
If your commands change the width or height of the resulting image, you should
|
||||
record so by changing $image->info['width'] and/or $image->info['height'].
|
||||
|
||||
General
|
||||
-------
|
||||
To ease your task, this effect makes some information regarding the image being
|
||||
processed available in 2 variables: $image and $image_context. These variables
|
||||
are readily available in your snippet.
|
||||
|
||||
$image is an associative array containing:
|
||||
- source: string, the source of the image, e.g. public://photo.jpg
|
||||
- info: array, example data:
|
||||
- width (int) 180
|
||||
- height (int) 180
|
||||
- extension (string) png
|
||||
- mime_type (string) image/png
|
||||
- file_size (int) 4417
|
||||
- toolkit: string, imagemagick or GD
|
||||
- resource: resource. The GD image resource.
|
||||
- ops: array. An array of strings with the ImageMagick commands.
|
||||
|
||||
$image_context is an associative array containing:
|
||||
- effect_data: array, the data of this image effect, example data for the custom
|
||||
action effect:
|
||||
- php (string)
|
||||
- managed_file: object|null. A managed file object containing these properties:
|
||||
- fid (string) 2
|
||||
- uid (string) 1
|
||||
- filename (string) photo.jpg
|
||||
- uri (string) public://photo.jpg
|
||||
- filemime (string) image/jpeg
|
||||
- filesize (string) 445751
|
||||
- status (string) 1
|
||||
- timestamp (string) 1327525851
|
||||
- metatags Array [0]
|
||||
- rdf_mapping Array [0]
|
||||
- referring_entities: array|null. A nested array with (fully loaded) entities
|
||||
referring to the current image. The 1st level of entries is keyed by the field
|
||||
name, the 2nd by entity type, and the 3rd by entity id. Example data:
|
||||
- field_photo Array [1]
|
||||
- node Array [1]
|
||||
- 12 Object of: stdClass
|
||||
- nid (string) 12
|
||||
- vid (string) 12
|
||||
- type (string) page
|
||||
- author ...
|
||||
- timestamp ...
|
||||
- ...
|
||||
- entity: object|null, the 1st entity in referring_entities. This is for easy
|
||||
access to the referring entity if it may be assumed that only 1 entity is
|
||||
referring to the current image.
|
||||
- image_field: array|null, the 1st image field in entity that is referring to
|
||||
the current image. This is for easy access to the image field data if it may
|
||||
be assumed that only 1 image field is referring to the current image. Example
|
||||
data:
|
||||
- fid (int) 2
|
||||
- alt (string) ...
|
||||
- title (string) ...
|
||||
- ...
|
||||
|
||||
Of course there are many other possible useful globals. Think of:
|
||||
- base_url
|
||||
- base_path
|
||||
- base_root
|
||||
- is_https
|
||||
- user
|
||||
- language
|
||||
and of course $_SERVER and $_GET.
|
||||
|
||||
Using these information you can access entity data as follows:
|
||||
|
||||
Specific case (1 entity, of known entity_type, referring to the image):
|
||||
<?php
|
||||
$entity_type = 'node';
|
||||
$field_name = 'my_field';
|
||||
$entity = $image_context['entity'];
|
||||
$field = field_get_items($entity_type, $entity, $field_name);
|
||||
?>
|
||||
|
||||
Or the more general case (not knowing the referring type, or multiple entities
|
||||
that may be referring to the image):
|
||||
<?php
|
||||
$referring_entities = $image_context['referring_entities'];
|
||||
foreach ($referring_entities as $field_name => $field_referring_entities) {
|
||||
foreach ($field_referring_entities as $entity_type => $entities) {
|
||||
foreach ($entities as $entity_id => $entity) {
|
||||
$field = field_get_items($entity_type, $entity, $field_name);
|
||||
}
|
||||
}
|
||||
}
|
||||
?>
|
@@ -0,0 +1,17 @@
|
||||
name = Imagecache Custom Actions
|
||||
description = Allow direct PHP code manipulation of imagecache images.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = imagecache_actions
|
||||
dependencies[] = image
|
||||
|
||||
files[] = imagecache_customactions.install
|
||||
files[] = imagecache_customactions.module
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,23 @@
|
||||
<?php
|
||||
/**
|
||||
* Rename 'text' to 'php' in custom action effect data
|
||||
*/
|
||||
function imagecache_customactions_update_7100(&$sandbox) {
|
||||
$effects = db_select('image_effects')
|
||||
->fields('image_effects')
|
||||
->condition('name', 'imagecache_customactions', '=')
|
||||
->execute()
|
||||
->fetchAll();
|
||||
foreach ($effects as $effect) {
|
||||
$data = unserialize($effect->data);
|
||||
if (array_key_exists('text', $data)) {
|
||||
$data['php'] = $data['text'];
|
||||
unset($data['text']);
|
||||
$data = serialize($data);
|
||||
db_update('image_effects')
|
||||
->condition('ieid', $effect->ieid)
|
||||
->fields(array('data' => $data))
|
||||
->execute();
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,270 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Allow advanced users to code their own PHP image manipulation routines
|
||||
* as part of imagecache processing.
|
||||
*
|
||||
* @author Originally contributed by crea http://drupal.org/node/325103#comment-
|
||||
* 1076011
|
||||
*
|
||||
* @author merged into imagecache_actions by dman http://coders.co.nz
|
||||
*
|
||||
* Needs review - currently a security risk etc
|
||||
*/
|
||||
|
||||
|
||||
/**
|
||||
* Implements hook_image_effect_info.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
function imagecache_customactions_image_effect_info() {
|
||||
$effects = array();
|
||||
|
||||
// @todo: implement summary theme callback
|
||||
$effects['imagecache_customactions'] = array(
|
||||
'label' => t('Custom action'),
|
||||
'help' => t('Runs custom PHP code.'),
|
||||
'effect callback' => 'imagecache_customactions_image',
|
||||
'dimensions callback' => 'imagecache_customactions_dimensions',
|
||||
'form callback' => 'imagecache_customactions_form',
|
||||
);
|
||||
|
||||
$effects['imagecache_subroutine'] = array(
|
||||
'label' => t('Subroutine'),
|
||||
'help' => t('Runs another defined preset on the image.'),
|
||||
'effect callback' => 'imagecache_subroutine_image',
|
||||
'dimensions callback' => 'imagecache_subroutine_dimensions',
|
||||
'form callback' => 'imagecache_subroutine_form',
|
||||
);
|
||||
|
||||
return $effects;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_image_style_flush.
|
||||
*
|
||||
* This hook checks if the image style that is being flushed is used in an
|
||||
* subroutine effect. If so, the style that contains the subroutine effect,
|
||||
* should be flushed as well as the flushed style was probably changed.
|
||||
*
|
||||
* @param array $flushed_style
|
||||
* The image style that is being flushed.
|
||||
*/
|
||||
function imagecache_customactions_image_style_flush($flushed_style) {
|
||||
$styles = image_styles();
|
||||
foreach ($styles as $style) {
|
||||
if ($style['name'] !== $flushed_style['name']) {
|
||||
foreach ($style['effects'] as $effect) {
|
||||
if ($effect['name'] === 'imagecache_subroutine') {
|
||||
if (isset($effect['data']['subroutine_presetname']) && $effect['data']['subroutine_presetname'] === $flushed_style['name']) {
|
||||
image_style_flush($style);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @deprecated replaced by summary theme callback
|
||||
* Implementation of theme_hook() for imagecache_customactions.module
|
||||
*/
|
||||
function imagecache_customactions_theme() {
|
||||
return array(
|
||||
'imagecache_subroutine' => array(
|
||||
'render element' => 'element',
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Implements hook_form().
|
||||
*
|
||||
* @param array $data
|
||||
* Array of settings for this action.
|
||||
* @return array
|
||||
* A form definition.
|
||||
*/
|
||||
function imagecache_customactions_form($data) {
|
||||
// Add defaults.
|
||||
$data += array('php' => 'return TRUE;');
|
||||
|
||||
// Note: we also need to check for the existence of the module: admin has
|
||||
// all rights, so user_acccess(...) returns TRUE even if the module is not
|
||||
// enabled and the permission does not exist.
|
||||
$allow_dynamic = user_access('use PHP for settings') && module_exists('php');
|
||||
|
||||
// @todo: for imagemagick, the PHP code should add a set of commands to the
|
||||
// ops aray of $image. Document this in description.
|
||||
$form = array(
|
||||
'php' => array(
|
||||
'#type' => 'textarea',
|
||||
'#rows' => 10,
|
||||
'#title' => t('PHP code'),
|
||||
'#default_value' => $data['php'],
|
||||
'#disabled' => !$allow_dynamic,
|
||||
'#description' => t("<p>A piece of PHP code that modifies the image.
|
||||
It should return a boolean indicating success or failure.
|
||||
You will need the '%use_php' permission, defined by the 'PHP filter' module.
|
||||
See the help for an extensive explanation of the possibilities.</p>",
|
||||
array('%use_php' => t('Use PHP for settings'))),
|
||||
'#wysiwyg' => FALSE,
|
||||
),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_image().
|
||||
*
|
||||
* @param object $image
|
||||
* @param array $data
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function imagecache_customactions_image($image, $data) {
|
||||
// Check that the PHP filter module is enabled.
|
||||
$result = module_exists('php');
|
||||
if ($result) {
|
||||
// Get context about the image.
|
||||
module_load_include('inc', 'imagecache_actions', 'utility');
|
||||
$GLOBALS['image_context'] = imagecache_actions_get_image_context($image, $data);
|
||||
$GLOBALS['image'] = $image;
|
||||
|
||||
$result = php_eval('<' . '?php global $image, $image_context; ' . $data['php'] . ' ?' . '>');
|
||||
// php_eval returns '1' if the snippet returns true.
|
||||
$result = $result === '1';
|
||||
|
||||
unset($GLOBALS['image']);
|
||||
unset($GLOBALS['image_context']);
|
||||
}
|
||||
|
||||
if ($result && $image->toolkit == 'GD') {
|
||||
$image->info['width'] = imagesx($image->resource);
|
||||
$image->info['height'] = imagesy($image->resource);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Image dimensions callback; Custom action.
|
||||
*
|
||||
* @param array $dimensions
|
||||
* Dimensions to be modified - an array with components width and height, in
|
||||
* pixels.
|
||||
* @param array $data
|
||||
* An array with the effect options.
|
||||
*/
|
||||
function imagecache_customactions_dimensions(array &$dimensions, array $data) {
|
||||
// @todo: add form field asking if dimensions stay the same (or if they know
|
||||
// the new dimesions).
|
||||
$dimensions['width'] = NULL;
|
||||
$dimensions['height'] = NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo change into summary theme callback
|
||||
* Implementation of theme_hook() for imagecache_ui.module
|
||||
*/
|
||||
function theme_imagecache_customactions($element) {
|
||||
// TODO: Should this theme imagecache_customactions be declared in hook_theme()?
|
||||
$data = $element['#value'];
|
||||
return "<em><strong>" . $data['text'] . "</strong></em>";
|
||||
}
|
||||
|
||||
/**
|
||||
* Subroutine - an imagecache action that just calls another one.
|
||||
*
|
||||
* Contributed by Alan D
|
||||
* http://drupal.org/node/618784
|
||||
*
|
||||
* Reworked into customactions by dman 2010-07
|
||||
*/
|
||||
|
||||
/**
|
||||
* Config form for this preset.
|
||||
*
|
||||
* Implementation of imagecache_hook_form()
|
||||
*
|
||||
* @param array $data
|
||||
* The effect data for this effect.
|
||||
* @return array
|
||||
* The form definition.
|
||||
*/
|
||||
function imagecache_subroutine_form($data) {
|
||||
$data = (array) $data;
|
||||
$form = array();
|
||||
|
||||
// List available presets
|
||||
// @todo: use image_style_options and remove current style?
|
||||
$presets = array();
|
||||
foreach (image_styles(TRUE) as $preset) {
|
||||
$presets[$preset['name']] = $preset['name'];
|
||||
}
|
||||
|
||||
$form['subroutine_presetname'] = array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Preset to call'),
|
||||
'#default_value' => $data['subroutine_presetname'],
|
||||
'#options' => $presets,
|
||||
);
|
||||
|
||||
return $form;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Actually invoke the action - which means just handing off to the named real
|
||||
* preset to do the job.
|
||||
*
|
||||
* Implementation of hook_image()
|
||||
*
|
||||
* @param object $image
|
||||
* @param array $data
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function imagecache_subroutine_image($image, $data) {
|
||||
if ($preset = image_style_load($data['subroutine_presetname'])) {
|
||||
foreach ($preset['effects'] as $effect) {
|
||||
image_effect_apply($image, $effect);
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Image dimensions callback; Subroutine.
|
||||
*
|
||||
* @param array $dimensions
|
||||
* Dimensions to be modified - an array with components width and height, in
|
||||
* pixels.
|
||||
* @param array $data
|
||||
* An array with the effect options.
|
||||
*/
|
||||
function imagecache_subroutine_dimensions(array &$dimensions, array $data) {
|
||||
// @todo: dimensions
|
||||
// @todo: call dimensions callback on subroutine style.
|
||||
$dimensions['width'] = NULL;
|
||||
$dimensions['height'] = NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo change into summary theme callback
|
||||
* This lets the user see what parameters were selected for the action
|
||||
*/
|
||||
function theme_imagecache_subroutine($variables) {
|
||||
// @todo: better decsription, do not use internal id's, imagecache_preset_by_name does not exist: fatal error?
|
||||
$element = $variables['element'];
|
||||
$data = $element['#value'];
|
||||
if ($preset = imagecache_preset_by_name($data['subroutine_presetname'])) {
|
||||
return t('%name (pid: !presetid)', array(
|
||||
'%name' => $preset['presetname'],
|
||||
'!presetid' => $preset['presetid']
|
||||
));
|
||||
}
|
||||
return t('<span class="error">Invalid reference. The referenced preset may have been deleted!</span>');
|
||||
}
|
BIN
sites/all/modules/imagecache_actions/docs/aspect-chaining.png
Normal file
After Width: | Height: | Size: 314 KiB |
BIN
sites/all/modules/imagecache_actions/docs/brightness.png
Normal file
After Width: | Height: | Size: 29 KiB |
BIN
sites/all/modules/imagecache_actions/docs/canvases.png
Normal file
After Width: | Height: | Size: 29 KiB |
BIN
sites/all/modules/imagecache_actions/docs/cheap_dropshadow.png
Normal file
After Width: | Height: | Size: 26 KiB |
BIN
sites/all/modules/imagecache_actions/docs/colorshift.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
sites/all/modules/imagecache_actions/docs/combo.png
Normal file
After Width: | Height: | Size: 54 KiB |
BIN
sites/all/modules/imagecache_actions/docs/corner-bl.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
sites/all/modules/imagecache_actions/docs/corner-br.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
sites/all/modules/imagecache_actions/docs/corner-tl.png
Normal file
After Width: | Height: | Size: 2.1 KiB |
BIN
sites/all/modules/imagecache_actions/docs/corner-tr.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
sites/all/modules/imagecache_actions/docs/overlays.png
Normal file
After Width: | Height: | Size: 64 KiB |
After Width: | Height: | Size: 4.7 KiB |
BIN
sites/all/modules/imagecache_actions/docs/polaroid.png
Normal file
After Width: | Height: | Size: 251 KiB |
BIN
sites/all/modules/imagecache_actions/docs/rounded.png
Normal file
After Width: | Height: | Size: 39 KiB |
BIN
sites/all/modules/imagecache_actions/docs/text.png
Normal file
After Width: | Height: | Size: 43 KiB |
BIN
sites/all/modules/imagecache_actions/docs/transparency.png
Normal file
After Width: | Height: | Size: 40 KiB |
BIN
sites/all/modules/imagecache_actions/docs/watermark.png
Normal file
After Width: | Height: | Size: 63 KiB |
146
sites/all/modules/imagecache_actions/help/aspect_switcher.html
Normal file
@@ -0,0 +1,146 @@
|
||||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
|
||||
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml">
|
||||
<head>
|
||||
<title>
|
||||
Aspect Switcher
|
||||
</title>
|
||||
</head>
|
||||
<body>
|
||||
<h3>
|
||||
Using aspect switcher to create a 'square' setting.
|
||||
</h3>
|
||||
<p>
|
||||
This is done through "chaining" two switches - "is it quite
|
||||
wide or not?", then "is it quite tall or not?" which leaves
|
||||
us with "must be square then."
|
||||
</p>
|
||||
<p>
|
||||
First we create the three sizes we will be using,
|
||||
small-landscape, small-square, small-portrait. I'll just set
|
||||
those up with scale_and_crop.
|
||||
</p>
|
||||
<p>
|
||||
We want wide images up to a ratio of 1:0.75 to be rendered
|
||||
wide. We want squarish images, with an aspect between 1:0.75
|
||||
and 1:1.25 to be rendered square, and anything taller to be
|
||||
rendered tall.
|
||||
</p>
|
||||
<p>
|
||||
To do this, we chain 2 rules. We need to build them
|
||||
backwards, the smaller sub-rule first, but to understand, I'l
|
||||
list them top down.
|
||||
</p>
|
||||
<p>
|
||||
Rule 1. is the master rule, <strong>3-aspects</strong>
|
||||
</p>
|
||||
<p>
|
||||
if ratio is less than 1:.75, use small-landscape. If greater,
|
||||
<strong>proceed to rule 2</strong>.
|
||||
</p>
|
||||
<p>
|
||||
</p>
|
||||
<p>
|
||||
Rule 2. <strong>square-or-portrait</strong>
|
||||
</p>
|
||||
<p>
|
||||
if ratio is less than 1:1.25, use small-square. If greater,
|
||||
use small-portrait.
|
||||
</p>
|
||||
<p>
|
||||
To do this, we use the aspect switcher to link to the two
|
||||
sizes, and the <em>ratio adjustment</em> to move
|
||||
the switching point a little. Set the ratio adjustment to
|
||||
1.25
|
||||
</p>
|
||||
<p>
|
||||
With these (5!) rules in place, you can get the desired
|
||||
effect. This is a little trickier than just making a 'square'
|
||||
setting, but it allows for the required fudge factor to
|
||||
handle almost-square images.
|
||||
</p>
|
||||
<p>
|
||||
You can nudge the adjustment factor to be looser or tighter.
|
||||
You can create even more chained rules, and define a
|
||||
'super-wide' size.
|
||||
</p>
|
||||
<table border="1" cellpadding="1" cellspacing="1">
|
||||
<tbody>
|
||||
<tr>
|
||||
<td rowspan="1" colspan="1">
|
||||
small-landscape
|
||||
</td>
|
||||
<td rowspan="1" colspan="1">
|
||||
[Scale And Crop] width: 200, height: 100
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td rowspan="1" colspan="1">
|
||||
small-portrait
|
||||
</td>
|
||||
<td rowspan="1" colspan="1">
|
||||
[Scale And Crop] width: 100, height: 200
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td rowspan="1" colspan="1">
|
||||
small-square
|
||||
</td>
|
||||
<td rowspan="1" colspan="1">
|
||||
[Scale And Crop] width: 140, height: 140
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td rowspan="1" colspan="1">
|
||||
small-square-or-portrait
|
||||
</td>
|
||||
<td rowspan="1" colspan="1">
|
||||
[Aspect Switcher] Portrait
|
||||
size: <strong>small-portrait</strong>. Landscape
|
||||
size: <strong>small-square</strong> (switch
|
||||
at 1:1.25)
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td rowspan="1" colspan="1">
|
||||
small-3-aspects
|
||||
</td>
|
||||
<td rowspan="1" colspan="1">
|
||||
[Aspect Switcher] Portrait
|
||||
size: <strong>small-square-or-portrait</strong>.
|
||||
Landscape
|
||||
size: <strong>small-landscape</strong> (switch
|
||||
at 1:.75)
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
<p>
|
||||
The illustration shows the result of this set-up on a
|
||||
collection of images. The listed dimensions are those of the
|
||||
source images. You'll see that the mostly-square ones are
|
||||
rendered square.
|
||||
</p>
|
||||
<img src="../docs/aspect-chaining.png" alt="Illustration of several different sized images passing through the above ruleset."/>
|
||||
<p>
|
||||
The rule being applied is: 1 Is it wide?
|
||||
</p>
|
||||
<p>
|
||||
For image 250x300, the aspect is ( 250/300 = 0.83 ) Normally
|
||||
that number (less than 1) would be classified as 'portrait',
|
||||
and with the adjustment (*0.75) that is still true, so the
|
||||
processing passes through to the portrait preset.
|
||||
</p>
|
||||
<p>
|
||||
rule #2 it it tall?
|
||||
</p>
|
||||
<p>
|
||||
This preset however does a different set of maths, and
|
||||
multiplies the aspect by 1.25, producing a result that causes
|
||||
it to trigger to 'landscape' choice. 'landscape' at this
|
||||
point is set to be the 'square' preset. And we get what we
|
||||
wanted.
|
||||
</p>
|
||||
</body>
|
||||
</html>
|
||||
|
@@ -0,0 +1,242 @@
|
||||
README
|
||||
------
|
||||
README for the Image effect text module.
|
||||
|
||||
Author Erwin Derksen (fietserwin: http://drupal.org/user/750928)
|
||||
|
||||
|
||||
Dependencies
|
||||
------------
|
||||
Hard dependencies:
|
||||
- Imagecache actions.
|
||||
- Image (Drupal core).
|
||||
|
||||
Soft dependencies/recommended modules:
|
||||
- Imagemagick (preferred toolkit, http://drupal.org/project/imagemagick).
|
||||
- PHP filter (Drupal core, if yuo want to use PHP to create the text to render).
|
||||
- System stream wrapper (http://drupal.org/project/system_stream_wrapper)
|
||||
- Remote stream wrapper (http://drupal.org/project/remote_stream_wrapper)
|
||||
The latter 2 provide additional stream wrappers. Especially the system stream
|
||||
wrapper is very handy as it provides, among others, a module:// and theme://
|
||||
wrapper.
|
||||
|
||||
|
||||
Toolkit
|
||||
-------
|
||||
Personally, I prefer the imagemagick toolkit:
|
||||
- It is better in anti-aliasing, try to rotate an image using both toolkits and
|
||||
you will see what I mean.
|
||||
- It does not execute in the PHP memory space, so is not restricted by the
|
||||
memory_limit PHP setting.
|
||||
- The GD toolkit will, at least on my Windows configuration, keep the font file
|
||||
open after a text operation, so you cannot delete, move or rename it anymore.
|
||||
- This module does a better job with Imagemagick (see below).
|
||||
|
||||
|
||||
Installing
|
||||
----------
|
||||
As usual. After enabling the module you can add texts to images. This image
|
||||
effect works with both the GD and imagemagick toolkit, though results differ
|
||||
depending on the toolkit you use.
|
||||
|
||||
|
||||
More information about the effect data options
|
||||
----------------------------------------------
|
||||
|
||||
Font
|
||||
----
|
||||
You have to supply the font file to use. The font type supported depend on the
|
||||
toolkit in use, but at least ttf files will always work. This option accepts
|
||||
either:
|
||||
- 1 of the (enabled) scheme's:
|
||||
* public://
|
||||
* private:// Preferred for site specific masks, overlays, etc, that do not
|
||||
need to be shared publicly.
|
||||
* temporary:// Unlikely to be useful, but supported anyway as all schemes are
|
||||
supported.
|
||||
* module:// Introduced by the system stream wrapper module and preferred for
|
||||
module provided resources.
|
||||
* theme:// idem.
|
||||
* profile:// idem.
|
||||
* library:// idem.
|
||||
- A relative (to the current directory, probably Drupal root) or absolute path.
|
||||
|
||||
|
||||
Text position
|
||||
-------------
|
||||
The text position defines the point in the image where you want to place (align)
|
||||
your text. It starts at the top left corner of the image with postion 0,0 and
|
||||
the positive directions are to the right and down.
|
||||
|
||||
The definition of the vertical position differs per toolkit. For GD it is the
|
||||
position of the font baseline, while for Imagemagick it is the bottom of the
|
||||
bounding box, i.e the descender or beard line in typography terminology.
|
||||
|
||||
|
||||
Text alignment
|
||||
--------------
|
||||
You can align your text with regard to the text position. Possible horizontal
|
||||
alignments are left (default), center and right. Vertical alignments are top,
|
||||
center and bottom (default).
|
||||
|
||||
Note: Given
|
||||
- the way that GD uses the vertical text position (as baseline),
|
||||
- and the way this module implements (vertical) alignment (translating the
|
||||
(vertical) position using the calculated bounding box),
|
||||
vertical alignment with the GD toolkit is a bit off. You will have to compensate
|
||||
for this yourself.
|
||||
|
||||
|
||||
Rotation
|
||||
--------
|
||||
The text can be rotated before being overlaid on the image. The vlaue is in
|
||||
degrees, so 90 degrees is straight down. Positive values are rotated clockwise,
|
||||
negative values counter clockwise.
|
||||
|
||||
In Imagemagick the text is rotated around the text position. Thus centered text
|
||||
is rotated around its own center. GD, on the other hand, always rotates around
|
||||
the left bottom (baseline) position, regardless the text alignment. Using
|
||||
rotation with a non default alignment (left bottom) will give surprising
|
||||
results.
|
||||
|
||||
|
||||
Text source
|
||||
-----------
|
||||
Note: this module is not build to handle multi line texts. Not when the text
|
||||
contains new lines and/or carriage returns, and not to split a given text over
|
||||
multiple lines given an available width. Using "\n" in GD seems to work though.
|
||||
|
||||
The text to place on the image may come from different sources:
|
||||
- Static: the text to place on the image is static and is defined in the image
|
||||
effect data. Use this e.g. for a fixed copyright notice.
|
||||
- PHP: the text to place on the image comes from a piece of PHP code that should
|
||||
return the text to place on the image. Only users with the 'use PHP for
|
||||
settings' permission are allowed to use this source. This permission and the
|
||||
evaluation of the PHP code come from the PHP filter module which is part of
|
||||
Drupal core and thus needs to be enabled, also during image generation.
|
||||
- To alleviate the need to enable the PHP filter module, 2 commonly used sources
|
||||
for dynamic texts are directly available without any coding, namely the alt
|
||||
and title properties of the image field linked to the image at hand. Note that
|
||||
multiple image fields, possibly in different languages, may be referring to
|
||||
the image that is being processed. This module will take the first image field
|
||||
it finds to extract the alt and title. If the field in itself is multi
|
||||
lingual, thus not a synced field, the current language will be taken, which is
|
||||
the language of the user that happens to request this styled image first.
|
||||
|
||||
|
||||
PHP snippets to determine the text
|
||||
----------------------------------
|
||||
Given the correct permission, you can write your own PHP snippet to compute the
|
||||
text to display. To ease this task, this module makes some information regarding
|
||||
the image being processed available in 2 variables: $image and $image_context.
|
||||
These variables are readily available in your snippet.
|
||||
|
||||
$image is an associative array containing:
|
||||
- source: string, the source of the image, e.g. public://photo.jpg
|
||||
- info: array, example data:
|
||||
- width (int) 180
|
||||
- height (int) 180
|
||||
- extension (string) png
|
||||
- mime_type (string) image/png
|
||||
- file_size (int) 4417
|
||||
- toolkit: string, imagemagick or GD
|
||||
|
||||
$image_context is an associative array containing:
|
||||
- effect_data: array, the data of this image effect, example data for the text
|
||||
effect:
|
||||
- size (string) 12
|
||||
- xpos (string) center
|
||||
- ypos (string) center
|
||||
- halign (string) left
|
||||
- valign (string) bottom
|
||||
- RGB Array [1]
|
||||
- HEX (string) 000000
|
||||
- alpha (string) 100
|
||||
- angle (string) 0
|
||||
- fontfile (string:10) lhandw.ttf
|
||||
- text_source (string) text
|
||||
- text (string) Hello World!
|
||||
- php (string) return 'Hello World!'
|
||||
- managed_file: object|null. A managed file object containing these properties:
|
||||
- fid (string) 2
|
||||
- uid (string) 1
|
||||
- filename (string) photo.jpg
|
||||
- uri (string) public://photo.jpg
|
||||
- filemime (string) image/jpeg
|
||||
- filesize (string) 445751
|
||||
- status (string) 1
|
||||
- timestamp (string) 1327525851
|
||||
- metatags Array [0]
|
||||
- rdf_mapping Array [0]
|
||||
- referring_entities: array|null. A nested array with (fully loaded) entities
|
||||
referring to the current image. The 1st level of entries is keyed by the field
|
||||
name, the 2nd by entity type, and the 3rd by entity id. Example data:
|
||||
- field_photo Array [1]
|
||||
- node Array [1]
|
||||
- 12 Object of: stdClass
|
||||
- nid (string) 12
|
||||
- vid (string) 12
|
||||
- type (string) page
|
||||
- author ...
|
||||
- timestamp ...
|
||||
- ...
|
||||
- entity: object|null, the 1st entity in referring_entities. This is for easy
|
||||
access to the referring entity if it may be assumed that only 1 entity is
|
||||
referring to the current image.
|
||||
- image_field: array|null, the 1st image field in entity that is referring to
|
||||
the current image. This is for easy access to the image field data if it may
|
||||
be assumed that only 1 image field is referring to the current image. Example
|
||||
data:
|
||||
- fid (int) 2
|
||||
- alt (string) ...
|
||||
- title (string) ...
|
||||
- ...
|
||||
|
||||
Of course there are many other possible useful globals. Think of:
|
||||
- base_url
|
||||
- base_path
|
||||
- base_root
|
||||
- is_https
|
||||
- user
|
||||
- language
|
||||
and of course $_SERVER and $_GET.
|
||||
|
||||
Using these information you can access entity data as follows:
|
||||
|
||||
Specific case (1 entity, of known entity_type, referring to the image):
|
||||
<?php
|
||||
$entity_type = 'node';
|
||||
$field_name = 'my_field';
|
||||
$entity = $image_context['entity'];
|
||||
$field = field_get_items($entity_type, $entity, $field_name);
|
||||
?>
|
||||
|
||||
Or the more general case (not knowing the referring type, or multiple entities
|
||||
that may be referring to the image):
|
||||
<?php
|
||||
$referring_entities = $image_context['referring_entities'];
|
||||
foreach ($referring_entities as $field_name => $field_referring_entities) {
|
||||
foreach ($field_referring_entities as $entity_type => $entities) {
|
||||
foreach ($entities as $entity_id => $entity) {
|
||||
$field = field_get_items($entity_type, $entity, $field_name);
|
||||
}
|
||||
}
|
||||
}
|
||||
?>
|
||||
|
||||
TODO
|
||||
----
|
||||
- Vertical alignment: add baseline as vertical alignment and make both toolkits
|
||||
behave the same for any given vertical alignment.
|
||||
- Rotation and alignment. Imagemagick seems to be more correct. Can GD made to
|
||||
do the same?
|
||||
- Language and alt/title: what if the first user to pass by and that generates
|
||||
the image is in a language that has no alt/title?
|
||||
- Newlines: seem to work in GD, not in Imagemagick.
|
||||
|
||||
To quote http://www.imagemagick.org/Usage/text/#draw:
|
||||
As of IM version 6.2.4, the "-draw text" operation no longer understands the use
|
||||
of '\n' as meaning newline, or the use of percent '%' image information escapes.
|
||||
(See Drawing a Percent Bug). These abilities, and problems, however remain
|
||||
available in the new IM v6 operator "-annotate". See the Annotate Text Drawing
|
||||
Operator below.
|
@@ -0,0 +1,602 @@
|
||||
<?php
|
||||
/**
|
||||
* Implementation of image_effects_text.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Real implementation of hook_help() called by image_effects_text_help().
|
||||
*
|
||||
* Experimental diagnostic page to assist locating valid fonts on the system.
|
||||
* Only tuned for Ubuntu so far. I've been unable do find ubiquitous tools that
|
||||
* provide useful font listings.'
|
||||
*/
|
||||
function image_effects_text_help_inc($path, $arg) {
|
||||
$output = "<p>
|
||||
For text rendering to work on a server, we <em>must</em>
|
||||
know the system path to the font <em>file</em>, not just the name.
|
||||
Font library handling differs too much on different systems and
|
||||
the available PHP toolkits are unable to return good diagnostics.
|
||||
</p><p>
|
||||
On Debian/Ubuntu, you may find your fonts in and under
|
||||
<code>/usr/share/fonts/truetype/</code>
|
||||
eg <code>'/usr/share/fonts/truetype/ttf-bitstream-vera/VeraMono.ttf'</code>
|
||||
</p><p>
|
||||
On OSX, they are probably in <code>/Library/Fonts/</code>
|
||||
eg <code>'/Library/Fonts/Times New Roman Bold Italic.ttf'</code>
|
||||
</p><p>
|
||||
On Windows, they are probably in <code>C:\\WINDOWS\Fonts\</code>
|
||||
eg <code>'C:\\WINDOWS\\Fonts\\comic.ttf'</code>
|
||||
</p><p>
|
||||
Of course, this will change if you deploy to a different server!
|
||||
so the best approach is to place your own TTF font file inside your private
|
||||
or public files directory and use that. Just give the filename with the
|
||||
'private://' or 'public://' scheme prefix and it should be found.
|
||||
</p>
|
||||
";
|
||||
$output .= t("<p>Files directory is !files</p>", array('!files' => variable_get('file_public_path', conf_path() . '/files')));
|
||||
|
||||
return $output;
|
||||
}
|
||||
|
||||
/**
|
||||
* Builds the form structure for the overlay text image effect.
|
||||
*
|
||||
* Note that this is not a complete form, it only contains the portion of the
|
||||
* form for configuring the effect options. Therefore it does not not need to
|
||||
* include metadata about the effect, nor a submit button.
|
||||
*
|
||||
* @param array $data
|
||||
* The current configuration for this image effect.
|
||||
*
|
||||
* @return array
|
||||
* The form definition for this effect.
|
||||
*/
|
||||
function image_effects_text_form_inc($data) {
|
||||
// Use of functions imagecache_file_...() creates a dependency on file utility.inc.
|
||||
module_load_include('inc', 'imagecache_actions', 'utility');
|
||||
// Use of function imagecache_rgb_form() creates a dependency on file utility-color.inc.
|
||||
module_load_include('inc', 'imagecache_actions', 'utility-color');
|
||||
|
||||
// Note: we also need to check for the existence of the module: admin has
|
||||
// all rights, so user_acccess(...) returns TRUE even if the module is not
|
||||
// enabled and the permission does not exist.
|
||||
// A user without the 'use PHP for settings' permission (defined by the core
|
||||
// PHP filter module) may not:
|
||||
// - Select the 'PHP code' text source option if it is currently not selected.
|
||||
// - Change the 'PHP code' textarea.
|
||||
$allow_dynamic = user_access('use PHP for settings') && module_exists('php');
|
||||
$defaults = array(
|
||||
'size' => 12,
|
||||
'angle' => 0,
|
||||
'xpos' => '0',
|
||||
'ypos' => '0',
|
||||
'halign' => 'left',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array('HEX' => '#000000'),
|
||||
'alpha' => 100,
|
||||
'fontfile' => 'lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => 'Hello World!',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
);
|
||||
$data += $defaults;
|
||||
$form = array(
|
||||
'size' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Font size'),
|
||||
'#default_value' => $data['size'],
|
||||
'#description' => t('The font size in points. Only in GD1 this is in pixels.'),
|
||||
'#size' => 3,
|
||||
),
|
||||
'xpos' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('X offset'),
|
||||
'#default_value' => $data['xpos'],
|
||||
'#description' => t('Enter an offset in pixels or use a keyword: <em>left</em>, <em>center</em>, or <em>right</em>. Syntax like <em>right-20</em> is also valid.'),
|
||||
'#size' => 10,
|
||||
),
|
||||
'ypos' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Y offset'),
|
||||
'#default_value' => $data['ypos'],
|
||||
'#description' => t('Enter an offset in pixels or use a keyword: <em>top</em>, <em>center</em>, or <em>bottom</em>. Syntax like <em>bottom-20</em> is also valid.'),
|
||||
'#size' => 10,
|
||||
),
|
||||
'halign' => array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Horizontal alignment'),
|
||||
'#default_value' => $data['halign'],
|
||||
'#description' => t('The horizontal alignment of the text around the given %xpos.', array('%xpos' => t('X offset'))),
|
||||
'#options' => array('left' => t('Left'), 'center' => t('Center'), 'right' => t('Right')),
|
||||
),
|
||||
'valign' => array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Vertical alignment'),
|
||||
'#default_value' => $data['valign'],
|
||||
'#description' => t('The vertical alignment of the text around the given %ypos.', array('%ypos' => t('Y offset'))),
|
||||
'#options' => array('top' => t('Top'), 'center' => t('Center'), 'bottom' => t('Bottom')),
|
||||
),
|
||||
'RGB' => imagecache_rgb_form($data['RGB']),
|
||||
'alpha' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Opacity'),
|
||||
'#default_value' => $data['alpha'] ? $data['alpha'] : 100,
|
||||
'#size' => 3,
|
||||
'#description' => t('Opacity: 1-100.'),
|
||||
),
|
||||
'angle' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Angle'),
|
||||
'#default_value' => $data['angle'],
|
||||
'#description' => t('Angle: The angle in degrees, with 0 degrees being left-to-right reading text. Higher values represent a counter-clockwise rotation. For example, a value of 90 would result in bottom-to-top reading text.'),
|
||||
'#size' => 3,
|
||||
),
|
||||
'fontfile' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Font file name'),
|
||||
'#default_value' => $data['fontfile'],
|
||||
'#description' => imagecache_actions_file_field_description(),
|
||||
'#element_validate' => array('imagecache_actions_validate_file'),
|
||||
),
|
||||
'text_help' => array(
|
||||
'#type' => 'item',
|
||||
'#title' => t('Text'),
|
||||
'#description' => t('<p>Select the source of the text:</p>
|
||||
<ul>
|
||||
<li><strong>Image alt</strong>: the alt text of an image field referring to this image is taken.</li>
|
||||
<li><strong>Image title</strong>: the title text of an image field referring to this image is taken.</li>
|
||||
<li><strong>Static text</strong>: a text that will be the same for each image, e.g. a copyright message. You can define the text in the text field below the drop down.</li>
|
||||
<li><strong>PHP code</strong>: a piece of PHP code that returns the text to display. You can define the PHP code in the text area below the drop down. You will need the \'%use_php\' permission, defined by the \'PHP filter\' module.</li>
|
||||
</ul>
|
||||
<p>See the help for an extensive explanation of the possibilities.</p>',
|
||||
array('%use_php' => t('Use PHP for settings'))),
|
||||
),
|
||||
'text_source' => array(
|
||||
'#type' => 'select',
|
||||
'#title' => t('Text source'),
|
||||
'#default_value' => $data['text_source'],
|
||||
'#options' => array('alt' => t('Image alt'), 'title' => t('Image title'), 'text' => t('Static text'), 'php' => t('PHP code')),
|
||||
),
|
||||
'text' => array(
|
||||
'#type' => 'textfield',
|
||||
'#title' => t('Text'),
|
||||
'#default_value' => $data['text'],
|
||||
'#states' => array(
|
||||
'visible' => array(':input[name="data[text_source]"]' => array('value' => 'text')),
|
||||
),
|
||||
),
|
||||
'php' => array(
|
||||
'#type' => 'textarea',
|
||||
'#rows' => 5,
|
||||
'#title' => t('PHP code'),
|
||||
'#default_value' => $data['php'],
|
||||
'#disabled' => !$allow_dynamic,
|
||||
'#states' => array(
|
||||
'visible' => array(':input[name="data[text_source]"]' => array('value' => 'php')),
|
||||
),
|
||||
),
|
||||
);
|
||||
if (!$allow_dynamic && $data['text_source'] !== 'php') {
|
||||
unset($form['text_fieldset']['text_source']['#options']['php']);
|
||||
}
|
||||
|
||||
$form['#element_validate'][] = 'image_effects_text_form_validate';
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Element validation callback for the effect form.
|
||||
* @see http://api.drupal.org/api/drupal/developer--topics--forms_api_reference.html/7#element_validate
|
||||
*/
|
||||
function image_effects_text_form_validate($element, &$form_state, $form) {
|
||||
if (!is_numeric($element['size']['#value']) || $element['size']['#value'] <= 0) {
|
||||
form_error($element['size'], t('%field must be a positive number.', array('%field' => t('Font size'))));
|
||||
}
|
||||
if (!is_numeric($element['alpha']['#value']) || $element['alpha']['#value'] < 0 || $element['alpha']['#value'] > 100) {
|
||||
form_error($element['alpha'], t('%field must be a number between 1 and 100.', array('%field' => t('Opacity'))));
|
||||
}
|
||||
if (!is_numeric($element['angle']['#value'])) {
|
||||
form_error($element['angle'], t('%field must be a number.', array('%field' => t('Angle'))));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation of theme_hook() for text image effect.
|
||||
*
|
||||
* @param array $variables
|
||||
* An associative array containing:
|
||||
* - data: The current configuration for this resize effect.
|
||||
*
|
||||
* @return string
|
||||
* The HTML for the summary of a text image effect.
|
||||
* @ingroup themeable
|
||||
*/
|
||||
function theme_image_effects_text_summary($variables) {
|
||||
$data = $variables['data'];
|
||||
switch ($data['text_source']) {
|
||||
case 'alt':
|
||||
$text = 'image alt';
|
||||
break;
|
||||
case 'title':
|
||||
$text = 'image title';
|
||||
break;
|
||||
case 'text':
|
||||
$text = $data['text'];
|
||||
break;
|
||||
case 'php':
|
||||
$text = 'PHP code';
|
||||
break;
|
||||
}
|
||||
return 'Text: ' . $text . '; Position: ' . $data['xpos'] . ',' . $data['ypos'] . '; Alignment: ' . $data['halign'] . ',' . $data['valign'];
|
||||
}
|
||||
|
||||
/**
|
||||
* (Real implementation of) Image effect callback; Overlay text on an image
|
||||
* resource.
|
||||
*
|
||||
* @param object $image
|
||||
* An image object returned by image_load().
|
||||
*
|
||||
* @param array $data
|
||||
* An array of attributes to use when performing the resize effect with the
|
||||
* following items:
|
||||
* - "width": An integer representing the desired width in pixels.
|
||||
* - "height": An integer representing the desired height in pixels.
|
||||
*
|
||||
* @return boolean
|
||||
* true on success, false on failure to apply the effect.
|
||||
*/
|
||||
function image_effects_text_effect_inc($image, $data) {
|
||||
// Use of imagecache_actions_hex2rgba() ,the imagecache_file_...() functions,
|
||||
// and imagecache_actions_get_image_context() create a dependency on
|
||||
// file utility.inc.
|
||||
module_load_include('inc', 'imagecache_actions', 'utility');
|
||||
|
||||
// Massage the data and pass it on to the toolkit dependent part.
|
||||
// Start with a straight copy.
|
||||
$params = $data;
|
||||
|
||||
// Find out where the font file is located and if it is readable.
|
||||
$params['fontpath'] = imagecache_actions_find_file($data['fontfile']);
|
||||
if ($params['fontpath'] === FALSE) {
|
||||
drupal_set_message(t("Failed to locate the requested font %fontfile. Cannot overlay text onto image.", array('%fontfile' => $data['fontfile'])), 'error');
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Get the text to overlay.
|
||||
$params['text'] = image_effects_text_get_text($image, $params);
|
||||
if ($params['text'] === FALSE) {
|
||||
drupal_set_message(t("Failed to evaluate text (is the 'PHP Filter' module enabled?). Not overlaying text."), 'error');
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Parse offsets
|
||||
$params['xpos'] = image_effects_text_get_offset($data['xpos'], $image->info['width'], $image->info['height'], $image->info['width']);
|
||||
$params['ypos'] = image_effects_text_get_offset($data['ypos'], $image->info['width'], $image->info['height'], $image->info['height']);
|
||||
|
||||
// Convert color from hex (as it is stored in the UI).
|
||||
$params['RGB'] = $data['RGB'];
|
||||
if($params['RGB']['HEX'] && $deduced = imagecache_actions_hex2rgba($params['RGB']['HEX'])) {
|
||||
$params['RGB'] += $deduced;
|
||||
}
|
||||
|
||||
// Make int's of various parameters
|
||||
$params['size'] = (int) $params['size'];
|
||||
$params['xpos'] = (int) $params['xpos'];
|
||||
$params['ypos'] = (int) $params['ypos'];
|
||||
// Hand over to toolkit
|
||||
return image_toolkit_invoke('image_effects_text', $image, array($params));
|
||||
}
|
||||
|
||||
/**
|
||||
* GD toolkit specific implementation of this image effect.
|
||||
*
|
||||
* @param object $image
|
||||
* @param array $params
|
||||
* An array containing the parameters for this effect.
|
||||
*
|
||||
* @return bool
|
||||
* true on success, false otherwise.
|
||||
*/
|
||||
function image_gd_image_effects_text($image, $params) {
|
||||
// Convert color and alpha to GD alpha and color value.
|
||||
// GD alpha value: 0 = opaque, 127 = transparent.
|
||||
$params['alpha'] = (int) ((1 - ($params['alpha'] / 100)) * 127);
|
||||
$color = imagecolorallocatealpha($image->resource, $params['RGB']['red'], $params['RGB']['green'], $params['RGB']['blue'], $params['alpha']);
|
||||
if ($color !== FALSE) {
|
||||
$bounds = NULL;
|
||||
// Adjust Y position for vertical alignment (if different from bottom).
|
||||
if ($params['valign'] !== 'bottom') {
|
||||
// Get bounding box.
|
||||
// PHP Manual: "This function requires both the GD library and the » FreeType library."
|
||||
// So it is not more demanding than imagettftext, which we need anyway.
|
||||
$bounds = imagettfbbox($params['size'], 0, $params['fontpath'], $params['text']);
|
||||
if ($bounds === FALSE) {
|
||||
drupal_set_message(t('Failed to calculate text dimensions using GD toolkit. Ignoring the alignment settings.'), 'warning');
|
||||
}
|
||||
else {
|
||||
// Get height of bounding box.
|
||||
$height = $bounds[1] - $bounds[7];
|
||||
// Shift ypos down (full height on bottom, half the height on center).
|
||||
$params['ypos'] += $params['valign'] === 'center' ? (int) ($height/2) : $height;
|
||||
}
|
||||
}
|
||||
// Adjust X position for horizontal alignment (if different from left).
|
||||
if ($params['halign'] !== 'left') {
|
||||
// Get bounding box.
|
||||
// PHP Manual: "This function requires both the GD library and the » FreeType library."
|
||||
// So it is not more demanding than imagettftext, which we need anyway.
|
||||
if ($bounds === NULL) {
|
||||
$bounds = imagettfbbox($params['size'], 0, $params['fontpath'], $params['text']);
|
||||
if ($bounds === FALSE) {
|
||||
drupal_set_message(t('Failed to calculate text dimensions using GD toolkit. Ignoring the alignment.'), 'warning');
|
||||
}
|
||||
}
|
||||
if ($bounds !== FALSE) {
|
||||
// Get width of bounding box.
|
||||
$width = $bounds[2] - $bounds[0];
|
||||
// Shift xpos to the left (full width on right, half the width on center).
|
||||
$params['xpos'] -= $params['halign'] === 'center' ? (int) ($width/2) : $width;
|
||||
}
|
||||
}
|
||||
|
||||
// PHP Manual: "This function requires both the GD library and the » FreeType library."
|
||||
$bounds = imagettftext($image->resource, $params['size'], $params['angle'], $params['xpos'], $params['ypos'], $color, $params['fontpath'], $params['text']);
|
||||
return $bounds !== FALSE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Imagemagick toolkit specific implementation of this image effect.
|
||||
*
|
||||
* Text in Imagemagick:
|
||||
* @link http://www.imagemagick.org/script/command-line-options.php?#draw
|
||||
* @link http://www.imagemagick.org/script/command-line-options.php?#annotate
|
||||
*
|
||||
* UTF-8/non-ascii characters
|
||||
* Though the online imagemagick manual mentions some problems with accented
|
||||
* characters, it worked fine for me in a Windows Vista shell. TBC on other
|
||||
* OS'es (including linux)
|
||||
* (@see: http://www.imagemagick.org/Usage/windows/#character_encoding)
|
||||
*
|
||||
* Alignment in Imagemagick:
|
||||
* This is not directly supported, though a justicifcation option has been
|
||||
* proposed: @link http://www.imagemagick.org/Usage/bugs/future/#justification.
|
||||
*
|
||||
* What we do have is the gravity option:
|
||||
* @link http://www.imagemagick.org/Usage/annotating/#gravity
|
||||
* Gravity is used to position a text, but it also automatically applies a
|
||||
* justification based on that placement. So we use gravity here for alignment,
|
||||
* but will thus have to rebase our positioning.
|
||||
*
|
||||
* Gravity |halign|valign |hpos change|vpos change
|
||||
* ------------------------------------------------
|
||||
* NorthWest left top 0 0
|
||||
* North center top -width/2 0
|
||||
* NorthEast right top -width 0
|
||||
* West left center 0 -height/2
|
||||
* Center center center -width/2 -height/2
|
||||
* East right center -width -height/2
|
||||
* SouthWest left bottom 0 -height
|
||||
* South center bottom -width/2 -height
|
||||
* SouthEast right bottom -width -height
|
||||
*/
|
||||
function image_imagemagick_image_effects_text($image, $params) {
|
||||
static $alignments2gravity = array(
|
||||
'left' => array(
|
||||
'top' => array(
|
||||
'gravity' => 'NorthWest',
|
||||
'tx' => 0,
|
||||
'ty' => 0,
|
||||
),
|
||||
'center' => array(
|
||||
'gravity' => 'West',
|
||||
'tx' => 0,
|
||||
'ty' => -0.5,
|
||||
),
|
||||
'bottom' => array(
|
||||
'gravity' => 'SouthWest',
|
||||
'tx' => 0,
|
||||
'ty' => 1, // reversed translation
|
||||
),
|
||||
),
|
||||
'center' => array(
|
||||
'top' => array(
|
||||
'gravity' => 'North',
|
||||
'tx' => -0.5,
|
||||
'ty' => 0,
|
||||
),
|
||||
'center' => array(
|
||||
'gravity' => 'Center',
|
||||
'tx' => -0.5,
|
||||
'ty' => -0.5,
|
||||
),
|
||||
'bottom' => array(
|
||||
'gravity' => 'South',
|
||||
'tx' => -0.5,
|
||||
'ty' => 1, // reversed translation
|
||||
),
|
||||
),
|
||||
'right' => array(
|
||||
'top' => array(
|
||||
'gravity' => 'NorthEast',
|
||||
'tx' => 1, // reversed translation
|
||||
'ty' => 0,
|
||||
),
|
||||
'center' => array(
|
||||
'gravity' => 'East',
|
||||
'tx' => 1, // reversed translation
|
||||
'ty' => -0.5,
|
||||
),
|
||||
'bottom' => array(
|
||||
'gravity' => 'SouthEast',
|
||||
'tx' => 1, // reversed translation
|
||||
'ty' => 1, // reversed translation
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
// Convert color and alpha to Imagemagick rgba color argument.
|
||||
$alpha = $params['alpha'] / 100;
|
||||
$color = 'rgba(' . $params['RGB']['red']. ',' . $params['RGB']['green'] . ',' . $params['RGB']['blue'] . ','. $alpha . ')';
|
||||
|
||||
// Alignment
|
||||
$alignment_corrections = $alignments2gravity[$params['halign']][$params['valign']];
|
||||
$gravity = $alignment_corrections['gravity'];
|
||||
if ($alignment_corrections['tx'] > 0) {
|
||||
$params['xpos'] = (int) ($alignment_corrections['tx'] * $image->info['width'] - $params['xpos']);
|
||||
}
|
||||
else {
|
||||
$params['xpos'] += (int) ($alignment_corrections['tx'] * $image->info['width']);
|
||||
}
|
||||
if ($alignment_corrections['ty'] > 0) {
|
||||
$params['ypos'] = (int) ($alignment_corrections['ty'] * $image->info['height'] - $params['ypos']);
|
||||
}
|
||||
else {
|
||||
$params['ypos'] += (int) ($alignment_corrections['ty'] * $image->info['height']);
|
||||
}
|
||||
|
||||
// Define the quote to use around the text. This is part of the argument of
|
||||
// the -draw command and thus should NOT be the shell argument enclosing
|
||||
// character.
|
||||
$quote = strstr($_SERVER['SERVER_SOFTWARE'], 'Win32') || strstr($_SERVER['SERVER_SOFTWARE'], 'IIS') ? "'" : '"';
|
||||
// and subsequently escape the use of that quote within the text.
|
||||
$text = $params['text'];
|
||||
$text = str_replace($quote, "\\$quote", $text);
|
||||
|
||||
$image->ops[] = '-font ' . escapeshellarg($params['fontpath']);
|
||||
$image->ops[] = "-pointsize {$params['size']}";
|
||||
$image->ops[] = '-fill ' . escapeshellarg($color);
|
||||
// See issue http://drupal.org/node/1561214, Bootstrap should reset locale settings to UTF-8.
|
||||
setlocale(LC_ALL, 'C.UTF-8');
|
||||
$image->ops[] = '-draw ' . escapeshellarg("gravity $gravity translate {$params['xpos']},{$params['ypos']} rotate {$params['angle']} text 0,0 $quote$text$quote");
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* UTILITY
|
||||
*/
|
||||
|
||||
/**
|
||||
* Convert a position into an offset in pixels.
|
||||
*
|
||||
* Position may be a number of additions and/or subtractions of:
|
||||
* - An value, positive or negative, in pixels.
|
||||
* - A, positive or negative, percentage (%). The given percentage of the
|
||||
* current dimension will be taken.
|
||||
* - 1 of the keywords:
|
||||
* * top: 0
|
||||
* * bottom: the height of the current image
|
||||
* * left: 0
|
||||
* * right: the width of the current image
|
||||
* * center: 50% (of the current dimension)
|
||||
* Examples:
|
||||
* 0, 20, -20, 90%, 33.3% + 10, right, center - 20, 300 - center, bottom - 50.
|
||||
* Note:
|
||||
* The algorithm will accept many more situations, though the result may be hard
|
||||
* to predict.
|
||||
*
|
||||
* @param int $width
|
||||
* The length of the horizontal dimension.
|
||||
* @param int $height
|
||||
* The length of the vertical dimension.
|
||||
* @param int $length
|
||||
* The length of the current dimension (should be either width or height).
|
||||
* @param string $position
|
||||
* The string defining the position.
|
||||
*
|
||||
* @return number
|
||||
* The computed offset in pixels.
|
||||
*/
|
||||
function image_effects_text_get_offset($position, $width, $height, $length) {
|
||||
$value = 0;
|
||||
|
||||
$tokens = preg_split('/ *(-|\+) */', $position, 0, PREG_SPLIT_DELIM_CAPTURE);
|
||||
$sign = 1;
|
||||
foreach ($tokens as $token) {
|
||||
switch ($token) {
|
||||
case '+';
|
||||
// Ignore, doesn't change the sign
|
||||
break;
|
||||
case '-';
|
||||
// Flip the sign.
|
||||
$sign = -$sign;
|
||||
break;
|
||||
case 'top':
|
||||
case 'left':
|
||||
// Actually, top and left are a no-op.
|
||||
$value += $sign * 0;
|
||||
$sign = 1;
|
||||
break;
|
||||
case 'bottom':
|
||||
// Use height of the image, even if this is for the horizontal position.
|
||||
$value += $sign * $height;
|
||||
$sign = 1;
|
||||
break;
|
||||
case 'right':
|
||||
// Use width of the image, even if this is for the vertical position.
|
||||
$value += $sign * $width;
|
||||
$sign = 1;
|
||||
break;
|
||||
case 'center':
|
||||
// half the current dimension as provided by $length.
|
||||
$value += $sign * $length/2;
|
||||
$sign = 1;
|
||||
break;
|
||||
default:
|
||||
// Value: absolute or percentage
|
||||
if (substr($token, -strlen('%')) === '%') {
|
||||
$percentage = ((float) substr($token, 0, -strlen('%'))) / 100.0;
|
||||
$value += $sign * ($percentage * $length);
|
||||
}
|
||||
else {
|
||||
$value += $sign * (float) $token;
|
||||
}
|
||||
$sign = 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the text to use for this image.
|
||||
*
|
||||
* @param object $image
|
||||
* The image the current effect is to be applied to.
|
||||
* @param array $data
|
||||
* An array containing the effect data.
|
||||
*
|
||||
* @return string
|
||||
* Plain string to be placed on the image.
|
||||
*/
|
||||
function image_effects_text_get_text($image, $data) {
|
||||
if ($data['text_source'] === 'text') {
|
||||
$text = $data['text'];
|
||||
}
|
||||
else {
|
||||
// Get context about the image.
|
||||
$image_context = imagecache_actions_get_image_context($image, $data);
|
||||
|
||||
if ($data['text_source'] === 'alt' || $data['text_source'] === 'title') {
|
||||
// Existence of an image field is not guaranteed, so check for that first.
|
||||
$text = isset($image_context['image_field'][$data['text_source']]) ? $image_context['image_field'][$data['text_source']] : '';
|
||||
}
|
||||
else { // $data['text_source'] === 'php'
|
||||
// Process the php using php_eval (rather than eval), but with GLOBAL
|
||||
// variables, so they can be passed successfully.
|
||||
$GLOBALS['image_context'] = $image_context;
|
||||
$GLOBALS['image'] = $image;
|
||||
// We don't need to check_plain() the resulting text, as the text is not
|
||||
// rendered in a browser but processed on the server.
|
||||
$text = module_exists('php') ? php_eval('<'.'?php global $image, $image_context; ' . $data['php'] . ' ?'.'>') : '';
|
||||
|
||||
unset($GLOBALS['image']);
|
||||
unset($GLOBALS['image_context']);
|
||||
}
|
||||
}
|
||||
return $text;
|
||||
}
|
@@ -0,0 +1,14 @@
|
||||
name = Image Text Effects
|
||||
description = Display simple or dynamic captions on images.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = image
|
||||
dependencies[] = imagecache_actions
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,25 @@
|
||||
<?php
|
||||
/**
|
||||
* @file Set up new text effects.
|
||||
*
|
||||
* @todo: is this cache we are clearing still in use?
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_install().
|
||||
*
|
||||
* Need to flush the cache when this module is enabled or disabled.
|
||||
*/
|
||||
function image_effects_text_install() {
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
drupal_set_message(t('Additional image effects to add text should now be available in the effects list on !settings_link', array('!settings_link' => l(t('settings'), 'admin/config/media/image-styles'))));
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_uninstall().
|
||||
*
|
||||
* This hook implementation clears the imagecache_actions cache.
|
||||
*/
|
||||
function image_effects_text_uninstall() {
|
||||
cache_clear_all('imagecache_actions', 'cache');
|
||||
}
|
@@ -0,0 +1,84 @@
|
||||
<?php
|
||||
/**
|
||||
* @file Provide text manipulation effects for image styles.
|
||||
*
|
||||
* Ported by dman
|
||||
* from http://drupal.org/node/264862#comment-865490 by patrickharris
|
||||
*
|
||||
* Ported to D7 by fietserwin
|
||||
* from imagecache_textactions 6.x-1.8.
|
||||
* - The module has been renamed to follow D7 terminology:
|
||||
* * imagecache -> image
|
||||
* * action(s) -> effect(s)
|
||||
* resulting in image_effects_text.
|
||||
* - The .module file is kept as small as possible. The real work is done in the
|
||||
* .inc file.
|
||||
* - Function and parameter naming has been changed to match the image effects
|
||||
* in the core image module.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_image_effect_info().
|
||||
*
|
||||
* Defines information about the supported effects.
|
||||
*/
|
||||
function image_effects_text_image_effect_info() {
|
||||
$effects = array();
|
||||
|
||||
$effects['image_effects_text'] = array(
|
||||
'label' => t('Text'),
|
||||
'help' => t('Add static or dynamic (coded) text to an image.'),
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
);
|
||||
|
||||
return $effects;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_help().
|
||||
*/
|
||||
function image_effects_text_help($path, $arg) {
|
||||
if ($path === 'admin/advanced_help' && count($arg) >= 3 && $arg[2] === 'image_effects_text'
|
||||
|| $path === 'admin/help/image_effects_text') {
|
||||
module_load_include('inc', 'image_effects_text', 'image_effects_text');
|
||||
return image_effects_text_help_inc($path, $arg);
|
||||
}
|
||||
else if ($path === 'admin/help#image_effects_text') {
|
||||
// This path just checks if there is (non-empty) help, so it can place a
|
||||
// link to it.
|
||||
return ' ';
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_theme().
|
||||
*
|
||||
* We register theme functions for the effect summaries.
|
||||
*/
|
||||
function image_effects_text_theme() {
|
||||
return array(
|
||||
'image_effects_text_summary' => array(
|
||||
'variables' => array('data' => NULL),
|
||||
'file' => 'image_effects_text.inc'
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Builds the form structure for the overlay text image effect.
|
||||
*/
|
||||
function image_effects_text_form($data) {
|
||||
module_load_include('inc', 'image_effects_text', 'image_effects_text');
|
||||
return image_effects_text_form_inc($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* Callback to perform the image effect on the given image.
|
||||
*/
|
||||
function image_effects_text_effect($image, $data) {
|
||||
module_load_include('inc', 'image_effects_text', 'image_effects_text');
|
||||
return image_effects_text_effect_inc($image, $data);
|
||||
}
|
@@ -0,0 +1,18 @@
|
||||
README
|
||||
------
|
||||
README for the Image effect text test module.
|
||||
|
||||
This module contains 2 image styles to test text effects. It uses an image
|
||||
containing a grid and a font which are included in the install package as well.
|
||||
|
||||
Hard Dependencies
|
||||
-----------------
|
||||
Hard dependencies:
|
||||
- Imagecache actions.
|
||||
- Image (Drupal core).
|
||||
- Features
|
||||
- System stream wrapper (http://drupal.org/project/system_stream_wrapper)
|
||||
|
||||
Soft Dependencies
|
||||
-----------------
|
||||
- Imagemagick (preferred toolkit, http://drupal.org/project/imagemagick).
|
After Width: | Height: | Size: 1.8 KiB |
@@ -0,0 +1,483 @@
|
||||
<?php
|
||||
/**
|
||||
* @file
|
||||
* image_effects_text_test.features.inc
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_image_default_styles().
|
||||
*/
|
||||
function image_effects_text_test_image_default_styles() {
|
||||
$styles = array();
|
||||
|
||||
// Exported image style: text-rotate-test.
|
||||
$styles['text-rotate-test'] = array(
|
||||
'name' => 'text-rotate-test',
|
||||
'effects' => array(
|
||||
32 => array(
|
||||
'label' => 'Echelle',
|
||||
'help' => 'La mise à l\'échelle maintiendra les proportions originales de l\'image. Si une seule dimension est précisée, l\'autre dimension sera calculée automatiquement.',
|
||||
'effect callback' => 'image_scale_effect',
|
||||
'dimensions callback' => 'image_scale_dimensions',
|
||||
'form callback' => 'image_scale_form',
|
||||
'summary theme' => 'image_scale_summary',
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'data' => array(
|
||||
'width' => '800',
|
||||
'height' => '',
|
||||
'upscale' => 0,
|
||||
),
|
||||
'weight' => '1',
|
||||
),
|
||||
33 => array(
|
||||
'label' => 'Overlay (watermark)',
|
||||
'help' => 'Choose the file image you wish to use as an overlay, and position it in a layer on top of the canvas.',
|
||||
'effect callback' => 'canvasactions_file2canvas_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_file2canvas_form',
|
||||
'summary theme' => 'canvasactions_file2canvas_summary',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array(
|
||||
'xpos' => 0,
|
||||
'ypos' => 0,
|
||||
'alpha' => '100',
|
||||
'path' => 'module://image_effects_text_test/grid800x600.png',
|
||||
),
|
||||
'weight' => '2',
|
||||
),
|
||||
34 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '100',
|
||||
'ypos' => '200',
|
||||
'halign' => 'left',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '45',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '100,200 45 left,bottom',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '3',
|
||||
),
|
||||
35 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '300',
|
||||
'ypos' => '200',
|
||||
'halign' => 'center',
|
||||
'valign' => 'center',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '45',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '300,200 45 center,center',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '4',
|
||||
),
|
||||
37 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '500',
|
||||
'ypos' => '200',
|
||||
'halign' => 'right',
|
||||
'valign' => 'top',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '45',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '500,200 45 right,top',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '5',
|
||||
),
|
||||
38 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '100',
|
||||
'ypos' => '500',
|
||||
'halign' => 'left',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '-30',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '100,500 -30 left,bottom',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '7',
|
||||
),
|
||||
39 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '300',
|
||||
'ypos' => '500',
|
||||
'halign' => 'center',
|
||||
'valign' => 'center',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '-30',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '300,500 -30 center,center',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '8',
|
||||
),
|
||||
40 => array(
|
||||
'label' => 'Texte',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '600',
|
||||
'ypos' => '400',
|
||||
'halign' => 'right',
|
||||
'valign' => 'top',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '-30',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '600,400 -30 right,top',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '9',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
// Exported image style: text-test.
|
||||
$styles['text-test'] = array(
|
||||
'name' => 'text-test',
|
||||
'effects' => array(
|
||||
30 => array(
|
||||
'label' => 'Overlay (watermark)',
|
||||
'help' => 'Choose the file image you wish to use as an overlay, and position it in a layer on top of the canvas.',
|
||||
'effect callback' => 'canvasactions_file2canvas_image',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'canvasactions_file2canvas_form',
|
||||
'summary theme' => 'canvasactions_file2canvas_summary',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array(
|
||||
'xpos' => 'left',
|
||||
'ypos' => 'top',
|
||||
'alpha' => '100',
|
||||
'path' => 'module://image_effects_text_test/grid800x600.png',
|
||||
),
|
||||
'weight' => '-10',
|
||||
),
|
||||
31 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '100',
|
||||
'ypos' => '200',
|
||||
'halign' => 'left',
|
||||
'valign' => 'top',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '100,200 left,top',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '-8',
|
||||
),
|
||||
32 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '300',
|
||||
'ypos' => '200',
|
||||
'halign' => 'left',
|
||||
'valign' => 'center',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '300,200 left,center',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '3',
|
||||
),
|
||||
33 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '500',
|
||||
'ypos' => '200',
|
||||
'halign' => 'left',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '500,200 left,bottom',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '4',
|
||||
),
|
||||
34 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '100',
|
||||
'ypos' => '400',
|
||||
'halign' => 'center',
|
||||
'valign' => 'top',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '100,400 center,top',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '5',
|
||||
),
|
||||
35 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '300',
|
||||
'ypos' => '400',
|
||||
'halign' => 'center',
|
||||
'valign' => 'center',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '300,400 center,center',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '6',
|
||||
),
|
||||
36 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '500',
|
||||
'ypos' => '400',
|
||||
'halign' => 'center',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '500,400 center, bottom',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '7',
|
||||
),
|
||||
37 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '100',
|
||||
'ypos' => '500',
|
||||
'halign' => 'right',
|
||||
'valign' => 'top',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '100,500 right,top',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '8',
|
||||
),
|
||||
38 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '300',
|
||||
'ypos' => '500',
|
||||
'halign' => 'right',
|
||||
'valign' => 'center',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '300,500 right,center',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '9',
|
||||
),
|
||||
39 => array(
|
||||
'label' => 'Tekst',
|
||||
'help' => 'Add static or dynamic (coded) text to an image.',
|
||||
'dimensions passthrough' => TRUE,
|
||||
'form callback' => 'image_effects_text_form',
|
||||
'effect callback' => 'image_effects_text_effect',
|
||||
'summary theme' => 'image_effects_text_summary',
|
||||
'module' => 'image_effects_text',
|
||||
'name' => 'image_effects_text',
|
||||
'data' => array(
|
||||
'size' => '18',
|
||||
'xpos' => '500',
|
||||
'ypos' => '500',
|
||||
'halign' => 'right',
|
||||
'valign' => 'bottom',
|
||||
'RGB' => array(
|
||||
'HEX' => '000000',
|
||||
),
|
||||
'alpha' => '100',
|
||||
'angle' => '0',
|
||||
'fontfile' => 'module://image_effects_text_test/lhandw.ttf',
|
||||
'text_source' => 'text',
|
||||
'text' => '500,500 right,bottom',
|
||||
'php' => 'return \'Hello World!\'',
|
||||
),
|
||||
'weight' => '10',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
return $styles;
|
||||
}
|
@@ -0,0 +1,19 @@
|
||||
name = Image Effects Text test
|
||||
description = Image effects that test the text effect
|
||||
core = 7.x
|
||||
package = Features
|
||||
php = 5.2.4
|
||||
project = image_effects_text_test
|
||||
dependencies[] = image
|
||||
dependencies[] = image_effects_text
|
||||
dependencies[] = system_stream_wrapper
|
||||
features[features_api][] = api:1
|
||||
features[image][] = text-rotate-test
|
||||
features[image][] = text-test
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,7 @@
|
||||
<?php
|
||||
/**
|
||||
* @file
|
||||
* Code for the Image Effects Text test feature.
|
||||
*/
|
||||
|
||||
include_once('image_effects_text_test.features.inc');
|
182
sites/all/modules/imagecache_actions/image_overlay.inc
Normal file
@@ -0,0 +1,182 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file extension to imageapi, provide an overlay action for blending two
|
||||
* layers, preserving transparency.
|
||||
*/
|
||||
|
||||
// Not sure where this library will live between upgrade versions.
|
||||
// Be careful to not conflict with another copy of myself.
|
||||
if (! function_exists('image_overlay')) {
|
||||
|
||||
/**
|
||||
* Place one image over another
|
||||
*
|
||||
* @param object $image
|
||||
* An image object.
|
||||
* @param object $overlay
|
||||
* An image object.
|
||||
* @param $x
|
||||
* Position of the overlay
|
||||
* @param $y
|
||||
* Position of the overlay
|
||||
* @param $alpha
|
||||
* Transparency of the overlay from 0-100. 0 is totally transparent. 100
|
||||
* (default) is totally opaque.
|
||||
* @param $reverse
|
||||
* BOOL flag to indicate the 'overlay' actually goes under the image. As
|
||||
* the imageapi callbacks modify the $image object by reference, this is needed
|
||||
* to replace the old image resource with the new one.
|
||||
* @return bool success
|
||||
*
|
||||
* @ingroup imageapi
|
||||
*/
|
||||
function image_overlay($image, &$layer, $x, $y, $alpha = 100, $reverse = FALSE) {
|
||||
if ($reverse) {
|
||||
$x = imagecache_actions_keyword_filter($x, $layer->info['width'], $image->info['width']);
|
||||
$y = imagecache_actions_keyword_filter($y, $layer->info['height'], $image->info['height']);
|
||||
}
|
||||
else {
|
||||
$x = imagecache_actions_keyword_filter($x, $image->info['width'], $layer->info['width']);
|
||||
$y = imagecache_actions_keyword_filter($y, $image->info['height'], $layer->info['height']);
|
||||
}
|
||||
return image_toolkit_invoke('overlay', $image, array($layer, $x, $y, $alpha, $reverse));
|
||||
}
|
||||
|
||||
/**
|
||||
* Place one image over another
|
||||
* This modifies the passed image by reference
|
||||
*
|
||||
* This func is nominated for inclusion in imageapi package. Until then, we do
|
||||
* it ourselves.
|
||||
*
|
||||
* NOTE that the PHP libraries are not great at merging images SO we include a
|
||||
* library that does it pixel-by-pixel which is INCREDIBLY inefficient. If this
|
||||
* can be improved, in a way that supports all transparency, please let us know!
|
||||
*
|
||||
* A watermark is layer onto image, return the image. An underlay is image onto
|
||||
* layer, return the layer. Almost identical, but seeing as we work with
|
||||
* resource handles, the handle needs to be swapped before returning.
|
||||
*
|
||||
* @ingroup imageapi
|
||||
* @param $image
|
||||
* Base imageapi object.
|
||||
* @param $overlay
|
||||
* May be a filename or an imageAPI object
|
||||
* @param $x
|
||||
* Position of the overlay
|
||||
* @param $y
|
||||
* Position of the overlay
|
||||
* @param $alpha
|
||||
* Transparency of the overlay from 0-100. 0 is totally transparent. 100
|
||||
* (default) is totally opaque.
|
||||
* @param $reverse
|
||||
* BOOL flag to indicate the 'overlay' actually goes under the image. As
|
||||
* the imageapi callbacks modify the $image object by reference, this is needed
|
||||
* to replace the old image resource with the new one.
|
||||
* @return bool success
|
||||
*/
|
||||
function image_gd_overlay($image, $layer, $x, $y, $alpha = 100, $reverse = FALSE) {
|
||||
if (empty($layer->resource)) {
|
||||
trigger_error("Invalid input to " . __FUNCTION__ . " 'layer' is not a valid resource");
|
||||
#dpm($layer);
|
||||
return FALSE;
|
||||
}
|
||||
// If the given alpha is 100%, we can use imagecopy - which actually works,
|
||||
// Is more efficient, and seems to retain the overlays partial transparancy
|
||||
// Still does not work great for indexed gifs though?
|
||||
if ($reverse) {
|
||||
$upper = &$image;
|
||||
$lower = &$layer;
|
||||
}
|
||||
else {
|
||||
$upper = &$layer;
|
||||
$lower = &$image;
|
||||
}
|
||||
if ($alpha == 100 && ($upper->info['mime_type'] != 'image/gif')) {
|
||||
imagealphablending($lower->resource, TRUE);
|
||||
imagesavealpha($lower->resource, TRUE);
|
||||
imagealphablending($upper->resource, TRUE);
|
||||
imagesavealpha($upper->resource, TRUE);
|
||||
imagecopy($lower->resource, $upper->resource, $x, $y, 0, 0, $upper->info['width'], $upper->info['height']);
|
||||
imagedestroy($upper->resource);
|
||||
$image->resource = $lower->resource;
|
||||
$image->info = $lower->info;
|
||||
}
|
||||
else {
|
||||
// Else imagecopymerge fails and we have to use the slow library.
|
||||
module_load_include('inc', 'imagecache_actions', 'watermark');
|
||||
$watermark = new watermark();
|
||||
$result_img = $watermark->create_watermark($lower->resource, $upper->resource, $x, $y, $alpha);
|
||||
// Watermark creates a new image resource, so clean up both old images.
|
||||
imagedestroy($lower->resource);
|
||||
imagedestroy($upper->resource);
|
||||
$image->resource = $result_img;
|
||||
$image->info = $lower->info;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Improvements on this are welcomed!
|
||||
*
|
||||
* Please be aware of the limitations of imagemagick libraries out there - the
|
||||
* versions distributed on hosted servers (if any) are often several years
|
||||
* behind. Using the latest imagemagick release features will make this function
|
||||
* unusable in real deployments.
|
||||
*
|
||||
* @param $image
|
||||
* Base imageapi object.
|
||||
* @param $layer
|
||||
* May be a filename or an imageAPI object, gets placed on top
|
||||
* If using reverse, this is going to be the result we carry on working with.
|
||||
*/
|
||||
function image_imagemagick_overlay($image, $layer, $x = 0, $y = 0, $alpha = 100, $reverse = FALSE) {
|
||||
// In imagemagick terms:
|
||||
// - $image is the destination (the image being constructed)
|
||||
// - $layer is the source (the source of the current operation)
|
||||
// Add the layer image to the imagemagick command line.
|
||||
$image->ops[] = escapeshellarg($layer->source) . ' ';
|
||||
|
||||
// Set its offset. Offset arguments require a sign in front.
|
||||
if ($x >= 0) {
|
||||
$x = "+$x";
|
||||
}
|
||||
if ($y >= 0) {
|
||||
$y = "+$y";
|
||||
}
|
||||
$image->ops[] = " -geometry $x$y";
|
||||
|
||||
// And compose it with the destination.
|
||||
if ($alpha == 100) {
|
||||
// Lay one image over the other. The transparency channel of the upper
|
||||
// image and/or its dimensions (being smaller than the lower image) will
|
||||
// determine what remains visible of the lower image).
|
||||
//
|
||||
// Note: In explicitly setting a -compose operator we reset/overwrite any
|
||||
// previously set one (former versions could produce erroneous results
|
||||
// in combination with other effects before this one).
|
||||
if ($reverse) {
|
||||
$compose_operator = 'dst-over';
|
||||
}
|
||||
else {
|
||||
$compose_operator = 'src-over';
|
||||
}
|
||||
$image->ops[] = "-compose $compose_operator -composite ";
|
||||
}
|
||||
else {
|
||||
// Alpha is not 100, so this image effect turns into a blend operation.
|
||||
// The alpha determines what percentage of the upper image pixel will be
|
||||
// taken. From the lower image pixel, 100 - alpha percent will be taken.
|
||||
//
|
||||
// Note 1: I'm not sure if and how transparency of one or both images is
|
||||
// used in or after the blend operation.
|
||||
// Note 2: As of IM v6.5.3-4 (around june 2009) we can use:
|
||||
// -compose blend -define compose:args=30[,70]
|
||||
$image->ops[] = "-compose blend -define compose:args=$alpha -composite ";
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
}
|
@@ -0,0 +1,38 @@
|
||||
README for the Image styleds admin Drupal module
|
||||
------------------------------------------------
|
||||
|
||||
Project page: http://drupal.org/project/imagecache_actions
|
||||
|
||||
Current and past maintainers for Image styles admin:
|
||||
- fietserwin (http://drupal.org/user/750928)
|
||||
|
||||
|
||||
Release notes for 7.x-1.x-dev
|
||||
-----------------------------
|
||||
- Clear the (menu) cache after installing or updating.
|
||||
|
||||
|
||||
Introduction
|
||||
------------
|
||||
The Image style admin module extends the administrative interface for image
|
||||
styles by providing additional features.
|
||||
|
||||
Currently a duplicate, import and export image style feature are implemented.
|
||||
More features may be added in the future. These features typically allow you to
|
||||
more easily handle image styles. It allows us to more easily set up
|
||||
a test/showcase sute of styles. Finally, it allows everybody to test D8 image
|
||||
module features in real life.
|
||||
|
||||
This module is not a replacement for the features module
|
||||
(http://drupal.org/project/features). If you are serious about configuration
|
||||
management and want to distribute styles to other systems, use features.
|
||||
|
||||
Use this module for 1 time export/imports between different sites, "copy &
|
||||
paste" reuse within a site, and when reporting issues to the imagecache_actions
|
||||
issue queue.
|
||||
|
||||
|
||||
TODO
|
||||
----
|
||||
Solving errors in the core image handling?
|
||||
- [#1554074]: scale does not work with imagemagick when dimensions are unknown?
|
@@ -0,0 +1,226 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Include file for image_styles_admin routines that do not need to be
|
||||
* loaded on each request.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Menu callback: Duplicates an image style and redirects to the image styles
|
||||
* overview page.
|
||||
*
|
||||
* @param array $style
|
||||
* An image style array.
|
||||
*
|
||||
* @see image_style_name_validate()
|
||||
*/
|
||||
function image_styles_admin_duplicate_page_callback($style) {
|
||||
$duplicate_style = image_styles_admin_duplicate($style);
|
||||
drupal_set_message(t('Style %name has been duplicated to %new_name.', array('%name' => $style['name'], '%new_name' => $duplicate_style['name'])));
|
||||
drupal_goto('admin/config/media/image-styles');
|
||||
}
|
||||
|
||||
/**
|
||||
* Duplicates an image style and saves it.
|
||||
*
|
||||
* @param array $style
|
||||
* An image style array.
|
||||
* @param string|null $new_style_name
|
||||
* The preferred name for the new style. If left empty, the new name will be
|
||||
* based on the name of the style to duplicate. In both cases and when
|
||||
* necessary, the new name will be made unique by adding some suffix to it.
|
||||
*
|
||||
* @return array
|
||||
* An image style array with the newly created copy of the given style.
|
||||
*
|
||||
* @see image_style_name_validate()
|
||||
*/
|
||||
function image_styles_admin_duplicate($style, $new_style_name = NULL) {
|
||||
// Find a unique name for copy.
|
||||
// Step 1: Find the base: name without things like '-copy' or '-copy-1'
|
||||
$style_name_base = empty($new_style_name) ? $style['name'] : $new_style_name;
|
||||
if (preg_match('/-copy(-\d+)?$/', $style_name_base)) {
|
||||
$style_name_base = substr($style_name_base, 0, strpos($style_name_base, '-copy'));
|
||||
}
|
||||
|
||||
// Step 2: Add -copy to it (if the name comes from the current style).
|
||||
if (empty($new_style_name)) {
|
||||
$style_name_base .= '-copy';
|
||||
}
|
||||
|
||||
// Step 3: Ensure the new name will be unique.
|
||||
$i = 0;
|
||||
$style_name = $style_name_base;
|
||||
$styles = image_styles();
|
||||
while (isset($styles[$style_name])) {
|
||||
$i++;
|
||||
$style_name = $style_name_base . '-' . $i;
|
||||
}
|
||||
$style['name'] = $style_name;
|
||||
|
||||
// Unset isid to save it as a new style.
|
||||
unset($style['isid']);
|
||||
$style = image_style_save($style);
|
||||
|
||||
// Save copies of each effect with the new image style ID (isid).
|
||||
foreach ($style['effects'] as &$effect) {
|
||||
// Unset ieid to save it as a new effect.
|
||||
unset($effect['ieid']);
|
||||
$effect['isid'] = $style['isid'];
|
||||
$effect = image_effect_save($effect);
|
||||
}
|
||||
|
||||
return $style;
|
||||
}
|
||||
|
||||
/**
|
||||
* drupal_get_form callback: form to export an image style.
|
||||
*
|
||||
* @param array $style
|
||||
* An image style array.
|
||||
*/
|
||||
function image_styles_admin_export_form($form, $form_state, $style) {
|
||||
drupal_set_title(format_string('%page_name @style_name', array('%page_name' => t('Export image style'), '@style_name' => $style['name'])), PASS_THROUGH);
|
||||
$form['serialized_style'] = array(
|
||||
'#type' => 'textarea',
|
||||
'#rows' => 5,
|
||||
'#title' => t('Image style export data'),
|
||||
'#default_value' => serialize($style),
|
||||
'#attributes' => array('readonly' =>'readonly'),
|
||||
'#description' => t('Copy the contents of this field to the clipboard and, on another site, paste it in the textarea of an %page_title page.', array('%page_title' => t('Import image style'))),
|
||||
);
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* drupal_get_form callback: form to import an image style.
|
||||
*
|
||||
* @param array $style
|
||||
* An image style array.
|
||||
*/
|
||||
function image_styles_admin_import_form($form, $form_state) {
|
||||
$form['serialized_style'] = array(
|
||||
'#type' => 'textarea',
|
||||
'#rows' => 5,
|
||||
'#title' => t('Image style import data'),
|
||||
'#default_value' => '',
|
||||
'#required' => TRUE,
|
||||
'#description' => t('Paste the contents of the textarea of an %page_title page into this field.', array('%page_title' => t('Export image style'))),
|
||||
);
|
||||
|
||||
$form['actions'] = array('#type' => 'actions');
|
||||
$form['actions']['submit'] = array(
|
||||
'#type' => 'submit',
|
||||
'#value' => t('Import'),
|
||||
);
|
||||
|
||||
return $form;
|
||||
}
|
||||
|
||||
/**
|
||||
* Callback to validate the import style form.
|
||||
*/
|
||||
function image_styles_admin_import_form_validate($form, &$form_state) {
|
||||
if (image_styles_admin_import_extract_style($form_state['values']['serialized_style']) === FALSE) {
|
||||
form_set_error('serialized_style', t('The %field cannot be imported as an image style.', array('%field' => t('Image style import data'))));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Callback to process form submission of the import style form.
|
||||
*/
|
||||
function image_styles_admin_import_form_submit($form, &$form_state) {
|
||||
$style = image_styles_admin_import_extract_style($form_state['values']['serialized_style']);
|
||||
// Import the style by "duplicating" it, but prevent adding the -copy suffix
|
||||
// by passing the requested name as 2nd parameter.
|
||||
$new_style = image_styles_admin_duplicate($style, $style['name']);
|
||||
if ($new_style['name'] === $style['name']) {
|
||||
drupal_set_message(t('Style %name has been imported.', array('%name' => $style['name'])));
|
||||
}
|
||||
else {
|
||||
drupal_set_message(t('Style %name has been imported as %new_name.', array('%name' => $style['name'], '%new_name' => $new_style['name'])));
|
||||
}
|
||||
drupal_goto('admin/config/media/image-styles');
|
||||
}
|
||||
|
||||
/**
|
||||
* Unserializes and validates a string into image style data.
|
||||
*
|
||||
* @param string $import
|
||||
* The string representation of a @see serialize()'d image style array.
|
||||
*
|
||||
* @return array|false
|
||||
* An image style array or false if the string could not be unserialized into
|
||||
* image style data.
|
||||
*/
|
||||
function image_styles_admin_import_extract_style($import) {
|
||||
$style = @unserialize($import);
|
||||
|
||||
// Check if the contents of the textarea could be unserialized into an array.
|
||||
if (!is_array($style)) {
|
||||
return FALSE;
|
||||
}
|
||||
// Check if the required keys are available, we will ignore the other.
|
||||
$style = array_intersect_key($style, array('name' => 0, 'effects' => 0));
|
||||
if (count($style) !== 2) {
|
||||
return FALSE;
|
||||
}
|
||||
// 'name' must be "machine name" string
|
||||
if (!is_string($style['name']) || preg_match('/[0-9a-z_\-]+/', $style['name']) !== 1) {
|
||||
return FALSE;
|
||||
}
|
||||
// 'effects' must be an array
|
||||
if (!is_array($style['effects'])) {
|
||||
return FALSE;
|
||||
}
|
||||
// Check effects elements
|
||||
foreach ($style['effects'] as &$effect) {
|
||||
// an effect must be an array.
|
||||
if (!is_array($effect)) {
|
||||
return FALSE;
|
||||
}
|
||||
// Check if the required keys are available, we will ignore the other.
|
||||
$effect = array_intersect_key($effect, array('weight' => 0, 'name' => 0, 'data' => 0));
|
||||
if (count($effect) !== 3) {
|
||||
return FALSE;
|
||||
}
|
||||
// effect weight must be an integer (data type in table is int, not float).
|
||||
if (!is_int($effect['weight']) && $effect['weight'] !== (string) (int) $effect['weight']) {
|
||||
return FALSE;
|
||||
}
|
||||
// effect name must be a string
|
||||
if (!is_string($effect['name'])) {
|
||||
return FALSE;
|
||||
}
|
||||
// Check whether the effect data is an array.
|
||||
if (!is_array($effect['data'])) {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
// @todo: are there any security implications for creating styles like this?
|
||||
// - Unserialize() is save in itself: it only creates data (except possibly
|
||||
// for__wakeup(), but that can only be in already existing code: safe
|
||||
// - Not expected array entries are removed (array_intersect_key): safe
|
||||
// - Basic types are checked: safe
|
||||
// - Effect data array is not checked. Possibly unsafe?! The effect data array
|
||||
// contains the effect parameters. Normally these are entered and validated
|
||||
// via a form and subsequently saved in the database (serialized as here).
|
||||
// The form validation is not executed on import and thus the data may
|
||||
// contain invalid values. This is acceptable as it can also be done by
|
||||
// operating directly on the database. In Drupal this is not normally
|
||||
// checked for during processing: error messages will make clear that the
|
||||
// data has been played with. Can incorrect dat be abused? It may contain:
|
||||
// - incorrect types: we do not know the data structure of the various
|
||||
// effects, so we cannot check that and have to accept it as it comes.
|
||||
// Effects should check_plain in summary theme and convert to int/float
|
||||
// whenever possible befoire using it in commands.
|
||||
// - PHP code, but that may be valid content for the text or custom effects:
|
||||
// Effects should check_plain in summary theme and convert to int/float
|
||||
// whenever possible befoire using it in commands.
|
||||
// @todo: if the style contains an effect that contains PHP code, the user
|
||||
// should need the 'use PHP for settings' permission.
|
||||
// - HTML and or JS code: when used as parameter, this normally won't hurt.
|
||||
// When showing on the screen (summary theme), proper escaping should
|
||||
// suffice and is needed anyway: responsibility of effect.
|
||||
return $style;
|
||||
}
|
@@ -0,0 +1,13 @@
|
||||
name = Image styles admin
|
||||
description = Provides additional administrative image style functionality.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = image
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,76 @@
|
||||
<?php
|
||||
/**
|
||||
* @file Hook and callback implementations that must be available at all times.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_menu().
|
||||
*/
|
||||
function image_styles_admin_menu() {
|
||||
$items = array();
|
||||
$items['admin/config/media/image-styles/duplicate/%image_style'] = array(
|
||||
'title' => 'Duplicate style',
|
||||
'description' => 'Make a copy of an image style.',
|
||||
'page callback' => 'image_styles_admin_duplicate_page_callback',
|
||||
'page arguments' => array(5),
|
||||
'access arguments' => array('administer image styles'),
|
||||
'file' => 'image_styles_admin.inc',
|
||||
);
|
||||
$items['admin/config/media/image-styles/export/%image_style'] = array(
|
||||
'title' => 'Export style',
|
||||
'description' => 'Export an image style.',
|
||||
'page callback' => 'drupal_get_form',
|
||||
'page arguments' => array('image_styles_admin_export_form', 5),
|
||||
'access arguments' => array('administer image styles'),
|
||||
'file' => 'image_styles_admin.inc',
|
||||
);
|
||||
$items['admin/config/media/image-styles/import'] = array(
|
||||
'title' => 'Import style',
|
||||
'description' => 'Import an image style.',
|
||||
'page callback' => 'drupal_get_form',
|
||||
'page arguments' => array('image_styles_admin_import_form'),
|
||||
'access arguments' => array('administer image styles'),
|
||||
'type' => MENU_LOCAL_ACTION,
|
||||
'weight' => 3,
|
||||
'file' => 'image_styles_admin.inc',
|
||||
);
|
||||
return $items;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_preprocess_HOOK for theme image_style_list.
|
||||
*/
|
||||
function image_styles_admin_preprocess_image_style_list(&$variables) {
|
||||
// Tell imagecache_actions_preprocess_image_style_list to preprocess the next
|
||||
// call to theme_table()
|
||||
$flag = TRUE;
|
||||
image_styles_admin_preprocess_table($flag);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implements hook_preprocess_HOOK for theme table.
|
||||
*/
|
||||
function image_styles_admin_preprocess_table(&$variables) {
|
||||
static $is_in_image_style_list = FALSE;
|
||||
|
||||
if (is_bool($variables)) {
|
||||
// Called from imagecache_actions_style_duplicate(): set flag
|
||||
$is_in_image_style_list = $variables;
|
||||
}
|
||||
else if ($is_in_image_style_list) {
|
||||
// Normal preprocess hook call: only process if theme('table', ...) has been
|
||||
// called by theme_image_style_list()
|
||||
$variables['header'][2]['colspan'] = 4;
|
||||
foreach ($variables['rows'] as &$row) {
|
||||
array_splice($row, 2, 0, array($row[2], $row[2]));
|
||||
// Replace edit with duplicate in text and href
|
||||
$row[3] = str_replace('>' . t('edit') . '<', '>' . t('duplicate') . '<', $row[3]);
|
||||
$row[3] = preg_replace('/\/edit\/([-a-z0-9_]+)"/', '/duplicate/\1"', $row[3]);
|
||||
// Replace edit with export in text and href
|
||||
$row[4] = str_replace('>' . t('edit') . '<', '>' . t('export') . '<', $row[4]);
|
||||
$row[4] = preg_replace('/\/edit\/([-a-z0-9_]+)"/', '/export/\1"', $row[4]);
|
||||
}
|
||||
// Don't preprocess subsequent calls to theme_table().
|
||||
$is_in_image_style_list = FALSE;
|
||||
}
|
||||
}
|
@@ -0,0 +1,9 @@
|
||||
<?php
|
||||
/**
|
||||
* Warn about the soft dependency on system steam wrapper module
|
||||
*/
|
||||
function imagecache_actions_update_7001(&$sandbox) {
|
||||
//$t = get_t();
|
||||
drupal_set_message(t("Imagecache Actions: If you use the module:// notation anywhere in an image effect, you must now install the !module module.",
|
||||
array('!module' => l('System Stream Wrapper', 'http://drupal.org/project/system_stream_wrapper', array('external' => TRUE)))), 'warning');
|
||||
}
|
16
sites/all/modules/imagecache_actions/imagecache_actions.info
Normal file
@@ -0,0 +1,16 @@
|
||||
name = Imagecache Actions
|
||||
description = Provides a number of additional image effects.
|
||||
package = Media
|
||||
core = 7.x
|
||||
|
||||
dependencies[] = image
|
||||
|
||||
; files with classes
|
||||
files[] = ImageCacheActionsModuleStreamWrapper.php
|
||||
|
||||
; Information added by drupal.org packaging script on 2012-12-04
|
||||
version = "7.x-1.1"
|
||||
core = "7.x"
|
||||
project = "imagecache_actions"
|
||||
datestamp = "1354653754"
|
||||
|
@@ -0,0 +1,40 @@
|
||||
/*
|
||||
* UI enhancements for the imagecache edit form
|
||||
*/
|
||||
(function($){
|
||||
|
||||
/**
|
||||
* Check if independent corners are enabled and disable other fields in the UI
|
||||
*/
|
||||
Drupal.canvasactions_roundedcorners_form_disable_fields = function () {
|
||||
// To get the right effect, we have to set the 'disabled' attribute on the
|
||||
// field, but set the class on the container item. Tedious.
|
||||
if (!$(":checkbox#edit-data-independent-corners-set-independent-corners").attr("checked")){
|
||||
$(".form-item-data-radius").removeClass("form-disabled");
|
||||
$(".form-item-data-radius input").attr("disabled", false);
|
||||
$("#independent-corners-set .form-item").addClass("form-disabled");
|
||||
$("#independent-corners-set input").attr("disabled", true);
|
||||
}
|
||||
else {
|
||||
$(".form-item-data-radius").addClass("form-disabled");
|
||||
$(".form-item-data-radius input").attr("disabled", true);
|
||||
$("#independent-corners-set .form-item").removeClass("form-disabled");
|
||||
$("#independent-corners-set input").attr("disabled", false);
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Trigger the update when the form is ready, and add listener to the checkbox
|
||||
*/
|
||||
Drupal.behaviors.canvasactions_roundedcorners = {
|
||||
attach: function (context, settings) {
|
||||
Drupal.canvasactions_roundedcorners_form_disable_fields();
|
||||
$(":checkbox#edit-data-independent-corners-set-independent-corners").change(
|
||||
function() {
|
||||
Drupal.canvasactions_roundedcorners_form_disable_fields();
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
})(jQuery);
|
||||
|
@@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file Home for the most basic imagecache_action routines.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Implements hook_theme().
|
||||
*/
|
||||
function imagecache_actions_theme() {
|
||||
return array(
|
||||
'imagecacheactions_rgb_form' => array(
|
||||
'file' => 'utility-color.inc',
|
||||
'render element' => 'form',
|
||||
),
|
||||
'imagecacheactions_rgb' => array(
|
||||
'file' => 'utility-color.inc',
|
||||
'variables' => array('rgb' => NULL),
|
||||
),
|
||||
);
|
||||
}
|
260
sites/all/modules/imagecache_actions/imagefilter.inc
Normal file
@@ -0,0 +1,260 @@
|
||||
<?php
|
||||
|
||||
//include this file whenever you have to use imageconvolution…
|
||||
//you can use in your project, but keep the comment below
|
||||
//great for any image manipulation library
|
||||
//Made by Chao Xu(Mgccl) 3/1/07
|
||||
//www.webdevlogs.com
|
||||
//V 1.0
|
||||
if (!function_exists('imagefilter')) {
|
||||
function imagefilter($source, $var, $arg1 = NULL, $arg2 = NULL, $arg3 = NULL) {
|
||||
#define('IMAGE_FILTER_NEGATE', 0);
|
||||
#define('IMAGE_FILTER_GRAYSCALE', 1);
|
||||
#define('IMAGE_FILTER_BRIGHTNESS', 2);
|
||||
#define('IMAGE_FILTER_CONTRAST', 3);
|
||||
#define('IMAGE_FILTER_COLORIZE', 4);
|
||||
#define('IMAGE_FILTER_EDGEDETECT', 5);
|
||||
#define('IMAGE_FILTER_EMBOSS', 6);
|
||||
#define('IMAGE_FILTER_GAUSSIAN_BLUR', 7);
|
||||
#define('IMAGE_FILTER_SELECTIVE_BLUR', 8);
|
||||
#define('IMAGE_FILTER_MEAN_REMOVAL', 9);
|
||||
#define('IMAGE_FILTER_SMOOTH', 10);
|
||||
$max_y = imagesy($source);
|
||||
$max_x = imagesx($source);
|
||||
switch ($var) {
|
||||
case 0:
|
||||
$y = 0;
|
||||
while ($y < $max_y) {
|
||||
$x = 0;
|
||||
while ($x < $max_x) {
|
||||
$rgb = imagecolorat($source, $x, $y);
|
||||
$r = 255 - (($rgb >> 16) & 0xFF);
|
||||
$g = 255 - (($rgb >> 8) & 0xFF);
|
||||
$b = 255 - ($rgb & 0xFF);
|
||||
$a = $rgb >> 24;
|
||||
$new_pxl = imagecolorallocatealpha($source, $r, $g, $b, $a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, $r, $g, $b, $a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
++$x;
|
||||
}
|
||||
++$y;
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 1:
|
||||
$y = 0;
|
||||
while ($y < $max_y) {
|
||||
$x = 0;
|
||||
while ($x < $max_x) {
|
||||
$rgb = imagecolorat($source, $x, $y);
|
||||
$a = $rgb >> 24;
|
||||
$r = ((($rgb >> 16) & 0xFF) * 0.299) + ((($rgb >> 8) & 0xFF) * 0.587) + (($rgb & 0xFF) * 0.114);
|
||||
$new_pxl = imagecolorallocatealpha($source, $r, $r, $r, $a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, $r, $r, $r, $a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
++$x;
|
||||
}
|
||||
++$y;
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 2:
|
||||
$y = 0;
|
||||
while ($y < $max_y) {
|
||||
$x = 0;
|
||||
while ($x < $max_x) {
|
||||
$rgb = imagecolorat($source, $x, $y);
|
||||
$r = (($rgb >> 16) & 0xFF) + $arg1;
|
||||
$g = (($rgb >> 8) & 0xFF) + $arg1;
|
||||
$b = ($rgb & 0xFF) + $arg1;
|
||||
$a = $rgb >> 24;
|
||||
$r = ($r > 255) ? 255 : (($r < 0) ? 0 : $r);
|
||||
$g = ($g > 255) ? 255 : (($g < 0) ? 0 : $g);
|
||||
$b = ($b > 255) ? 255 : (($b < 0) ? 0 : $b);
|
||||
$new_pxl = imagecolorallocatealpha($source, $r, $g, $b, $a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, $r, $g, $b, $a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
++$x;
|
||||
}
|
||||
++$y;
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 3:
|
||||
$contrast = pow((100 - $arg1) / 100, 2);
|
||||
$y = 0;
|
||||
while ($y < $max_y) {
|
||||
$x = 0;
|
||||
while ($x < $max_x) {
|
||||
$rgb = imagecolorat($source, $x, $y);
|
||||
$a = $rgb >> 24;
|
||||
$r = (((((($rgb >> 16) & 0xFF) / 255) - 0.5) * $contrast) + 0.5) * 255;
|
||||
$g = (((((($rgb >> 8) & 0xFF) / 255) - 0.5) * $contrast) + 0.5) * 255;
|
||||
$b = ((((($rgb & 0xFF) / 255) - 0.5) * $contrast) + 0.5) * 255;
|
||||
$r = ($r > 255) ? 255 : (($r < 0) ? 0 : $r);
|
||||
$g = ($g > 255) ? 255 : (($g < 0) ? 0 : $g);
|
||||
$b = ($b > 255) ? 255 : (($b < 0) ? 0 : $b);
|
||||
$new_pxl = imagecolorallocatealpha($source, $r, $g, $b, $a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, $r, $g, $b, $a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
++$x;
|
||||
}
|
||||
++$y;
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 4:
|
||||
$x = 0;
|
||||
while ($x < $max_x) {
|
||||
$y = 0;
|
||||
while ($y < $max_y) {
|
||||
$rgb = imagecolorat($source, $x, $y);
|
||||
$r = (($rgb >> 16) & 0xFF) + $arg1;
|
||||
$g = (($rgb >> 8) & 0xFF) + $arg2;
|
||||
$b = ($rgb & 0xFF) + $arg3;
|
||||
$a = $rgb >> 24;
|
||||
$r = ($r > 255) ? 255 : (($r < 0) ? 0 : $r);
|
||||
$g = ($g > 255) ? 255 : (($g < 0) ? 0 : $g);
|
||||
$b = ($b > 255) ? 255 : (($b < 0) ? 0 : $b);
|
||||
$new_pxl = imagecolorallocatealpha($source, $r, $g, $b, $a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, $r, $g, $b, $a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
++$y;
|
||||
}
|
||||
++$x;
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 5:
|
||||
return imageconvolution($source, array(
|
||||
array(-1, 0, -1),
|
||||
array(0, 4, 0),
|
||||
array(-1, 0, -1)
|
||||
), 1, 127);
|
||||
break;
|
||||
case 6:
|
||||
return imageconvolution($source, array(
|
||||
array(1.5, 0, 0),
|
||||
array(0, 0, 0),
|
||||
array(0, 0, -1.5)
|
||||
), 1, 127);
|
||||
break;
|
||||
case 7:
|
||||
return imageconvolution($source, array(
|
||||
array(1, 2, 1),
|
||||
array(2, 4, 2),
|
||||
array(1, 2, 1)
|
||||
), 16, 0);
|
||||
break;
|
||||
case 8:
|
||||
for ($y = 0; $y < $max_y; $y++) {
|
||||
for ($x = 0; $x < $max_x; $x++) {
|
||||
$flt_r_sum = $flt_g_sum = $flt_b_sum = 0;
|
||||
$cpxl = imagecolorat($source, $x, $y);
|
||||
for ($j = 0; $j < 3; $j++) {
|
||||
for ($i = 0; $i < 3; $i++) {
|
||||
if (($j == 1) && ($i == 1)) {
|
||||
$flt_r[1][1] = $flt_g[1][1] = $flt_b[1][1] = 0.5;
|
||||
}
|
||||
else {
|
||||
$pxl = imagecolorat($source, $x - (3 >> 1) + $i, $y - (3 >> 1) + $j);
|
||||
|
||||
$new_a = $pxl >> 24;
|
||||
//$r = (($pxl >> 16) & 0xFF);
|
||||
//$g = (($pxl >> 8) & 0xFF);
|
||||
//$b = ($pxl & 0xFF);
|
||||
$new_r = abs((($cpxl >> 16) & 0xFF) - (($pxl >> 16) & 0xFF));
|
||||
if ($new_r != 0) {
|
||||
$flt_r[$j][$i] = 1 / $new_r;
|
||||
}
|
||||
else {
|
||||
$flt_r[$j][$i] = 1;
|
||||
}
|
||||
|
||||
$new_g = abs((($cpxl >> 8) & 0xFF) - (($pxl >> 8) & 0xFF));
|
||||
if ($new_g != 0) {
|
||||
$flt_g[$j][$i] = 1 / $new_g;
|
||||
}
|
||||
else {
|
||||
$flt_g[$j][$i] = 1;
|
||||
}
|
||||
|
||||
$new_b = abs(($cpxl & 0xFF) - ($pxl & 0xFF));
|
||||
if ($new_b != 0) {
|
||||
$flt_b[$j][$i] = 1 / $new_b;
|
||||
}
|
||||
else {
|
||||
$flt_b[$j][$i] = 1;
|
||||
}
|
||||
}
|
||||
|
||||
$flt_r_sum += $flt_r[$j][$i];
|
||||
$flt_g_sum += $flt_g[$j][$i];
|
||||
$flt_b_sum += $flt_b[$j][$i];
|
||||
}
|
||||
}
|
||||
|
||||
for ($j = 0; $j < 3; $j++) {
|
||||
for ($i = 0; $i < 3; $i++) {
|
||||
if ($flt_r_sum != 0) {
|
||||
$flt_r[$j][$i] /= $flt_r_sum;
|
||||
}
|
||||
if ($flt_g_sum != 0) {
|
||||
$flt_g[$j][$i] /= $flt_g_sum;
|
||||
}
|
||||
if ($flt_b_sum != 0) {
|
||||
$flt_b[$j][$i] /= $flt_b_sum;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$new_r = $new_g = $new_b = 0;
|
||||
|
||||
for ($j = 0; $j < 3; $j++) {
|
||||
for ($i = 0; $i < 3; $i++) {
|
||||
$pxl = imagecolorat($source, $x - (3 >> 1) + $i, $y - (3 >> 1) + $j);
|
||||
$new_r += (($pxl >> 16) & 0xFF) * $flt_r[$j][$i];
|
||||
$new_g += (($pxl >> 8) & 0xFF) * $flt_g[$j][$i];
|
||||
$new_b += ($pxl & 0xFF) * $flt_b[$j][$i];
|
||||
}
|
||||
}
|
||||
|
||||
$new_r = ($new_r > 255) ? 255 : (($new_r < 0) ? 0 : $new_r);
|
||||
$new_g = ($new_g > 255) ? 255 : (($new_g < 0) ? 0 : $new_g);
|
||||
$new_b = ($new_b > 255) ? 255 : (($new_b < 0) ? 0 : $new_b);
|
||||
$new_pxl = imagecolorallocatealpha($source, (int) $new_r, (int) $new_g, (int) $new_b, $new_a);
|
||||
if ($new_pxl == FALSE) {
|
||||
$new_pxl = imagecolorclosestalpha($source, (int) $new_r, (int) $new_g, (int) $new_b, $new_a);
|
||||
}
|
||||
imagesetpixel($source, $x, $y, $new_pxl);
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
break;
|
||||
case 9:
|
||||
return imageconvolution($source, array(
|
||||
array(-1, -1, -1),
|
||||
array(-1, 9, -1),
|
||||
array(-1, -1, -1)
|
||||
), 1, 0);
|
||||
break;
|
||||
case 10:
|
||||
return imageconvolution($source, array(
|
||||
array(1, 1, 1),
|
||||
array(1, $arg1, 1),
|
||||
array(1, 1, 1)
|
||||
), $arg1 + 8, 0);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
BIN
sites/all/modules/imagecache_actions/tests/150x300.jpg
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
sites/all/modules/imagecache_actions/tests/200x300.jpg
Normal file
After Width: | Height: | Size: 19 KiB |
BIN
sites/all/modules/imagecache_actions/tests/250x250.jpg
Normal file
After Width: | Height: | Size: 20 KiB |
BIN
sites/all/modules/imagecache_actions/tests/250x300.jpg
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
sites/all/modules/imagecache_actions/tests/300x150.jpg
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
sites/all/modules/imagecache_actions/tests/300x200.jpg
Normal file
After Width: | Height: | Size: 19 KiB |
BIN
sites/all/modules/imagecache_actions/tests/300x250.jpg
Normal file
After Width: | Height: | Size: 23 KiB |
9
sites/all/modules/imagecache_actions/tests/README.txt
Normal file
@@ -0,0 +1,9 @@
|
||||
This module provides an overview of all the imagecache presets in use on this
|
||||
site, as well as a number of samples used to test if everything is working as
|
||||
expected.
|
||||
|
||||
Sample images are provided to illustrate a number of the sample presets.
|
||||
Each image shown should match the one next to it.
|
||||
Where they don't match illustrates a current weakness in the system or the code
|
||||
coverage.
|
||||
|
BIN
sites/all/modules/imagecache_actions/tests/background-2.jpg
Normal file
After Width: | Height: | Size: 20 KiB |
BIN
sites/all/modules/imagecache_actions/tests/black-ribbon.gif
Normal file
After Width: | Height: | Size: 16 KiB |
BIN
sites/all/modules/imagecache_actions/tests/corner-bl.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
sites/all/modules/imagecache_actions/tests/corner-br.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
sites/all/modules/imagecache_actions/tests/corner-tl.png
Normal file
After Width: | Height: | Size: 2.1 KiB |
BIN
sites/all/modules/imagecache_actions/tests/corner-tr.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
@@ -0,0 +1,140 @@
|
||||
<?php
|
||||
// $ID: $
|
||||
/**
|
||||
* @file
|
||||
* Test imagecache preset.
|
||||
*
|
||||
* Created on Dec 29, 2009
|
||||
*
|
||||
* @author 'dman' Dan Morrison http://coders.co.nz/
|
||||
*/
|
||||
|
||||
$presets['corners_combo'] = array (
|
||||
'name' => 'corners_combo',
|
||||
'#weight' => 9,
|
||||
'effects' => array (
|
||||
0 => array (
|
||||
'weight' => '-10',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_definecanvas',
|
||||
'data' => array (
|
||||
'RGB' => array (
|
||||
'HEX' => 'e5e8a1',
|
||||
),
|
||||
'under' => 1,
|
||||
'exact' => array (
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' => array (
|
||||
'leftdiff' => '4',
|
||||
'rightdiff' => '4',
|
||||
'topdiff' => '4',
|
||||
'bottomdiff' => '4',
|
||||
),
|
||||
),
|
||||
),
|
||||
1 => array (
|
||||
'weight' => '-9',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_definecanvas',
|
||||
'data' => array (
|
||||
'RGB' => array (
|
||||
'HEX' => '',
|
||||
),
|
||||
'under' => 1,
|
||||
'exact' => array (
|
||||
'width' => '',
|
||||
'height' => '',
|
||||
'xpos' => 'center',
|
||||
'ypos' => 'center',
|
||||
),
|
||||
'relative' => array (
|
||||
'leftdiff' => '4',
|
||||
'rightdiff' => '4',
|
||||
'topdiff' => '4',
|
||||
'bottomdiff' => '4',
|
||||
),
|
||||
),
|
||||
),
|
||||
2 => array (
|
||||
'weight' => '-8',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'left',
|
||||
'ypos' => 'top',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/corner-tl.png",
|
||||
),
|
||||
),
|
||||
3 => array (
|
||||
'weight' => '-7',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'right',
|
||||
'ypos' => 'top',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/corner-tr.png",
|
||||
),
|
||||
),
|
||||
4 =>
|
||||
array (
|
||||
'weight' => '-6',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'left',
|
||||
'ypos' => 'bottom',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/corner-bl.png",
|
||||
),
|
||||
),
|
||||
5 => array (
|
||||
'weight' => '-5',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_file2canvas',
|
||||
'data' => array (
|
||||
'xpos' => 'right',
|
||||
'ypos' => 'bottom',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/corner-br.png",
|
||||
),
|
||||
),
|
||||
6 => array (
|
||||
'weight' => '-4',
|
||||
'module' => 'image',
|
||||
'name' => 'image_rotate',
|
||||
'data' => array (
|
||||
'degrees' => '7',
|
||||
'random' => 0,
|
||||
'bgcolor' => '',
|
||||
),
|
||||
),
|
||||
7 => array (
|
||||
'weight' => '-3',
|
||||
'module' => 'image',
|
||||
'name' => 'image_scale',
|
||||
'data' => array (
|
||||
'width' => '300',
|
||||
'height' => '',
|
||||
'upscale' => 0,
|
||||
),
|
||||
),
|
||||
8 => array (
|
||||
'weight' => '-2',
|
||||
'module' => 'imagecache_canvasactions',
|
||||
'name' => 'canvasactions_canvas2file',
|
||||
'data' => array (
|
||||
'xpos' => '',
|
||||
'ypos' => '',
|
||||
'alpha' => '100',
|
||||
'path' => "$filepath/background-2.jpg",
|
||||
'dimensions' => 'original',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|