Преобразование RGBA8888 в RGBA5551

Я пишу ImageEncoder что пишет TGA образ. Я был в состоянии успешно написать TGA файл, но вместо того, чтобы в конечном итоге [RRRRRGGGGGBBBBBA] я получил [RGBBBBBA] вот соответствующий код:

int lastRow = minY + height;
for (int row = minY; row < lastRow; row += 8) {
    int rows = Math.min(8, lastRow - row);
    int size = rows * width * numBands;

    // Grab the pixels
    Raster src = im.getData(new Rectangle(minX, row, width, rows));
    src.getPixels(minX, row, width, rows, pixels);

    for (int i = 0; i < size; i++) {
        //output.write(pixels[i] & 0xFF);
        //corrected
        //before conversion (source image) pixel in RGBA8888 format.
        int o = (int) pixels[i];
        //Need to convert here...
        short converted = ?;
        //need to write out RGB5551
        output.write(converted);
    }
}

Чтобы уточнить, что я пытаюсь сделать... У меня есть исходное изображение в формате PNG, глубина цвета RGBA8888. Мне нужно преобразовать это изображение в формат TGA с глубиной цвета RGBA5551. Цикл для выше, где я обращаюсь к отдельным пикселям. Итак, я спрашиваю: как правильно прочитать 32-разрядное целое число (RGBA8888) и преобразовать его в 16-разрядное короткое (RGBA5551)?

2 ответа

Решение

К сожалению, никто не смог мне помочь с этим, но я хотел бы поделиться ответом с сообществом.

 private void writeoutFromPNGFile(int width, int numBands, int minY, int height,         RenderedImage im, int minX) throws IOException {
    System.out.println("Writing from PNG data.");
    //convertTo2DWithoutUsingGetRGB(im);
    //now start writing the file...
    int[] pixels = new int[8 * width * numBands];

    int count = 0;

    // Process 8 rows at a time so all but the last will have
    // a multiple of 8 pixels.  This simplifies PBM_RAW encoding.
    int lastRow = minY + height;
    for (int row = minY; row < lastRow; row += 8) {
        int rows = Math.min(8, lastRow - row);
        int size = rows * width * numBands;

        // Grab the pixels
        Raster src = im.getData(new Rectangle(minX, row, width, rows));
        src.getPixels(minX, row, width, rows, pixels);

        for (int i = 0; i < size; i += 4) {
            //output.write(pixels[i] & 0xFF);
            int red = pixels[i];
            int green = pixels[i + 1];
            int blue = pixels[i + 2];
            int alpha = pixels[i + 3];
            //System.out.println("Pixel Value: " + o);
            convertTo5551(red, green, blue, alpha);
        }

    }
}


int convertTo5551(int r, int g, int b, int a) throws IOException {
    int r5 = r * 31 / 255;
    int g5 = (int) g * 31 / 255;
    int b5 = (int) b * 31 / 255;
    int a1 = (a > 0) ? 0 : -1;
    int rShift = (int) r5 << 11;
    int bShift = (int) g5 << 6;
    int gShift = (int) b5 << 1;
    // Combine and return
    int abgr5551 = (int) (bShift | gShift | rShift | a1);
    output.write(new BigDecimal((abgr5551) & 0xFF).byteValue());
    output.write(new BigDecimal((abgr5551 >> 8) & 0xFF).byteValue());
    return abgr5551;
}

Цвета конвертируются с использованием линейной математики - не очень хорошая идея для хороших результатов. Снижение цвета действительно должно быть сделано с помощью программы, такой как ImageMagick.

Так как Растр знает о своем собственном источнике (прямоугольнике), я ожидал бы прочитать src.getPixels( 0, 0, width,rows,pixels), Хотя не пробовал.

Другие вопросы по тегам