// Template array classes
/*
Copyright (C) 1996-2015 John W. Eaton
Copyright (C) 2010 VZLU Prague
This file is part of Octave.
Octave 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.
Octave 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 Octave; see the file COPYING. If not, see
.
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include
#include "DiagArray2.h"
#include "lo-error.h"
template
DiagArray2::DiagArray2 (const Array& a, octave_idx_type r,
octave_idx_type c)
: Array (a.as_column ()), d1 (r), d2 (c)
{
octave_idx_type rcmin = std::min (r, c);
if (rcmin != a.length ())
Array::resize (dim_vector (rcmin, 1));
}
template
Array
DiagArray2::diag (octave_idx_type k) const
{
return extract_diag (k);
}
template
Array
DiagArray2::extract_diag (octave_idx_type k) const
{
Array d;
if (k == 0)
// The main diagonal is shallow-copied.
d = *this;
else if (k > 0 && k < cols ())
d = Array (dim_vector (std::min (cols () - k, rows ()), 1), T ());
else if (k < 0 && -k < rows ())
d = Array (dim_vector (std::min (rows () + k, cols ()), 1), T ());
else // Matlab returns [] 0x1 for out-of-range diagonal
d.resize (dim_vector (0, 1));
return d;
}
template
DiagArray2
DiagArray2::transpose (void) const
{
return DiagArray2 (*this, d2, d1);
}
template
DiagArray2
DiagArray2::hermitian (T (* fcn) (const T&)) const
{
return DiagArray2 (Array::template map (fcn), d2, d1);
}
// A two-dimensional array with diagonal elements only.
template
void
DiagArray2::resize (octave_idx_type r, octave_idx_type c,
const T& rfv)
{
if (r < 0 || c < 0)
{
(*current_liboctave_error_handler) ("can't resize to negative dimensions");
return;
}
if (r != dim1 () || c != dim2 ())
{
Array::resize (dim_vector (std::min (r, c), 1), rfv);
d1 = r; d2 = c;
}
}
template
Array
DiagArray2::array_value (void) const
{
Array result (dims (), T (0));
for (octave_idx_type i = 0, len = length (); i < len; i++)
result.xelem (i, i) = dgelem (i);
return result;
}
template
bool
DiagArray2::check_idx (octave_idx_type r, octave_idx_type c) const
{
bool ok = true;
if (r < 0 || r >= dim1 ())
{
gripe_index_out_of_range (2, 1, r+1, dim1 ());
ok = false;
}
if (c < 0 || c >= dim2 ())
{
gripe_index_out_of_range (2, 2, c+1, dim2 ());
ok = false;
}
return ok;
}