4cc3010
#!/usr/bin/perl
4cc3010
4cc3010
# converts the <rules>.xml file to the old format <rules>.lst file
4cc3010
#
4cc3010
# Usage:
4cc3010
#
4cc3010
# perl xml2lst.pl < filename.xml > filename.lst
4cc3010
#
4cc3010
# author Ivan Pascal
4cc3010
# modified by Vitezslav Crhonek
4cc3010
4cc3010
$doc = new_document( 0, '');
4cc3010
parse('', $doc);
4cc3010
4cc3010
($reg)   = node_by_name($doc, '/xkbConfigRegistry');
4cc3010
@models  = node_by_name($reg, 'modelList/model/configItem');
4cc3010
@layouts = node_by_name($reg, 'layoutList/layout/configItem');
4cc3010
@options = node_by_name($reg, 'optionList/group/configItem');
4cc3010
4cc3010
for $i (@layouts) {
4cc3010
   ($name) = node_by_name($i, 'name');
4cc3010
   @variants = node_by_name($i, '../variantList/variant/configItem');
4cc3010
   for $v (@variants) {
4cc3010
      ($variant) = node_by_name($v, 'name');
4cc3010
      printf("%s %s\n", text_child($name), text_child($variant));
4cc3010
   }
4cc3010
}
4cc3010
4cc3010
sub with_attribute {
4cc3010
    local ($nodelist, $attrexpr) = @_;
4cc3010
    local ($attr, $value) = split (/=/, $attrexpr);
4cc3010
    local ($node, $attrvalue);
4cc3010
    if (defined $value && $value ne '') {
4cc3010
        $value =~ s/"//g;
4cc3010
        foreach $node (@{$nodelist}) {
4cc3010
           $attrvalue = node_attribute($node, $attr); 
4cc3010
           if (defined $attrvalue && $attrvalue eq $value) {
4cc3010
               return $node;
4cc3010
           }
4cc3010
        }
4cc3010
    } else {
4cc3010
        foreach $node (@{$nodelist}) {
4cc3010
           if (! defined node_attribute($node, $attr)) {
4cc3010
               return $node;
4cc3010
           }
4cc3010
        }
4cc3010
    }
4cc3010
    undef;
4cc3010
}
4cc3010
4cc3010
# Subroutines
4cc3010
4cc3010
sub parse {
4cc3010
   local $intag = 0;
4cc3010
   my (@node_stack, $parent);
4cc3010
   $parent = @_[1];
4cc3010
   local ($tag, $text);
4cc3010
4cc3010
   while (<>) {
4cc3010
      chomp;
4cc3010
      @str = split /([<>])/;
4cc3010
      shift @str if ($str[0] eq '' || $str[0] =~ /^[ \t]*$/);
4cc3010
4cc3010
      while (scalar @str) {
4cc3010
         $token = shift @str;
4cc3010
         if ($token eq '<') {
4cc3010
            $intag = 1;
4cc3010
            if (defined $text) {
4cc3010
               add_text_node($parent, $text);
4cc3010
               undef $text;
4cc3010
            }
4cc3010
         } elsif ($token eq '>') {
4cc3010
            $intag = 0;
4cc3010
            if ($tag =~ /^\/(.*)/) { # close tag
4cc3010
               $parent = pop @node_stack;
4cc3010
            } elsif ($tag =~ /^([^\/]*)\/$/) {
4cc3010
               empty_tag($parent, $1);
4cc3010
            } else {
4cc3010
               if (defined ($node = open_tag($parent, $tag))) {
4cc3010
                  push @node_stack, $parent;
4cc3010
                  $parent = $node;
4cc3010
               }
4cc3010
            }
4cc3010
            undef $tag;
4cc3010
         } else {
4cc3010
            if ($intag == 1) {
4cc3010
               if (defined $tag) {
4cc3010
                  $tag .= ' '. $token;
4cc3010
               } else {
4cc3010
                  $tag = $token;
4cc3010
               }
4cc3010
            } else {
4cc3010
               if (defined $text) {
4cc3010
                  $text .= "\n" . $token;
4cc3010
               } else {
4cc3010
                  $text = $token;
4cc3010
               }
4cc3010
            }
4cc3010
         }
4cc3010
      }
4cc3010
   }
4cc3010
}
4cc3010
4cc3010
sub new_document {
4cc3010
   $doc = new_node( 0, '', 'DOCUMENT');
4cc3010
   $doc->{CHILDREN} = [];
4cc3010
   return $doc;
4cc3010
}
4cc3010
4cc3010
sub new_node {
4cc3010
  local ($parent_node, $tag, $type) = @_;
4cc3010
4cc3010
  my %node;
4cc3010
  $node{PARENT} = $parent_node;
4cc3010
  $node{TYPE} = $type;
4cc3010
4cc3010
  if ($type eq 'COMMENT' || $type eq 'TEXT') {
4cc3010
     $node{TEXT} = $tag;
4cc3010
     $node{NAME} = $type;
4cc3010
     return \%node;
4cc3010
  }
4cc3010
4cc3010
  local ($tname, $attr) = split(' ', $tag, 2);
4cc3010
  $node{NAME} = $tname;
4cc3010
4cc3010
  if (defined $attr && $attr ne '') {
4cc3010
     my %attr_table;
4cc3010
     local @attr_list = split ( /"/, $attr);
4cc3010
     local ($name, $value);
4cc3010
     while (scalar @attr_list) {
4cc3010
        $name = shift @attr_list;
4cc3010
        $name =~ s/[ =]//g;
4cc3010
        next if ($name eq '');
4cc3010
        $value =  shift @attr_list;
4cc3010
        $attr_table{$name} =$value;
4cc3010
     }
4cc3010
     $node{ATTRIBUTES} = \%attr_table;
4cc3010
  }
4cc3010
  return \%node;
4cc3010
}
4cc3010
4cc3010
sub add_node {
4cc3010
  local ($parent_node, $node) = @_;
4cc3010
  push @{$parent_node->{CHILDREN}}, $node;
4cc3010
4cc3010
  local $tname = $node->{NAME};
4cc3010
  if (defined $parent_node->{$tname}) {
4cc3010
      push @{$parent_node->{$tname}}, $node
4cc3010
  } else {
4cc3010
      $parent_node->{$tname} = [ $node ];
4cc3010
  }
4cc3010
}
4cc3010
4cc3010
sub empty_tag {
4cc3010
   local ($parent_node, $tag) = @_;
4cc3010
   local $node = new_node($parent_node, $tag, 'EMPTY');
4cc3010
   add_node($parent_node, $node);
4cc3010
}
4cc3010
4cc3010
sub open_tag {
4cc3010
   local ($parent_node, $tag) = @_;
4cc3010
   local $node;
4cc3010
4cc3010
   if ($tag =~ /^\?.*/ || $tag =~ /^\!.*/) {
4cc3010
      $node = new_node($parent_node, $tag, 'COMMENT');
4cc3010
      add_node($parent_node, $node);
4cc3010
      undef; return;
4cc3010
   } else {
4cc3010
      $node = new_node($parent_node, $tag, 'NODE');
4cc3010
      $node->{CHILDREN} = [];
4cc3010
      add_node($parent_node, $node);
4cc3010
      return $node;
4cc3010
   }
4cc3010
}
4cc3010
4cc3010
sub add_text_node {
4cc3010
   local ($parent_node, $text) = @_;
4cc3010
   local $node = new_node($parent_node, $text, 'TEXT');
4cc3010
   add_node($parent_node, $node);
4cc3010
}
4cc3010
4cc3010
sub node_by_name {
4cc3010
   local ($node, $name) = @_;
4cc3010
   local ($tagname, $path) = split(/\//, $name, 2);
4cc3010
4cc3010
   my @nodelist;
4cc3010
4cc3010
   if ($tagname eq '') {
4cc3010
      while ($node->{PARENT} != 0) {
4cc3010
         $node = $node->{PARENT};
4cc3010
      }
4cc3010
      sublist_by_name($node, $path, \@nodelist);
4cc3010
   } else {
4cc3010
      sublist_by_name($node, $name, \@nodelist);
4cc3010
   }
4cc3010
   return @nodelist;
4cc3010
}
4cc3010
4cc3010
sub sublist_by_name {
4cc3010
   local ($node, $name, $res) = @_;
4cc3010
   local ($tagname, $path) = split(/\//, $name, 2);
4cc3010
4cc3010
   if (! defined $path) {
4cc3010
       push @{$res}, (@{$node->{$tagname}});
4cc3010
       return;
4cc3010
   }
4cc3010
4cc3010
   if ($tagname eq '..' && $node->{PARENT} != 0) {
4cc3010
      $node = $node->{PARENT};
4cc3010
      sublist_by_name($node, $path, $res);
4cc3010
   } else {
4cc3010
      local $n;
4cc3010
      for $n (@{$node->{$tagname}}) {
4cc3010
         sublist_by_name($n, $path, $res);
4cc3010
      }
4cc3010
   }
4cc3010
}
4cc3010
4cc3010
sub node_attribute {
4cc3010
    local $node = @_[0];
4cc3010
    if (defined $node->{ATTRIBUTES}) {
4cc3010
       return $node->{ATTRIBUTES}{@_[1]};
4cc3010
    }
4cc3010
    undef;
4cc3010
}
4cc3010
4cc3010
sub text_child {
4cc3010
    local ($node) = @_;
4cc3010
    local ($child) = node_by_name($node, 'TEXT');
4cc3010
    return $child->{TEXT};
4cc3010
}