PDFlib
KAUFEN
PDFlib

text_output/wrap_text_around_images

Place images within a Textflow. Place an image at a fixed position on the page and use a matchbox to wrap the text around the image. To place some images left-aligned or right-aligned at defined positions  within the Textflow use return values when fitting the Textflow. To place some small icons at the beginning of some text lines in a Textflow use the inline options "matchbox" and "matchbox end" within the Textflow as well as the info_matchbox function to retrieve the matchbox instances and dimensions.

Download Java Code      Show Output PDF

/* $Id: wrap_text_around_images.java,v 1.7 2007/10/30 16:16:33 katja Exp $

 * Wrap text around images:

 * Place images within a Textflow

 *

 * Place an image at a fixed position on the page and use a matchbox to

 * wrap the text around the image.

 * To place some images left-aligned or right-aligned at defined positions

 * within the Textflow use return values when fitting the Textflow.

 * To place some small icons at the beginning of some text lines in a Textflow

 * use the inline options "matchbox" and "matchbox end" within the Textflow as

 * well as the info_matchbox function to retrieve the matchbox instances and

 * dimensions.

 *

 * Required software: PDFlib/PDFlib+PDI/PPS 7

 * Required data: image file

 */

package com.pdflib.cookbook.pdflib.text_output;


import com.pdflib.pdflib;

import com.pdflib.PDFlibException;


public class wrap_text_around_images

{

    public static void main (String argv[])

    {

    pdflib p = null;

    String searchpath = "../input";

    String outfile = "wrap_text_around_images.pdf";

    String title = "Wrap Text around Images";

   

    int i, m, tf = -1, image, icon, numberOfMatchbox;

    String result;

    String optlist = "", imageoptlist = "";

    final double llx = 100, lly = 50, urx = 450, ury = 800;

    double x1, y1, width, height;

    double posx = 0, posy = 0;

    boolean ismatchbox;

    String[][] matchboxname = {

        {"new", "new.jpg"},

        {"arrow", "arrow.jpg"}

    };


    /* Repeat the dummy text to produce more contents */

    final int count = 7;


    final String optlist1 =

        "fontname=Helvetica fontsize=10.5 encoding=unicode " +

        "fillcolor={gray 0} alignment=justify";


    final String optlist2 =

        "fontname=Helvetica-Bold fontsize=14 encoding=unicode " +

        "fillcolor={rgb 0.6 0.6 0.8} charref";


    /* Text which is repeatedly placed on the page. Soft hyphens are marked

     * with the character reference "­" (character references are enabled

     * by the charref option).

     */

    final String text=

        "Our paper planes are the ideal way of passing the time. We offer " +

        "revolutionary new develop­ments of the traditional common paper " +

        "planes. If your lesson, conference, or lecture turn out to be " +

        "deadly boring, you can have a wonderful time with our planes. " +

        "All our models are fol­ded from one paper sheet. They are " +

        "exclu­sively folded with­out using any adhesive. Several " +

        "models are equipped with a folded landing gear enabling a safe " +

        "landing on the intended loca­tion provided that you have aimed " +

        "well. Other models are able to fly loops or cover long distances. " +

        "Let them start from a vista point in the mountains and see where " +

        "they touch the ground. ";

   

    /* Option list with some text options and the three macros "arrow", "new",

     * and "end" to be used as inline options in the "features" text below

     * to indicate where to leave some space for the respective images to

     * be placed and the text to wrap around it.

     */


    final String optlist3 =

        "macro {" +

        "new {matchbox={name=new boxwidth=15 boxheight {ascender descender} " +

        "    createwrapbox} leftindent=15 parindent=-15} " +

        "arrow {matchbox={name=arrow boxwidth=15 boxheight {ascender " +

        "    descender} createwrapbox} leftindent=15 parindent=-15} " +

        "end {matchbox={end}} } " +

        "fontname=Helvetica fontsize=10.5 encoding=unicode " +

        "fillcolor={gray 0} alignment=justify";

   

    /* Text containing the macros defined in the option list above */

    final String features =

        "Our paper planes are the ideal way of passing the time. We offer " +

        "revolutionary new developments of the traditional common paper " +

        "planes. If your lesson, conference, or lecture turn out to be " +

        "deadly boring, you can have a wonderful time with our planes. All " +

        "our models are folded from one paper sheet. They are exclusively " +

        "folded without using any adhesive. Several models are equipped " +

        "with a folded landing gear enabling a safe landing on the intended " +

        "location provided that you have aimed well. Other models are able " +

        "to fly loops or cover long distances. Let them start from a vista " +

        "point in the mountains and see where they touch the ground." +

        "<nextline><nextline>" +

        "Have a look at our new paper plane models!" +

        "<nextline><nextparagraph>" +

        "<&new><&end>Long Distance Glider <nextline>"+

        "With this paper rocket you can send all your messages " +

        "even when sitting in a hall or in the cinema pretty near the back. " +

        "<nextline><nextparagraph>" +

        "<&arrow><&end>Giant Wing<nextline>" +

        "An unbelievable sailplane! It is amazingly robust and can even do " +

        "aerobatics. But it best suited to gliding." +

        "<nextline><nextparagraph>" +

        "<&new><&end>Cone Head Rocket<nextline>" +

        "This paper arrow can be thrown with big swing. We launched it from " +

        "the roof of a hotel. It stayed in the air a long time and covered " +

        "a considerable distance. " +

        "<nextline><nextparagraph>" +

        "<&arrow><&end>Super Dart<nextline>" +

        "The super dart can fly giant loops with a radius of 4 or 5 meters " +

        "and cover very long distances. Its heavy cone point is slightly " +

        "bowed upwards to get the lift required for loops." +

        "<nextline><nextparagraph>" +

        "<&arrow><&end>German Bi-Plane<nextline>" +

        "Brand-new and ready for take-off. If you have lessons in the " +

        "history of aviation you can show your interest by letting it land " +

        "on your teacher's desk." +

        "<nextline leftindent=0><nextparagraph>" +

        "To fold the famous rocket looper proceed as follows:" +

        "<nextparagraph><nextline>" +

        "Take a A4 sheet." +

        "Fold it lengthwise in the middle." +

        "Then, fold the upper corners down. " +

        "Fold the long sides inwards " +

        "that the points A and B meet on the central fold." +

        "<nextparagraph><nextline>" +

        "Fold the points C and D that the upper " +

        "corners meet with the central fold as well. " +

        "Fold the plane in the middle. Fold the wings " +

        "down that they close with the lower border of the plane.";


    try {

        p = new pdflib();


        p.set_parameter("SearchPath", searchpath);


        /* This means we must check return values of load_font() etc. */

        p.set_parameter("errorpolicy", "return");

        p.set_parameter("charref", "true");


        if (p.begin_document(outfile, "") == -1)

        throw new Exception("Error: " + p.get_errmsg());


        p.set_info("Creator", "PDFlib Cookbook");

        p.set_info("Title", title + " $Revision: 1.7 $");

       

       

        /* --- Place an image at a fixed position on the page and use --- *

         * --- a matchbox to wrap the text around the image           --- *

         */


        /* Load the image. Assign a matchbox called "img"

         * to it to be used later when wrapping the text around it.

         */


        image = p.load_image("auto", "kraxi_logo_text.tif", "");

        if (image == -1)

            throw new Exception("Error: " + p.get_errmsg());


        /* Create some amount of dummy text and feed it to a Textflow

         * object with alternating options.

         */

        for (i=1; i<=count; i++)

        {

        String num = i + " ";


        tf = p.add_textflow(tf, num, optlist2);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());


        tf = p.add_textflow(tf, text, optlist1);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());

        }


        /* Loop until all of the text is placed; create new pages

         * as long as more text needs to be placed.

         */

        p.create_bookmark("Use the image matchbox and the wrap Textflow " +

            "option with matchboxes", "");

        do

        {

        p.begin_page_ext(0, 0, "width=a4.width height=a4.height");

     

        /* Place the image on a fixed position on the page. Assign a matchbox

         * called "img" to it to be used to indicate the image rectangle later

         * when wrapping the text around it.

         */

        p.fit_image(image, 200, 370, "boxsize={300 200} fitmethod=meet " +

            "position=center matchbox={name=img margin=-10}");


        /* Place the text while wrapping it around the matchbox called "img" */

        result = p.fit_textflow(tf, llx, lly, urx, ury,

            "verticalalign=justify linespreadlimit=120% " +

            "wrap={usematchboxes={{img}}}");


        p.end_page_ext("");


        /* "_boxfull" means we must continue because there is more text;

         * "_nextpage" is interpreted as "start new column"

         */

        } while (result.equals("_boxfull") || result.equals("_nextpage"));


        /* Check for errors */

        if (!result.equals("_stop"))

        {

            /* "_boxempty" happens if the box is very small and doesn't

             * hold any text at all.

             */

            if (result.equals( "_boxempty"))

                throw new Exception ("Error: Textflow box too small");

            else

            {

                /* Any other return value is a user exit caused by

                 * the "return" option; this requires dedicated code to

                 * deal with.

                 */

                throw new Exception ("User return '" + result +

                        "' found in Textflow");

            }

        }

        p.close_image(image);


        p.delete_textflow(tf);


        /* --- Place images at defined positions within a Textflow by --- *

         * --- using the "matchbox" and "matchbox end" inline options --- *

         * --- and return values when placing the Textflow for        --- *

         * --- for indicating the image positions                     --- *

         */


        /* Add some text to the Textflow. Then add two nextlines and

         * define the return value "imageleft". Later,

         * the Textflow portion defined above will be placed with

         * fit_textflow() and the "imageleft" value will be returned indicating

         * that now the left-aligned image should be placed.

         */

        tf = -1;

        tf = p.add_textflow(tf, text, optlist1);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());

       


        tf = p.add_textflow(tf, "", "nextline nextline return imageleft");

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());

       

        /* Add some more text to the Textflow. Similar to the "imageleft"

         * return value above define the "imageright" return value to

         * indicate that now the right-aligned image should be placed.

         */

        tf = p.add_textflow(tf, text, optlist1);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());

        tf = p.add_textflow(tf, "", "nextline nextline return imageright");

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());


        tf = p.add_textflow(tf, text, optlist1);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());


        image = p.load_image("auto", "kraxi_logo.tif", "");

        if (image == -1)

            throw new Exception("Error: " + p.get_errmsg());


        p.begin_page_ext(0, 0, "width=a4.width height=a4.height");

        p.create_bookmark("Use dedicated Textflow return values to indicate " +

            "image positions", "");


        posy = ury;

        ismatchbox=false;

       

        do

        {

                /* Fit the text and wrap it around the matchbox called "image"

                 * if it is defined yet

                 */

                if (ismatchbox) {

                        optlist = "verticalalign=justify linespreadlimit=120% " +

                "wrap={usematchboxes={{image}}} ";

                }

                else {

                        optlist = "verticalalign=justify linespreadlimit=120% ";

                }

               

                result = p.fit_textflow(tf, llx, lly, urx, posy, optlist);

           

            /* Retrieve the current text position */

            posy = p.info_textflow(tf, "textendy");


            if (result.equals("imageleft")){

                /* Textflow interrupted returning the keyword "imageleft".

                 * Place the image on the current left position of the Textflow

                 * fitbox.

                 */

                posx = llx;

                imageoptlist = "position {0 100} matchbox={name=image " +

                    "offsetright=10 offsettop=10 offsetbottom=-10}";

                /* Reduce the posy position by a value similar to the

                 * "offsettop" value defined above to create some distance

                 * from the previous text

                 */

                p.fit_image(image, posx, posy-10, imageoptlist);

                ismatchbox=true;

            }

            if (result.equals("imageright")){

                /* Textflow interrupted with the keyword "imageleft".

                 * Place the image to the current left position of the fitbox.

                 */

                posx = urx;

                imageoptlist = "position {100 100} matchbox={name=image " +

                    "offsetleft=-10 offsettop=10 offsetbottom=-10}";

                /* Reduce the posy position by a value similar to the

                 * "offsettop" value defined above to create some distance from

                 * the previous text

                 */

                p.fit_image(image, posx, posy-10, imageoptlist);

                ismatchbox=true;

            }

            /* Create a new page if the text cannot be fit completely into the

             * box

             */

            if (result.equals("_boxfull") || result.equals( "_boxempty")){

                p.end_page_ext("");

                p.begin_page_ext(0, 0, "width=a4.width height=a4.height");

                posy = ury;

            }

            /* Go ahead with the rest of the text, until the text has been

             * finished

             */

        } while (!result.equals("_stop"));


        p.delete_textflow(tf);


        p.end_page_ext("");

        p.close_image(image);

       

        /* --- Place some small icons at the beginning of some text lines --- *

         * --- in a Textflow using the inline options "matchbox" and      --- *

         * --- "matchbox end" within the Textflow.                        --- *

         */

        p.begin_page_ext(0, 0, "width=a4.width height=a4.height");

        p.create_bookmark("Use the inline options \"matchbox\" and " +

            "\"matchbox end\"", "");

       

        /* Create a Textflow containing inline options to define the matchboxes

         * "arrow" and "new" which indicates the positions for the arrow and the

         * new image to be placed.

         */       

        tf = p.create_textflow(features, optlist3);

        if (tf == -1)

            throw new Exception("Error: " + p.get_errmsg());


        do {

                /* Fit the Textflow */

            result = p.fit_textflow(tf, llx, lly, urx, ury,

                "verticalalign=justify linespreadlimit=120% ");

           

            /* Loop over all icons ("new" and "arrow" in our case) to be placed

             * in the respective matchboxes

             */

            for (m=0; m < (int) matchboxname.length; m++){

                icon = p.load_image("auto", matchboxname[m][1], "");

                if (icon == -1)

                    throw new Exception("Error: " + p.get_errmsg());


                /* Retrieve the number of instances of the matchbox */

                numberOfMatchbox =

                    (int) p.info_matchbox(matchboxname[m][0], 0, "count");

               

                /* Iterate over all matchbox instances and fill them with the

                 * icon

                 */

                for (i=1; i<= numberOfMatchbox; i++)

                {

                    x1 = p.info_matchbox(matchboxname[m][0], i, "x1");

                    y1 = p.info_matchbox(matchboxname[m][0], i, "y1");

                    width = p.info_matchbox(matchboxname[m][0], i, "width");

                    height = p.info_matchbox(matchboxname[m][0], i, "height");

                    p.fit_image(icon, x1, y1,

                        "boxsize {" + width + " " + height +

                        "} fitmethod meet");

                }

            }

            if (result.equals("_boxfull")){

                p.end_page_ext("");

                p.begin_page_ext(0, 0, "width=a4.width height=a4.height");

                posy = ury;

            }

        } while (!result.equals("_stop"));


        p.delete_textflow(tf);


        p.end_page_ext("");

        p.end_document("");


        } catch (PDFlibException e) {

            System.err.print("PDFlib exception occurred:\n");

            System.err.print("[" + e.get_errnum() + "] " + e.get_apiname() +

                ": " + e.get_errmsg() + "\n");

        } catch (Exception e) {

            System.err.println(e.getMessage());

        } finally {

            if (p != null) {

                p.delete();

            }

        }

    }

}