/*
* PROJECT: NyARToolkit
* --------------------------------------------------------------------------------
* This work is based on the original ARToolKit developed by
* Hirokazu Kato
* Mark Billinghurst
* HITLab, University of Washington, Seattle
* http://www.hitl.washington.edu/artoolkit/
*
* The NyARToolkit is Java edition ARToolKit class library.
* Copyright (C)2008-2009 Ryo Iizuka
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
* For further information please contact.
* http://nyatla.jp/nyatoolkit/
* <airmail(at)ebony.plala.or.jp> or <nyatla(at)nyatla.jp>
*
*/
package jp.nyatla.nyartoolkit.core.raster;
import jp.nyatla.nyartoolkit.NyARException;
import jp.nyatla.nyartoolkit.core.types.*;
/**
* このクラスは、グレースケース画像を格納するラスタクラスです。
* 外部バッファ、内部バッファの両方に対応します。
*/
public class NyARGrayscaleRaster extends NyARRaster_BasicClass
{
private IdoFilterImpl _impl;
/** バッファオブジェクト*/
protected Object _buf;
/** バッファオブジェクトがアタッチされていればtrue*/
protected boolean _is_attached_buffer;
/**
* コンストラクタです。
* 内部参照のバッファ({@link NyARBufferType#INT1D_GRAY_8}形式)を持つインスタンスを生成します。
* @param i_width
* ラスタのサイズ
* @param i_height
* ラスタのサイズ
* @throws NyARException
*/
public NyARGrayscaleRaster(int i_width, int i_height) throws NyARException
{
super(i_width, i_height, NyARBufferType.INT1D_GRAY_8);
if (!initInstance(this._size, NyARBufferType.INT1D_GRAY_8, true))
{
throw new NyARException();
}
}
/**
* コンストラクタです。
* 画像のサイズパラメータとバッファ参照方式を指定して、インスタンスを生成します。
* バッファの形式は、{@link NyARBufferType#INT1D_GRAY_8}です。
* @param i_width
* ラスタのサイズ
* @param i_height
* ラスタのサイズ
* @param i_is_alloc
* バッファを外部参照にするかのフラグ値。
* trueなら内部バッファ、falseなら外部バッファを使用します。
* falseの場合、初期のバッファはnullになります。インスタンスを生成したのちに、{@link #wrapBuffer}を使って割り当ててください。
* @throws NyARException
*/
public NyARGrayscaleRaster(int i_width, int i_height, boolean i_is_alloc)
throws NyARException
{
super(i_width, i_height, NyARBufferType.INT1D_GRAY_8);
if (!initInstance(this._size, NyARBufferType.INT1D_GRAY_8, i_is_alloc))
{
throw new NyARException();
}
}
/**
* コンストラクタです。
* 画像のサイズパラメータとバッファ形式を指定して、インスタンスを生成します。
* @param i_width
* ラスタのサイズ
* @param i_height
* ラスタのサイズ
* @param i_raster_type
* ラスタのバッファ形式。
* {@link NyARBufferType}に定義された定数値を指定してください。指定できる値は、以下の通りです。
* <ul>
* <li>{@link NyARBufferType#INT1D_GRAY_8}
* <ul>
* @param i_is_alloc
* バッファを外部参照にするかのフラグ値。
* trueなら内部バッファ、falseなら外部バッファを使用します。
* falseの場合、初期のバッファはnullになります。インスタンスを生成したのちに、{@link #wrapBuffer}を使って割り当ててください。
* @throws NyARException
*/
public NyARGrayscaleRaster(int i_width, int i_height, int i_raster_type,boolean i_is_alloc) throws NyARException
{
super(i_width, i_height, i_raster_type);
if (!initInstance(this._size, i_raster_type, i_is_alloc)) {
throw new NyARException();
}
}
/**
* このクラスの初期化シーケンスです。コンストラクタから呼び出します。
* @param i_size
* ラスタサイズ
* @param i_buf_type
* バッファ形式
* @param i_is_alloc
* バッファ参照方法値
* @return
* 初期化に成功するとtrueを返します。
*/
protected boolean initInstance(NyARIntSize i_size, int i_buf_type,boolean i_is_alloc)
{
switch (i_buf_type) {
case NyARBufferType.INT1D_GRAY_8:
this._impl=new IdoFilterImpl_INT1D_GRAY_8();
this._buf = i_is_alloc ? new int[i_size.w * i_size.h] : null;
break;
default:
return false;
}
this._is_attached_buffer = i_is_alloc;
return true;
}
/**
* この関数は、ラスタのバッファへの参照値を返します。
* バッファの形式は、コンストラクタに指定した形式と同じです。
*/
public Object getBuffer()
{
return this._buf;
}
/**
* この関数は、インスタンスがバッファを所有するかを返します。
* 内部参照バッファの場合は、常にtrueです。
* 外部参照バッファの場合は、バッファにアクセスする前に、このパラメタを確認してください。
*/
public boolean hasBuffer()
{
return this._buf != null;
}
/**
* この関数は、ラスタに外部参照バッファをセットします。
* 外部参照バッファを持つインスタンスでのみ使用できます。内部参照バッファを持つインスタンスでは使用できません。
*/
public void wrapBuffer(Object i_ref_buf) throws NyARException
{
assert (!this._is_attached_buffer);// バッファがアタッチされていたら機能しない。
this._buf = i_ref_buf;
}
/**
* この関数は、指定した数値でラスタを埋めます。
* この関数は高速化していません。
* @param i_value
* 埋める数値を指定します。0から255の数値を指定して下さい。
*/
public void fill(int i_value)
{
this._impl.fill(this,i_value);
}
/**
* この関数は、指定ラスタに、このインスタンスの内容をコピーをします。
* 異解像度間コピーもできますが、入力範囲とラスタサイズの間には、一定の関係が成立する必要があります。
* @param i_left
* 入力ラスタの左上点を指定します。
* @param i_top
* 入力ラスタの左上点を指定します。
* @param i_skip
* skip値。1なら等倍、2なら1/2倍、3なら1/3倍の画像を出力します。
* @param o_output
* 出力先ラスタ。このラスタの解像度は、w=(i_input.w-i_left)/i_skip,h=(i_input.h-i_height)/i_skipを満たす必要があります。
* 出力先ラスタと入力ラスタの画素形式は、同じである必要があります。
*/
public void copyTo(int i_left,int i_top,int i_skip, NyARGrayscaleRaster o_output)
{
assert (this.getSize().isInnerSize(i_left + o_output.getWidth() * i_skip, i_top+ o_output.getHeight() * i_skip));
assert (this.isEqualBufferType(o_output.getBufferType()));
this._impl.copyTo(this, i_left, i_top, i_skip, o_output);
return;
}
////////////////////////////////////////////////////////////////////////////////
//ここからラスタドライバ
/** ラスタドライバの関数を定義します。*/
protected interface IdoFilterImpl
{
public void fill(NyARGrayscaleRaster i_raster,int i_value);
public void copyTo(NyARGrayscaleRaster i_input, int i_left,int i_top,int i_skip, NyARGrayscaleRaster o_output);
}
/** {@link NyARBufferType#INT1D_GRAY_8}形式のラスタドライバです。*/
protected final class IdoFilterImpl_INT1D_GRAY_8 implements IdoFilterImpl
{
public void fill(NyARGrayscaleRaster i_raster,int i_value)
{
assert (i_raster._buffer_type == NyARBufferType.INT1D_GRAY_8);
int[] buf = (int[]) i_raster._buf;
for (int i = i_raster._size.h * i_raster._size.w - 1; i >= 0; i--) {
buf[i] = i_value;
}
}
public void copyTo(NyARGrayscaleRaster i_input, int i_left,int i_top,int i_skip, NyARGrayscaleRaster o_output)
{
assert (i_input.getSize().isInnerSize(i_left + o_output.getWidth() * i_skip, i_top+ o_output.getHeight() * i_skip));
final int[] input = (int[]) i_input.getBuffer();
final int[] output = (int[]) o_output.getBuffer();
int pt_src, pt_dst;
NyARIntSize dest_size = o_output.getSize();
NyARIntSize src_size = i_input.getSize();
int skip_src_y = (src_size.w - dest_size.w * i_skip) + src_size.w * (i_skip - 1);
final int pix_count = dest_size.w;
final int pix_mod_part = pix_count - (pix_count % 8);
// 左上から1行づつ走査していく
pt_dst = 0;
pt_src = (i_top * src_size.w + i_left);
for (int y = dest_size.h - 1; y >= 0; y -= 1) {
int x;
for (x = pix_count - 1; x >= pix_mod_part; x--) {
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
}
for (; x >= 0; x -= 8) {
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
output[pt_dst++] = input[pt_src];
pt_src += i_skip;
}
// スキップ
pt_src += skip_src_y;
}
return;
}
}
}