root/src/add-ons/translators/jpeg/JPEGTranslator.cpp
/*

Copyright (c) 2002-2003, Marcin 'Shard' Konicki
All rights reserved.

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

    * Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
    * Redistributions in binary form must reproduce the above copyright notice,
      this list of conditions and the following disclaimer in the documentation and/or
      other materials provided with the distribution.
    * Name "Marcin Konicki", "Shard" or any combination of them,
      must not be used to endorse or promote products derived from this
      software without specific prior written permission from Marcin Konicki.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

*/


#include "JPEGTranslator.h"

#include <syslog.h>

#include <Alignment.h>
#include <Catalog.h>
#include <LayoutBuilder.h>
#include <TabView.h>
#include <TextView.h>

#include "be_jerror.h"
#include "exif_parser.h"
#include "TranslatorWindow.h"


#undef B_TRANSLATION_CONTEXT
#define B_TRANSLATION_CONTEXT "JPEGTranslator"

#define MARKER_EXIF     0xe1

// Set these accordingly
#define JPEG_ACRONYM "JPEG"
#define JPEG_FORMAT 'JPEG'
#define JPEG_MIME_STRING "image/jpeg"
#define JPEG_DESCRIPTION "JPEG image"

// The translation kit's native file type
#define B_TRANSLATOR_BITMAP_MIME_STRING "image/x-be-bitmap"
#define B_TRANSLATOR_BITMAP_DESCRIPTION "Be Bitmap Format (JPEGTranslator)"


static const int32 sTranslatorVersion = B_TRANSLATION_MAKE_VERSION(1, 2, 0);

static const char* sTranslatorName = B_TRANSLATE("JPEG images");
static const char* sTranslatorInfo = B_TRANSLATE("©2002-2003, Marcin Konicki\n"
        "©2005-2007, Haiku\n"
        "\n"
        "Based on IJG library ©  1994-2009, Thomas G. Lane, Guido Vollbeding.\n"
        "\thttp://www.ijg.org/files/\n"
        "\n"
        "with \"lossless\" encoding support patch by Ken Murchison\n"
        "\thttp://www.oceana.com/ftp/ljpeg/\n"
        "\n"
        "With some colorspace conversion routines by Magnus Hellman\n"
        "\thttp://www.bebits.com/app/802\n");

// Define the formats we know how to read
static const translation_format sInputFormats[] = {
        { JPEG_FORMAT, B_TRANSLATOR_BITMAP, 0.5, 0.5,
                JPEG_MIME_STRING, JPEG_DESCRIPTION },
        { B_TRANSLATOR_BITMAP, B_TRANSLATOR_BITMAP, 0.5, 0.5,
                B_TRANSLATOR_BITMAP_MIME_STRING, B_TRANSLATOR_BITMAP_DESCRIPTION }
};

// Define the formats we know how to write
static const translation_format sOutputFormats[] = {
        { JPEG_FORMAT, B_TRANSLATOR_BITMAP, 0.5, 0.5,
                JPEG_MIME_STRING, JPEG_DESCRIPTION },
        { B_TRANSLATOR_BITMAP, B_TRANSLATOR_BITMAP, 0.5, 0.5,
                B_TRANSLATOR_BITMAP_MIME_STRING, B_TRANSLATOR_BITMAP_DESCRIPTION }
};


static const TranSetting sDefaultSettings[] = {
        {JPEG_SET_SMOOTHING, TRAN_SETTING_INT32, 0},
        {JPEG_SET_QUALITY, TRAN_SETTING_INT32, 95},
        {JPEG_SET_PROGRESSIVE, TRAN_SETTING_BOOL, true},
        {JPEG_SET_OPT_COLORS, TRAN_SETTING_BOOL, true},
        {JPEG_SET_SMALL_FILES, TRAN_SETTING_BOOL, false},
        {JPEG_SET_GRAY1_AS_RGB24, TRAN_SETTING_BOOL, false},
        {JPEG_SET_ALWAYS_RGB32, TRAN_SETTING_BOOL, true},
        {JPEG_SET_PHOTOSHOP_CMYK, TRAN_SETTING_BOOL, true},
        {JPEG_SET_SHOWREADWARNING, TRAN_SETTING_BOOL, true}
};

const uint32 kNumInputFormats = sizeof(sInputFormats) / sizeof(translation_format);
const uint32 kNumOutputFormats = sizeof(sOutputFormats) / sizeof(translation_format);
const uint32 kNumDefaultSettings = sizeof(sDefaultSettings) / sizeof(TranSetting);


namespace conversion {


static bool
x_flipped(int32 orientation)
{
        return orientation == 2 || orientation == 3
                || orientation == 6 || orientation == 7;
}


static bool
y_flipped(int32 orientation)
{
        return orientation == 3 || orientation == 4
                || orientation == 7 || orientation == 8;
}


static int32
dest_index(uint32 width, uint32 height, uint32 x, uint32 y, int32 orientation)
{
        if (orientation > 4) {
                uint32 temp = x;
                x = y;
                y = temp;
        }
        if (y_flipped(orientation))
                y = height - 1 - y;
        if (x_flipped(orientation))
                x = width - 1 - x;

        return y * width + x;
}


//      #pragma mark - conversion for compression


inline void
convert_from_gray1_to_gray8(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        while (index < inRowBytes) {
                unsigned char c = in[index++];
                for (int b = 128; b; b = b>>1) {
                        unsigned char color;
                        if (c & b)
                                color = 0;
                        else
                                color = 255;
                        out[index2++] = color;
                }
        }
}


inline void
convert_from_gray1_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        while (index < inRowBytes) {
                unsigned char c = in[index++];
                for (int b = 128; b; b = b>>1) {
                        unsigned char color;
                        if (c & b)
                                color = 0;
                        else
                                color = 255;
                        out[index2++] = color;
                        out[index2++] = color;
                        out[index2++] = color;
                }
        }
}


inline void
convert_from_cmap8_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        const color_map * map = system_colors();
        int32 index = 0;
        int32 index2 = 0;
        while (index < inRowBytes) {
                rgb_color color = map->color_list[in[index++]];

                out[index2++] = color.red;
                out[index2++] = color.green;
                out[index2++] = color.blue;
        }
}


inline void
convert_from_15_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        int16 in_pixel;
        while (index < inRowBytes) {
                in_pixel = in[index] | (in[index + 1] << 8);
                index += 2;

                out[index2++] = (((in_pixel & 0x7c00)) >> 7) | (((in_pixel & 0x7c00)) >> 12);
                out[index2++] = (((in_pixel & 0x3e0)) >> 2) | (((in_pixel & 0x3e0)) >> 7);
                out[index2++] = (((in_pixel & 0x1f)) << 3) | (((in_pixel & 0x1f)) >> 2);
        }
}


inline void
convert_from_15b_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        int16 in_pixel;
        while (index < inRowBytes) {
                in_pixel = in[index + 1] | (in[index] << 8);
                index += 2;

                out[index2++] = (((in_pixel & 0x7c00)) >> 7) | (((in_pixel & 0x7c00)) >> 12);
                out[index2++] = (((in_pixel & 0x3e0)) >> 2) | (((in_pixel & 0x3e0)) >> 7);
                out[index2++] = (((in_pixel & 0x1f)) << 3) | (((in_pixel & 0x1f)) >> 2);
        }
}


inline void
convert_from_16_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        int16 in_pixel;
        while (index < inRowBytes) {
                in_pixel = in[index] | (in[index + 1] << 8);
                index += 2;

                out[index2++] = (((in_pixel & 0xf800)) >> 8) | (((in_pixel & 0xf800)) >> 13);
                out[index2++] = (((in_pixel & 0x7e0)) >> 3) | (((in_pixel & 0x7e0)) >> 9);
                out[index2++] = (((in_pixel & 0x1f)) << 3) | (((in_pixel & 0x1f)) >> 2);
        }
}


inline void
convert_from_16b_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        int16 in_pixel;
        while (index < inRowBytes) {
                in_pixel = in[index + 1] | (in[index] << 8);
                index += 2;

                out[index2++] = (((in_pixel & 0xf800)) >> 8) | (((in_pixel & 0xf800)) >> 13);
                out[index2++] = (((in_pixel & 0x7e0)) >> 3) | (((in_pixel & 0x7e0)) >> 9);
                out[index2++] = (((in_pixel & 0x1f)) << 3) | (((in_pixel & 0x1f)) >> 2);
        }
}


inline void
convert_from_24_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        int32 index = 0;
        int32 index2 = 0;
        while (index < inRowBytes) {
                out[index2++] = in[index + 2];
                out[index2++] = in[index + 1];
                out[index2++] = in[index];
                index+=3;
        }
}


inline void
convert_from_32_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        inRowBytes /= 4;

        for (int32 i = 0; i < inRowBytes; i++) {
                out[0] = in[2];
                out[1] = in[1];
                out[2] = in[0];

                in += 4;
                out += 3;
        }
}


inline void
convert_from_32b_to_24(uint8* in, uint8* out, int32 inRowBytes)
{
        inRowBytes /= 4;

        for (int32 i = 0; i < inRowBytes; i++) {
                out[0] = in[1];
                out[1] = in[2];
                out[2] = in[3];

                in += 4;
                out += 3;
        }
}


//      #pragma mark - conversion for decompression


inline void
convert_from_CMYK_to_32_photoshop(uint8* in, uint8* out, int32 inRowBytes, int32 xStep)
{
        for (int32 i = 0; i < inRowBytes; i += 4) {
                int32 black = in[3];
                out[0] = in[2] * black / 255;
                out[1] = in[1] * black / 255;
                out[2] = in[0] * black / 255;
                out[3] = 255;

                in += 4;
                out += xStep;
        }
}


//!     !!! UNTESTED !!!
inline void
convert_from_CMYK_to_32(uint8* in, uint8* out, int32 inRowBytes, int32 xStep)
{
        for (int32 i = 0; i < inRowBytes; i += 4) {
                int32 black = 255 - in[3];
                out[0] = ((255 - in[2]) * black) / 255;
                out[1] = ((255 - in[1]) * black) / 255;
                out[2] = ((255 - in[0]) * black) / 255;
                out[3] = 255;

                in += 4;
                out += xStep;
        }
}


//!     RGB24 8:8:8 to xRGB 8:8:8:8
inline void
convert_from_24_to_32(uint8* in, uint8* out, int32 inRowBytes, int32 xStep)
{
        for (int32 i = 0; i < inRowBytes; i += 3) {
                out[0] = in[2];
                out[1] = in[1];
                out[2] = in[0];
                out[3] = 255;

                in += 3;
                out += xStep;
        }
}


//! 8-bit to 8-bit, only need when rotating the image
void
translate_8(uint8* in, uint8* out, int32 inRowBytes, int32 xStep)
{
        for (int32 i = 0; i < inRowBytes; i++) {
                out[0] = in[0];

                in++;
                out += xStep;
        }
}


} // namespace conversion


//      #pragma mark -


SSlider::SSlider(const char* name, const char* label,
                BMessage* message, int32 minValue, int32 maxValue, orientation posture,
                thumb_style thumbType, uint32 flags)
        : BSlider(name, label, message, minValue, maxValue,
                posture, thumbType, flags)
{
}


//!     Update status string - show actual value
const char*
SSlider::UpdateText() const
{
        snprintf(fStatusLabel, sizeof(fStatusLabel), "%" B_PRId32, Value());
        return fStatusLabel;
}


//      #pragma mark -


TranslatorReadView::TranslatorReadView(const char* name,
        TranslatorSettings* settings)
        :
        BView(name, 0, new BGroupLayout(B_HORIZONTAL)),
        fSettings(settings)
                // settings should already be Acquired()
{
        fAlwaysRGB32 = new BCheckBox("alwaysrgb32",
                B_TRANSLATE("Read greyscale images as RGB32"),
                new BMessage(VIEW_MSG_SET_ALWAYSRGB32));
        if (fSettings->SetGetBool(JPEG_SET_ALWAYS_RGB32, NULL))
                fAlwaysRGB32->SetValue(B_CONTROL_ON);

        fPhotoshopCMYK = new BCheckBox("photoshopCMYK",
                B_TRANSLATE("Use CMYK code with 0 for 100% ink coverage"),
                new BMessage(VIEW_MSG_SET_PHOTOSHOPCMYK));
        if (fSettings->SetGetBool(JPEG_SET_PHOTOSHOP_CMYK, NULL))
                fPhotoshopCMYK->SetValue(B_CONTROL_ON);

        fShowErrorBox = new BCheckBox("error",
                B_TRANSLATE("Show warning messages"),
                new BMessage(VIEW_MSG_SET_SHOWREADERRORBOX));
        if (fSettings->SetGetBool(JPEG_SET_SHOWREADWARNING, NULL))
                fShowErrorBox->SetValue(B_CONTROL_ON);

        BLayoutBuilder::Group<>(this, B_VERTICAL)
                .SetInsets(B_USE_DEFAULT_SPACING)
                .Add(fAlwaysRGB32)
                .Add(fPhotoshopCMYK)
                .Add(fShowErrorBox)
                .AddGlue();
}


TranslatorReadView::~TranslatorReadView()
{
        fSettings->Release();
}


void
TranslatorReadView::AttachedToWindow()
{
        BView::AttachedToWindow();

        fAlwaysRGB32->SetTarget(this);
        fPhotoshopCMYK->SetTarget(this);
        fShowErrorBox->SetTarget(this);
}


void
TranslatorReadView::MessageReceived(BMessage* message)
{
        switch (message->what) {
                case VIEW_MSG_SET_ALWAYSRGB32:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_ALWAYS_RGB32, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_PHOTOSHOPCMYK:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_PHOTOSHOP_CMYK, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_SHOWREADERRORBOX:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_SHOWREADWARNING, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                default:
                        BView::MessageReceived(message);
                        break;
        }
}


//      #pragma mark - TranslatorWriteView


TranslatorWriteView::TranslatorWriteView(const char* name,
        TranslatorSettings* settings)
        :
        BView(name, 0, new BGroupLayout(B_VERTICAL)),
        fSettings(settings)
                // settings should already be Acquired()
{
        fQualitySlider = new SSlider("quality", B_TRANSLATE("Output quality"),
                new BMessage(VIEW_MSG_SET_QUALITY), 0, 100, B_HORIZONTAL, B_TRIANGLE_THUMB);
        fQualitySlider->SetHashMarks(B_HASH_MARKS_BOTTOM);
        fQualitySlider->SetHashMarkCount(10);
        fQualitySlider->SetLimitLabels(B_TRANSLATE("Low"), B_TRANSLATE("High"));
        fQualitySlider->SetValue(fSettings->SetGetInt32(JPEG_SET_QUALITY, NULL));

        fSmoothingSlider = new SSlider("smoothing",
                B_TRANSLATE("Output smoothing strength"),
                new BMessage(VIEW_MSG_SET_SMOOTHING), 0, 100, B_HORIZONTAL, B_TRIANGLE_THUMB);
        fSmoothingSlider->SetHashMarks(B_HASH_MARKS_BOTTOM);
        fSmoothingSlider->SetHashMarkCount(10);
        fSmoothingSlider->SetLimitLabels(B_TRANSLATE("None"), B_TRANSLATE("High"));
        fSmoothingSlider->SetValue(
                fSettings->SetGetInt32(JPEG_SET_SMOOTHING, NULL));

        fProgress = new BCheckBox("progress",
                B_TRANSLATE("Use progressive compression"),
                new BMessage(VIEW_MSG_SET_PROGRESSIVE));
        if (fSettings->SetGetBool(JPEG_SET_PROGRESSIVE, NULL))
                fProgress->SetValue(B_CONTROL_ON);

        fSmallerFile = new BCheckBox("smallerfile",
                B_TRANSLATE("Make file smaller (sligthtly worse quality)"),
                new BMessage(VIEW_MSG_SET_SMALLERFILE));
        if (fSettings->SetGetBool(JPEG_SET_SMALL_FILES))
                fSmallerFile->SetValue(B_CONTROL_ON);

        fOptimizeColors = new BCheckBox("optimizecolors",
                B_TRANSLATE("Prevent colors 'washing out'"),
                new BMessage(VIEW_MSG_SET_OPTIMIZECOLORS));
        if (fSettings->SetGetBool(JPEG_SET_OPT_COLORS, NULL))
                fOptimizeColors->SetValue(B_CONTROL_ON);
        else
                fSmallerFile->SetEnabled(false);

        fGrayAsRGB24 = new BCheckBox("gray1asrgb24",
                B_TRANSLATE("Write black-and-white images as RGB24"),
                new BMessage(VIEW_MSG_SET_GRAY1ASRGB24));
        if (fSettings->SetGetBool(JPEG_SET_GRAY1_AS_RGB24))
                fGrayAsRGB24->SetValue(B_CONTROL_ON);

        BLayoutBuilder::Group<>(this, B_VERTICAL)
                .SetInsets(B_USE_DEFAULT_SPACING)
                .Add(fQualitySlider)
                .Add(fSmoothingSlider)
                .Add(fProgress)
                .Add(fOptimizeColors)
                .Add(fSmallerFile)
                .Add(fGrayAsRGB24)
                .AddGlue();
}


TranslatorWriteView::~TranslatorWriteView()
{
        fSettings->Release();
}


void
TranslatorWriteView::AttachedToWindow()
{
        BView::AttachedToWindow();

        fQualitySlider->SetTarget(this);
        fSmoothingSlider->SetTarget(this);
        fProgress->SetTarget(this);
        fOptimizeColors->SetTarget(this);
        fSmallerFile->SetTarget(this);
        fGrayAsRGB24->SetTarget(this);
}


void
TranslatorWriteView::MessageReceived(BMessage* message)
{
        switch (message->what) {
                case VIEW_MSG_SET_QUALITY:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                fSettings->SetGetInt32(JPEG_SET_QUALITY, &value);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_SMOOTHING:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                fSettings->SetGetInt32(JPEG_SET_SMOOTHING, &value);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_PROGRESSIVE:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_PROGRESSIVE, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_OPTIMIZECOLORS:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_OPT_COLORS, &boolValue);
                                fSmallerFile->SetEnabled(value);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_SMALLERFILE:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_SMALL_FILES, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                case VIEW_MSG_SET_GRAY1ASRGB24:
                {
                        int32 value;
                        if (message->FindInt32("be:value", &value) == B_OK) {
                                bool boolValue = value;
                                fSettings->SetGetBool(JPEG_SET_GRAY1_AS_RGB24, &boolValue);
                                fSettings->SaveSettings();
                        }
                        break;
                }
                default:
                        BView::MessageReceived(message);
                        break;
        }
}


//      #pragma mark -


TranslatorAboutView::TranslatorAboutView(const char* name)
        :
        BView(name, 0, new BGroupLayout(B_VERTICAL))
{
        BAlignment labelAlignment = BAlignment(B_ALIGN_LEFT, B_ALIGN_TOP);
        BStringView* title = new BStringView("Title", sTranslatorName);
        title->SetFont(be_bold_font);
        title->SetExplicitAlignment(labelAlignment);

        char versionString[100];
        snprintf(versionString, sizeof(versionString),
                B_TRANSLATE("Version %d.%d.%d"),
                (int)(sTranslatorVersion >> 8),
                (int)((sTranslatorVersion >> 4) & 0xf),
                (int)(sTranslatorVersion & 0xf));

        BStringView* version = new BStringView("Version", versionString);
        version->SetExplicitAlignment(labelAlignment);

        BTextView* infoView = new BTextView("info");
        infoView->SetText(sTranslatorInfo);
        infoView->SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
        rgb_color textColor = ui_color(B_PANEL_TEXT_COLOR);
        infoView->SetFontAndColor(be_plain_font, B_FONT_ALL, &textColor);
        infoView->MakeEditable(false);

        BLayoutBuilder::Group<>(this, B_VERTICAL, 0)
                .SetInsets(B_USE_DEFAULT_SPACING)
                .Add(title)
                .Add(version)
                .Add(infoView);
}


TranslatorView::TranslatorView(const char* name, TranslatorSettings* settings)
        :
        BTabView(name, B_WIDTH_FROM_LABEL)
{
        SetBorder(B_NO_BORDER);

        AddTab(new TranslatorWriteView(B_TRANSLATE("Write"), settings->Acquire()));
        AddTab(new TranslatorReadView(B_TRANSLATE("Read"), settings->Acquire()));
        AddTab(new TranslatorAboutView(B_TRANSLATE("About")));

        settings->Release();
}


//      #pragma mark - Translator Add-On


BView*
JPEGTranslator::NewConfigView(TranslatorSettings* settings)
{
        BView* configView = new TranslatorView("TranslatorView", settings);
        return configView;
}


/*! Determine whether or not we can handle this data */
status_t
JPEGTranslator::DerivedIdentify(BPositionIO* inSource,
        const translation_format* inFormat, BMessage* ioExtension,
        translator_info* outInfo, uint32 outType)
{
        if (outType != 0 && outType != B_TRANSLATOR_BITMAP && outType != JPEG_FORMAT)
                return B_NO_TRANSLATOR;

        // !!! You might need to make this buffer bigger to test for your native format
        off_t position = inSource->Position();
        char header[sizeof(TranslatorBitmap)];
        status_t err = inSource->Read(header, sizeof(TranslatorBitmap));
        inSource->Seek(position, SEEK_SET);
        if (err < B_OK)
                return err;

        if (B_BENDIAN_TO_HOST_INT32(((TranslatorBitmap *)header)->magic) == B_TRANSLATOR_BITMAP) {
                if (PopulateInfoFromFormat(outInfo, B_TRANSLATOR_BITMAP) != B_OK)
                        return B_NO_TRANSLATOR;
        } else {
                // First 3 bytes in jpg files are always the same from what i've seen so far
                // check them
                if (header[0] == (char)0xff && header[1] == (char)0xd8 && header[2] == (char)0xff) {
                        if (PopulateInfoFromFormat(outInfo, JPEG_FORMAT) != B_OK)
                                return B_NO_TRANSLATOR;

                } else
                        return B_NO_TRANSLATOR;
        }

        return B_OK;
}


status_t
JPEGTranslator::DerivedTranslate(BPositionIO* inSource,
        const translator_info* inInfo, BMessage* ioExtension, uint32 outType,
        BPositionIO* outDestination, int32 baseType)
{
        // Setup a "breakpoint" since throwing exceptions does not seem to work
        // at all in an add-on. (?)
        // In the be_jerror.cpp we implement a handler for critical library errors
        // (be_error_exit()) and there we use the longjmp() function to return to
        // this place. If this happens, it is as if the setjmp() call is called
        // a second time, but this time the return value will be 1. The first
        // invokation will return 0.
        jmp_buf longJumpBuffer;
        int jmpRet = setjmp(longJumpBuffer);
        if (jmpRet == 1)
                return B_ERROR;

        try {
                // What action to take, based on the findings of Identify()
                if (outType == inInfo->type) {
                        return Copy(inSource, outDestination);
                } else if (inInfo->type == B_TRANSLATOR_BITMAP
                                && outType == JPEG_FORMAT) {
                        return Compress(inSource, outDestination, &longJumpBuffer);
                } else if (inInfo->type == JPEG_FORMAT
                                && (outType == B_TRANSLATOR_BITMAP || outType == 0)) {
                        // This is the default if no specific outType was requested.
                        return Decompress(inSource, outDestination, ioExtension,
                                &longJumpBuffer);
                }
        } catch (...) {
                syslog(LOG_ERR, "libjpeg encountered a critical error (caught C++ "
                        "exception).\n");
                return B_ERROR;
        }

        return B_NO_TRANSLATOR;
}


/*!     The user has requested the same format for input and output, so just copy */
status_t
JPEGTranslator::Copy(BPositionIO* in, BPositionIO* out)
{
        int block_size = 65536;
        void* buffer = malloc(block_size);
        char temp[1024];
        if (buffer == NULL) {
                buffer = temp;
                block_size = 1024;
        }
        status_t err = B_OK;

        // Read until end of file or error
        while (1) {
                ssize_t to_read = block_size;
                err = in->Read(buffer, to_read);
                // Explicit check for EOF
                if (err == -1) {
                        if (buffer != temp) free(buffer);
                        return B_OK;
                }
                if (err <= B_OK) break;
                to_read = err;
                err = out->Write(buffer, to_read);
                if (err != to_read) if (err >= 0) err = B_DEVICE_FULL;
                if (err < B_OK) break;
        }

        if (buffer != temp) free(buffer);
        return (err >= 0) ? B_OK : err;
}


/*!     Encode into the native format */
status_t
JPEGTranslator::Compress(BPositionIO* in, BPositionIO* out,
        const jmp_buf* longJumpBuffer)
{
        using namespace conversion;

        // Read info about bitmap
        TranslatorBitmap header;
        status_t err = in->Read(&header, sizeof(TranslatorBitmap));
        if (err < B_OK)
                return err;
        else if (err < (int)sizeof(TranslatorBitmap))
                return B_ERROR;

        // Grab dimension, color space, and size information from the stream
        BRect bounds;
        bounds.left = B_BENDIAN_TO_HOST_FLOAT(header.bounds.left);
        bounds.top = B_BENDIAN_TO_HOST_FLOAT(header.bounds.top);
        bounds.right = B_BENDIAN_TO_HOST_FLOAT(header.bounds.right);
        bounds.bottom = B_BENDIAN_TO_HOST_FLOAT(header.bounds.bottom);

        int32 in_row_bytes = B_BENDIAN_TO_HOST_INT32(header.rowBytes);

        int width = bounds.IntegerWidth() + 1;
        int height = bounds.IntegerHeight() + 1;

        // Function pointer to convert function
        // It will point to proper function if needed
        void (*converter)(uchar* inscanline, uchar* outscanline,
                int32 inRowBytes) = NULL;

        // Default color info
        J_COLOR_SPACE jpg_color_space = JCS_RGB;
        int jpg_input_components = 3;
        int32 out_row_bytes;
        int padding = 0;

        switch ((color_space)B_BENDIAN_TO_HOST_INT32(header.colors)) {
                case B_CMAP8:
                        converter = convert_from_cmap8_to_24;
                        padding = in_row_bytes - width;
                        break;

                case B_GRAY1:
                        if (fSettings->SetGetBool(JPEG_SET_GRAY1_AS_RGB24, NULL)) {
                                converter = convert_from_gray1_to_24;
                        } else {
                                jpg_input_components = 1;
                                jpg_color_space = JCS_GRAYSCALE;
                                converter = convert_from_gray1_to_gray8;
                        }
                        padding = in_row_bytes - (width / 8);
                        break;

                case B_GRAY8:
                        jpg_input_components = 1;
                        jpg_color_space = JCS_GRAYSCALE;
                        padding = in_row_bytes - width;
                        break;

                case B_RGB15:
                case B_RGBA15:
                        converter = convert_from_15_to_24;
                        padding = in_row_bytes - (width * 2);
                        break;

                case B_RGB15_BIG:
                case B_RGBA15_BIG:
                        converter = convert_from_15b_to_24;
                        padding = in_row_bytes - (width * 2);
                        break;

                case B_RGB16:
                        converter = convert_from_16_to_24;
                        padding = in_row_bytes - (width * 2);
                        break;

                case B_RGB16_BIG:
                        converter = convert_from_16b_to_24;
                        padding = in_row_bytes - (width * 2);
                        break;

                case B_RGB24:
                        converter = convert_from_24_to_24;
                        padding = in_row_bytes - (width * 3);
                        break;

                case B_RGB24_BIG:
                        padding = in_row_bytes - (width * 3);
                        break;

                case B_RGB32:
                case B_RGBA32:
                        converter = convert_from_32_to_24;
                        padding = in_row_bytes - (width * 4);
                        break;

                case B_RGB32_BIG:
                case B_RGBA32_BIG:
                        converter = convert_from_32b_to_24;
                        padding = in_row_bytes - (width * 4);
                        break;

                case B_CMYK32:
                        jpg_color_space = JCS_CMYK;
                        jpg_input_components = 4;
                        padding = in_row_bytes - (width * 4);
                        break;

                default:
                        syslog(LOG_ERR, "Wrong type: Color space not implemented.\n");
                        return B_ERROR;
        }
        out_row_bytes = jpg_input_components * width;

        // Set basic things needed for jpeg writing
        struct jpeg_compress_struct cinfo;
        struct be_jpeg_error_mgr jerr;
        cinfo.err = be_jpeg_std_error(&jerr, fSettings, longJumpBuffer);
        jpeg_create_compress(&cinfo);
        be_jpeg_stdio_dest(&cinfo, out);

        // Set basic values
        cinfo.image_width = width;
        cinfo.image_height = height;
        cinfo.input_components = jpg_input_components;
        cinfo.in_color_space = jpg_color_space;
        jpeg_set_defaults(&cinfo);

        // Set better accuracy
        cinfo.dct_method = JDCT_ISLOW;

        // This is needed to prevent some colors loss
        // With it generated jpegs are as good as from Fireworks (at last! :D)
        if (fSettings->SetGetBool(JPEG_SET_OPT_COLORS, NULL)) {
                int index = 0;
                while (index < cinfo.num_components) {
                        cinfo.comp_info[index].h_samp_factor = 1;
                        cinfo.comp_info[index].v_samp_factor = 1;
                        // This will make file smaller, but with worse quality more or less
                        // like with 93%-94% (but it's subjective opinion) on tested images
                        // but with smaller size (between 92% and 93% on tested images)
                        if (fSettings->SetGetBool(JPEG_SET_SMALL_FILES))
                                cinfo.comp_info[index].quant_tbl_no = 1;
                        // This will make bigger file, but also better quality ;]
                        // from my tests it seems like useless - better quality with smaller
                        // can be acheived without this
//                      cinfo.comp_info[index].dc_tbl_no = 1;
//                      cinfo.comp_info[index].ac_tbl_no = 1;
                        index++;
                }
        }

        // Set quality
        jpeg_set_quality(&cinfo, fSettings->SetGetInt32(JPEG_SET_QUALITY, NULL), true);

        // Set progressive compression if needed
        // if not, turn on optimizing in libjpeg
        if (fSettings->SetGetBool(JPEG_SET_PROGRESSIVE, NULL))
                jpeg_simple_progression(&cinfo);
        else
                cinfo.optimize_coding = TRUE;

        // Set smoothing (effect like Blur)
        cinfo.smoothing_factor = fSettings->SetGetInt32(JPEG_SET_SMOOTHING, NULL);

        // Initialize compression
        jpeg_start_compress(&cinfo, TRUE);

        // Declare scanlines
        JSAMPROW in_scanline = NULL;
        JSAMPROW out_scanline = NULL;
        JSAMPROW writeline;
                // Pointer to in_scanline (default) or out_scanline (if there will be conversion)

        // Allocate scanline
        // Use libjpeg memory allocation functions, so in case of error it will free them itself
        in_scanline = (unsigned char *)(cinfo.mem->alloc_large)((j_common_ptr)&cinfo,
                JPOOL_PERMANENT, in_row_bytes);

        // We need 2nd scanline storage ony for conversion
        if (converter != NULL) {
                // There will be conversion, allocate second scanline...
                // Use libjpeg memory allocation functions, so in case of error it will free them itself
            out_scanline = (unsigned char *)(cinfo.mem->alloc_large)((j_common_ptr)&cinfo,
                JPOOL_PERMANENT, out_row_bytes);
                // ... and make it the one to write to file
                writeline = out_scanline;
        } else
                writeline = in_scanline;

        while (cinfo.next_scanline < cinfo.image_height) {
                // Read scanline
                err = in->Read(in_scanline, in_row_bytes);
                if (err < in_row_bytes)
                        return err < B_OK ? Error((j_common_ptr)&cinfo, err)
                                : Error((j_common_ptr)&cinfo, B_ERROR);

                // Convert if needed
                if (converter != NULL)
                        converter(in_scanline, out_scanline, in_row_bytes - padding);

                // Write scanline
                jpeg_write_scanlines(&cinfo, &writeline, 1);
        }

        jpeg_finish_compress(&cinfo);
        jpeg_destroy_compress(&cinfo);
        return B_OK;
}


/*!     Decode the native format */
status_t
JPEGTranslator::Decompress(BPositionIO* in, BPositionIO* out,
        BMessage* ioExtension, const jmp_buf* longJumpBuffer)
{
        using namespace conversion;

        // Set basic things needed for jpeg reading
        struct jpeg_decompress_struct cinfo;
        struct be_jpeg_error_mgr jerr;
        cinfo.err = be_jpeg_std_error(&jerr, fSettings, longJumpBuffer);
        jpeg_create_decompress(&cinfo);
        be_jpeg_stdio_src(&cinfo, in);

        jpeg_save_markers(&cinfo, MARKER_EXIF, 131072);
                // make sure the EXIF tag is stored

        // Read info about image
        jpeg_read_header(&cinfo, TRUE);

        BMessage exif;

        // parse EXIF data and add it ioExtension, if any
        jpeg_marker_struct* marker = cinfo.marker_list;
        while (marker != NULL) {
                if (marker->marker == MARKER_EXIF
                        && !strncmp((char*)marker->data, "Exif", 4)) {
                        if (ioExtension != NULL) {
                                // Strip EXIF header from TIFF data
                                ioExtension->AddData("exif", B_RAW_TYPE,
                                        (uint8*)marker->data + 6, marker->data_length - 6);
                        }

                        BMemoryIO io(marker->data + 6, marker->data_length - 6);
                        convert_exif_to_message(io, exif);
                }
                marker = marker->next;
        }

        // Default color info
        color_space outColorSpace = B_RGB32;
        int outColorComponents = 4;

        // Function pointer to convert function
        // It will point to proper function if needed
        void (*converter)(uchar* inScanLine, uchar* outScanLine,
                int32 inRowBytes, int32 xStep) = convert_from_24_to_32;

        // If color space isn't rgb
        if (cinfo.out_color_space != JCS_RGB) {
                switch (cinfo.out_color_space) {
                        case JCS_UNKNOWN:               /* error/unspecified */
                                syslog(LOG_ERR, "From Type: Jpeg uses unknown color type\n");
                                break;
                        case JCS_GRAYSCALE:             /* monochrome */
                                // Check if user wants to read only as RGB32 or not
                                if (!fSettings->SetGetBool(JPEG_SET_ALWAYS_RGB32, NULL)) {
                                        // Grayscale
                                        outColorSpace = B_GRAY8;
                                        outColorComponents = 1;
                                        converter = translate_8;
                                } else {
                                        // RGB
                                        cinfo.out_color_space = JCS_RGB;
                                        cinfo.output_components = 3;
                                        converter = convert_from_24_to_32;
                                }
                                break;
                        case JCS_YCbCr:         /* Y/Cb/Cr (also known as YUV) */
                                cinfo.out_color_space = JCS_RGB;
                                converter = convert_from_24_to_32;
                                break;
                        case JCS_YCCK:          /* Y/Cb/Cr/K */
                                // Let libjpeg convert it to CMYK
                                cinfo.out_color_space = JCS_CMYK;
                                // Fall through to CMYK since we need the same settings
                        case JCS_CMYK:          /* C/M/Y/K */
                                // Use proper converter
                                if (fSettings->SetGetBool(JPEG_SET_PHOTOSHOP_CMYK))
                                        converter = convert_from_CMYK_to_32_photoshop;
                                else
                                        converter = convert_from_CMYK_to_32;
                                break;
                        default:
                                syslog(LOG_ERR,
                                                "From Type: Jpeg uses hmm... i don't know really :(\n");
                                break;
                }
        }

        // Initialize decompression
        jpeg_start_decompress(&cinfo);

        // retrieve orientation from settings/EXIF
        int32 orientation;
        if (ioExtension == NULL
                || ioExtension->FindInt32("exif:orientation", &orientation) != B_OK) {
                if (exif.FindInt32("Orientation", &orientation) != B_OK)
                        orientation = 1;
        }

        if (orientation != 1 && converter == NULL)
                converter = translate_8;

        int32 outputWidth = orientation > 4 ? cinfo.output_height : cinfo.output_width;
        int32 outputHeight = orientation > 4 ? cinfo.output_width : cinfo.output_height;

        int32 destOffset = dest_index(outputWidth, outputHeight,
                0, 0, orientation) * outColorComponents;
        int32 xStep = dest_index(outputWidth, outputHeight,
                1, 0, orientation) * outColorComponents - destOffset;
        int32 yStep = dest_index(outputWidth, outputHeight,
                0, 1, orientation) * outColorComponents - destOffset;
        bool needAll = orientation != 1;

        // Initialize this bounds rect to the size of your image
        BRect bounds(0, 0, outputWidth - 1, outputHeight - 1);

#if 0
printf("destOffset = %ld, xStep = %ld, yStep = %ld, input: %ld x %ld, output: %ld x %ld, orientation %ld\n",
        destOffset, xStep, yStep, (int32)cinfo.output_width, (int32)cinfo.output_height,
        bounds.IntegerWidth() + 1, bounds.IntegerHeight() + 1, orientation);
#endif

        // Bytes count in one line of image (scanline)
        int32 inRowBytes = cinfo.output_width * cinfo.output_components;
        int32 rowBytes = (bounds.IntegerWidth() + 1) * outColorComponents;
        int32 dataSize = cinfo.output_width * cinfo.output_height
                * outColorComponents;

        // Fill out the B_TRANSLATOR_BITMAP's header
        TranslatorBitmap header;
        header.magic = B_HOST_TO_BENDIAN_INT32(B_TRANSLATOR_BITMAP);
        header.bounds.left = B_HOST_TO_BENDIAN_FLOAT(bounds.left);
        header.bounds.top = B_HOST_TO_BENDIAN_FLOAT(bounds.top);
        header.bounds.right = B_HOST_TO_BENDIAN_FLOAT(bounds.right);
        header.bounds.bottom = B_HOST_TO_BENDIAN_FLOAT(bounds.bottom);
        header.colors = (color_space)B_HOST_TO_BENDIAN_INT32(outColorSpace);
        header.rowBytes = B_HOST_TO_BENDIAN_INT32(rowBytes);
        header.dataSize = B_HOST_TO_BENDIAN_INT32(dataSize);

        // Write out the header
        status_t err = out->Write(&header, sizeof(TranslatorBitmap));
        if (err < B_OK)
                return Error((j_common_ptr)&cinfo, err);
        else if (err < (int)sizeof(TranslatorBitmap))
                return Error((j_common_ptr)&cinfo, B_ERROR);

        // Declare scanlines
        JSAMPROW inScanLine = NULL;
        uint8* dest = NULL;
        uint8* destLine = NULL;

        // Allocate scanline
        // Use libjpeg memory allocation functions, so in case of error it will free them itself
    inScanLine = (unsigned char *)(cinfo.mem->alloc_large)((j_common_ptr)&cinfo,
        JPOOL_PERMANENT, inRowBytes);

        // We need 2nd scanline storage only for conversion
        if (converter != NULL) {
                // There will be conversion, allocate second scanline...
                // Use libjpeg memory allocation functions, so in case of error it will
                // free them itself
            dest = (uint8*)(cinfo.mem->alloc_large)((j_common_ptr)&cinfo,
                JPOOL_PERMANENT, needAll ? dataSize : rowBytes);
            destLine = dest + destOffset;
        } else
                destLine = inScanLine;

        while (cinfo.output_scanline < cinfo.output_height) {
                // Read scanline
                jpeg_read_scanlines(&cinfo, &inScanLine, 1);

                // Convert if needed
                if (converter != NULL)
                        converter(inScanLine, destLine, inRowBytes, xStep);

                if (!needAll) {
                        // Write the scanline buffer to the output stream
                        ssize_t bytesWritten = out->Write(destLine, rowBytes);
                        if (bytesWritten < rowBytes) {
                                return bytesWritten < B_OK
                                        ? Error((j_common_ptr)&cinfo, bytesWritten)
                                        : Error((j_common_ptr)&cinfo, B_ERROR);
                        }
                } else
                        destLine += yStep;
        }

        if (needAll) {
                ssize_t bytesWritten = out->Write(dest, dataSize);
                if (bytesWritten < dataSize) {
                        return bytesWritten < B_OK
                                ? Error((j_common_ptr)&cinfo, bytesWritten)
                                : Error((j_common_ptr)&cinfo, B_ERROR);
                }
        }

        jpeg_finish_decompress(&cinfo);
        jpeg_destroy_decompress(&cinfo);
        return B_OK;
}

/*! have the other PopulateInfoFromFormat() check both inputFormats & outputFormats */
status_t
JPEGTranslator::PopulateInfoFromFormat(translator_info* info,
        uint32 formatType, translator_id id)
{
        int32 formatCount;
        const translation_format* formats = OutputFormats(&formatCount);
        for (int i = 0; i <= 1 ;formats = InputFormats(&formatCount), i++) {
                if (PopulateInfoFromFormat(info, formatType,
                        formats, formatCount) == B_OK) {
                        info->translator = id;
                        return B_OK;
                }
        }

        return B_ERROR;
}


status_t
JPEGTranslator::PopulateInfoFromFormat(translator_info* info,
        uint32 formatType, const translation_format* formats, int32 formatCount)
{
        for (int i = 0; i < formatCount; i++) {
                if (formats[i].type == formatType) {
                        info->type = formatType;
                        info->group = formats[i].group;
                        info->quality = formats[i].quality;
                        info->capability = formats[i].capability;
                        BString str1(formats[i].name);
                        str1.ReplaceFirst("Be Bitmap Format (JPEGTranslator)",
                                B_TRANSLATE("Be Bitmap Format (JPEGTranslator)"));
                        strlcpy(info->name, str1.String(), sizeof(info->name));
                        strcpy(info->MIME,  formats[i].MIME);
                        return B_OK;
                }
        }

        return B_ERROR;
}

/*!
        Frees jpeg alocated memory
        Returns given error (B_ERROR by default)
*/
status_t
JPEGTranslator::Error(j_common_ptr cinfo, status_t error)
{
        jpeg_destroy(cinfo);
        return error;
}


JPEGTranslator::JPEGTranslator()
        : BaseTranslator(sTranslatorName, sTranslatorInfo, sTranslatorVersion,
                sInputFormats,  kNumInputFormats,
                sOutputFormats, kNumOutputFormats,
                SETTINGS_FILE,
                sDefaultSettings, kNumDefaultSettings,
                B_TRANSLATOR_BITMAP, JPEG_FORMAT)
{}


BTranslator*
make_nth_translator(int32 n, image_id you, uint32 flags, ...)
{
        if (n == 0)
                return new JPEGTranslator();

        return NULL;
}


int
main(int, char**)
{
        BApplication app("application/x-vnd.Haiku-JPEGTranslator");
        JPEGTranslator* translator = new JPEGTranslator();
        if (LaunchTranslatorWindow(translator, sTranslatorName) == B_OK)
                app.Run();

        return 0;
}