// --------------------------------------------------------------------------- // - Printer.cpp - // - afnix engine - printer class implementation - // --------------------------------------------------------------------------- // - This program is free software; you can redistribute it and/or modify - // - it provided that this copyright notice is kept intact. - // - - // - 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. In no event shall - // - the copyright holder be liable for any direct, indirect, incidental or - // - special damages arising in any way out of the use of this software. - // --------------------------------------------------------------------------- // - copyright (c) 1999-2007 amaury darsch - // --------------------------------------------------------------------------- #include "Cons.hpp" #include "Interp.hpp" #include "Printer.hpp" #include "Exception.hpp" namespace afnix { // ------------------------------------------------------------------------- // - class section - // ------------------------------------------------------------------------- // create a new printer class Printer::Printer (t_type type ) { d_type = type; } // return the class name String Printer::repr (void) const { return "Printer"; } // ------------------------------------------------------------------------- // - object section - // ------------------------------------------------------------------------- // apply this object on the runnable streams Object* Printer::apply (Runnable* robj, Nameset* nset, Cons* args) { rdlock (); try { // get the runnable stream based on the printer type Output* os = nilp; if ((d_type == Printer::OUTPUT) || (d_type == Printer::OUTPUTLN)) os = robj->getos (); if ((d_type == Printer::ERROR) || (d_type == Printer::ERRORLN)) os = robj->getes (); // loop for each argument String result; while (args != nilp) { Object* car = args->getcar (); try { Object* obj = (car == nilp) ? nilp : car->eval (robj,nset); if (obj == nilp) result = result + "nil"; else { Literal* lit = dynamic_cast <Literal*> (obj); if (lit == nilp) { String repr = Object::repr (obj); Object::cref (obj); throw Exception ("type-error", "non literal object for printer", repr); } result = result + lit->tostring (); Object::cref (obj); } } catch (Exception& e) { e.setnlf (true); throw e; } args = args->getcdr (); } // check for newline if ((d_type == Printer::OUTPUTLN) || (d_type == Printer::ERRORLN)) result = result + eolc; os->write (result); unlock (); return nilp; } catch (...) { unlock (); throw; } } }