# frozen_string_literal: true

module Vedeu

  module DSL

    # @todo This documentation needs editing and is out of date.
    # (GL: 2015-12-25)
    #
    # Provides methods to be used to define views.
    #
    #   Vedeu.renders do
    #     view :my_interface do
    #       lines do
    #         background '#000000'
    #         foreground '#ffffff'
    #         line 'This is white text on a black background.'
    #         line 'Next is a blank line:'
    #         line ''
    #
    #         streams { stream 'We can define ' }
    #
    #         streams do
    #           foreground '#ff0000'
    #           stream 'parts of a line '
    #         end
    #
    #         streams { stream 'independently using ' }
    #
    #         streams do
    #           foreground '#00ff00'
    #           stream 'streams.'
    #         end
    #       end
    #     end
    #   end
    #
    # @api public
    #
    module Elements

      include Vedeu::Common
      include Vedeu::DSL::Presentation

      # @todo This documentation needs editing. (GL: 2015-12-17)
      #
      # Specify multiple lines in a view.
      #
      # @example
      #   Vedeu.view :my_interface do
      #     lines do
      #       # ... see {Vedeu::DSL::Line} and {Vedeu::DSL::Stream}
      #     end
      #   end
      #
      #   Vedeu.view :my_interface do
      #     line do
      #       # ... see {Vedeu::DSL::Line} and {Vedeu::DSL::Stream}
      #     end
      #   end
      #
      # @macro param_block
      # @macro raise_requires_block
      # @macro raise_fatal
      # @return [void]
      def lines(opts = {}, &block)
        requires_block!(&block)
        requires_model!

        attrs = build_attributes(nil, opts, &block)

        if view_model?
          if model.lines?
            model.add(build_line(attrs, &block))

          else
            model.value = build_view(attrs, &block).value

          end

        elsif line_model?
          model.value = build_line(attrs, &block).value

        else
          model.value = build_view(attrs, &block).value

        end
      end
      alias streams lines

      # @todo This documentation needs editing. (GL: 2015-12-17)
      #
      # Specify a single line in a view.
      #
      #   Vedeu.renders do
      #     view :my_interface do
      #       lines do
      #         line 'some text...'
      #         # ... some code
      #
      #         line 'some more text...'
      #         # ... some code
      #       end
      #     end
      #   end
      #
      # @param value [String] The value for the line. Ignored when a
      #   block is given.
      # @param opts [Hash]
      # @option opts ... [void]
      # @macro param_block
      # @macro raise_fatal
      # @return [void]
      def line(value = '', opts = {}, &block)
        requires_model!

        attrs = build_attributes(value, opts, &block)

        l = if block_given?
              build_line(attrs, &block)

            else
              new_line(attrs)

            end

        if view_model?
          model.add(l)

        elsif line_model?
          model.add(l.value)

        else
          raise Vedeu::Error::Fatal,
                "Cannot add line to '#{model.class.name}' model."

        end
      end

      # @param value [String] The value for the stream. Ignored when a
      #   block is given.
      # @param opts [Hash]
      # @option opts ... [void]
      # @macro param_block
      # @macro raise_fatal
      # @return [void]
      def stream(value = '', opts = {}, &block)
        requires_model!

        attrs = build_attributes(value, opts, &block)

        m = if block_given?
              if view_model?
                build_line(attrs, &block)

              else
                build_stream(attrs, &block)

              end

            else
              new_line(attrs)

            end

        if view_model? || line_model? || stream_model?
          model.add(m)

        else
          raise Vedeu::Error::Fatal,
                "Cannot add stream to '#{model.class.name}' model."

        end
      end

      # @todo This documentation needs editing. (GL: 2015-12-17)
      #
      # Specify the content for a view. Provides the means to align a
      # string (or object responding to `to_s`), and add it as a Line
      # or to the Stream.
      #
      # @note If using the convenience methods; left, centre, center
      #   or right, then a specified align option will be ignored.
      #
      # @example
      #   lines do
      #     centre '...'
      #   end
      #
      #   line do
      #     right '...'
      #   end
      #
      #   line do
      #     stream do
      #       text '...'
      #     end
      #   end
      #
      #   left 'This will be left aligned.', width: 35
      #   # => 'This will be left aligned.         '
      #
      #   centre 'This will be aligned centrally.', width: 35
      #   # => '  This will be aligned centrally.  '
      #   # centre is also aliased to center
      #
      #   right 'This will be right aligned.', width: 35
      #   # => '        This will be right aligned.'
      #
      #   right 'This will be right aligned.', width: 35,
      #     align: centre
      #
      #   text 'This will be truncated here. More text here.',
      #     width: 28 # => 'This will be truncated here.'
      #
      #   text 'Padded with hyphens.', width: 25, pad: '-',
      #     align: :right # => '-----Padded with hyphens.'
      #
      # @param value [String|Object] A string or object that responds
      #   to `to_s`.
      # @param opts [Hash<Symbol => void>] Text options.
      # @option opts :align [Symbol] One of `:left`,
      #   `:centre`/`:center`, or `:right`.
      # @option opts :width [Integer|NilClass] The width of the
      #   text stream to add. If the `string` provided is longer than
      #   this value, the string will be truncated. If no width is
      #   provided in the context of 'lines', then the interface width
      #   is used. If no width is provided in the context of a
      #   'stream', then no alignment will occur.
      # @option opts :pad [String] The character to use to pad the
      #   width, by default uses an empty space (0x20). Only when the
      #   string is shorter than the specified width.
      # @macro raise_fatal
      # @return [void]
      def text(value = '', opts = {})
        requires_model!

        attrs = build_attributes(value, opts)

        if view_model?
          model.add(new_line(attrs))

        elsif line_model?
          model.add(new_stream(attrs))

        elsif stream_model?
          model.add(new_stream(attrs).value)

        else
          raise Vedeu::Error::Fatal,
                "Cannot add text to '#{model.class.name}' model."

        end
      end

      # @param (see #text)
      # @return (see #text)
      def centre(value = '', opts = {})
        text(value, opts.merge(align: :centre))
      end
      alias center centre

      # @param (see #text)
      # @return (see #text)
      def left(value = '', opts = {})
        text(value, opts.merge(align: :left))
      end

      # @param (see #text)
      # @return (see #text)
      def right(value = '', opts = {})
        text(value, opts.merge(align: :right))
      end

      private

      # @param value [String] The value for the stream.
      # @param opts [Hash]
      # @option opts ... [void]
      # @macro param_block
      # @return [Vedeu::DSL::Attributes]
      def build_attributes(value = '', opts = {}, &block)
        Vedeu::DSL::Attributes.build(self, model, value, opts, &block)
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @macro param_block
      # @return [Vedeu::Views::Line]
      def build_line(attrs, &block)
        Vedeu::Views::Line.build(attrs, &block)
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @macro param_block
      # @return [Vedeu::Views::Stream]
      def build_stream(attrs, &block)
        Vedeu::Views::Stream.build(attrs, &block)
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @macro param_block
      # @return [Vedeu::Views::View]
      def build_view(attrs, &block)
        Vedeu::Views::View.build(attrs, &block)
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @return [Vedeu::Views::Line]
      def new_line(attrs)
        Vedeu::Views::Line.new(attrs.merge(value: new_streams(attrs)))
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @return [Vedeu::Views::Stream]
      def new_stream(attrs)
        Vedeu::Views::Stream.new(attrs)
      end

      # @param attrs [Vedeu::DSL::Attributes]
      # @return [Vedeu::Views::Streams]
      def new_streams(attrs)
        Vedeu::Views::Streams.coerce([new_stream(attrs)])
      end

      # @macro param_block
      # @macro raise_requires_block
      # @return [NilClass]
      def requires_block!(&block)
        raise Vedeu::Error::RequiresBlock unless block_given?
      end

      # @macro raise_fatal
      # @return [NilClass]
      def requires_model!
        raise Vedeu::Error::Fatal,
              'No model, cannot continue.' unless present?(model)
      end

    end # Elements

  end # DSL

end # Vedeu