Annotation of OSAS/OSA_system/OSAS_addfiles.py, revision 1.25

1.1       dwinter     1: # Methoden und Classen zum Hinzufuegen von Dokumenten
                      2: 
1.18      dwinter     3: 
                      4: from OSAS_helpers import readArchimedesXML, getISO, getText
                      5: import os
1.7       dwinter     6: import xml.dom.minidom
1.24      dwinter     7: import re
1.3       dwinter     8: import urllib
1.1       dwinter     9: import OSAS_add
1.9       dwinter    10: import OSAS_show
1.5       dwinter    11: import string
1.1       dwinter    12: from OFS.Folder import Folder
                     13: from AccessControl import ClassSecurityInfo
                     14: from Globals import InitializeClass
                     15: from Products.PageTemplates.PageTemplateFile import PageTemplateFile
                     16: from Products.PageTemplates.PageTemplate import PageTemplate
1.18      dwinter    17: import archive # check if this is necessary
1.1       dwinter    18: 
1.21      dwinter    19: 
                     20: from xml.sax import make_parser
                     21: from xml.sax.handler import ContentHandler
                     22: 
                     23: def spaces(depth):
                     24:     """needed in XMLtoTree"""
                     25:     tmp=""
                     26:     k=0
                     27:     while k<2*depth:
                     28:         k+=1
                     29:         tmp=tmp+"&nbsp;"+"&nbsp;"
                     30:     return tmp
                     31: 
                     32: 
1.1       dwinter    33: class OSAS_add_Document(Folder):
                     34:     """Hinzufuegen eines Dokumentes zum Storage"""
                     35:     security=ClassSecurityInfo()
                     36: 
                     37:     def __init__(self,id):
                     38:         """initialize a new instance"""
                     39:         self.id = id
                     40:         
                     41:         
                     42:     meta_type='OSAS_add_Document'    
                     43:     manage_options = Folder.manage_options+(
                     44:             {'label':'Main Config','action':'add_Document_config'},
                     45:             )
                     46:         
                     47:      
                     48:         
                     49:     security.declarePublic('add_Document_config')
                     50:     def add_Document_config(self):
                     51:         """Main configuration"""
1.17      dwinter    52:         pt=PageTemplateFile('Products/OSA_system/zpt/ChangeOSAS_addDocument.zpt').__of__(self)
1.1       dwinter    53:         return pt()
                     54:     
                     55:     security.declarePublic('change_OSAS_add_Document')
                     56:     def change_OSAS_add_Document(self,RESPONSE=None):
                     57:         """Change"""
                     58: #        self.RootFolderName=RootFolderName
                     59:         if RESPONSE is not None:
                     60:             RESPONSE.redirect('manage_main')
                     61: 
                     62:     security.declarePublic('index_html')
                     63:     def index_html(self):
                     64:         """stantard aufruf"""
1.24      dwinter    65:         return OSAS_add.add(self.standardMD,no_upload=1)
1.1       dwinter    66: 
                     67: 
                     68:     security.declarePublic('add2')
                     69:     def add2(self):
                     70:         """ anlegen naechster schritt"""
1.24      dwinter    71:         return OSAS_add.add2(self.standardMD)
1.1       dwinter    72: 
                     73:     def add3(self):
                     74:         """Foldername"""
                     75:         return OSAS_add.add3(self)
                     76: 
                     77:     def add4(self):
                     78:         """Applet"""
                     79:         return OSAS_add.add4(self)
                     80: 
                     81:     def add5(self):
                     82:         """Foldername"""
                     83:         return OSAS_add.add5(self)
                     84: 
                     85:     def add6(self):
                     86:         """write new index.meta file"""
                     87:         return OSAS_add.add6(self)
                     88: 
                     89: 
                     90:     def addImages(self,path):
                     91:         """Hinzufügen eines neuen Imagesfolders"""
                     92:         return OSAS_add.addImages(self,path)
                     93: 
                     94:     def addImages2(self):
                     95:         """Upload des neuen Imagefolders"""
1.2       dwinter    96:         return OSAS_add.addImages2(self)
                     97: 
1.1       dwinter    98:     
                     99: def manage_AddOSAS_add_DocumentForm(self):
                    100:     """interface for adding the OSAS_add_Metadata"""
1.17      dwinter   101:     pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_document.zpt').__of__(self)
1.1       dwinter   102:     return pt()
                    103: 
                    104: def manage_AddOSAS_add_Document(self,id,RESPONSE=None):
                    105:     """add the OSAS_root"""
                    106:     newObj=OSAS_add_Document(id)
                    107:     self.Destination()._setObject(id,newObj)
                    108:     if RESPONSE is not None:
                    109:         RESPONSE.redirect('manage_main')
                    110: 
                    111:             
                    112: InitializeClass(OSAS_add_Document)
                    113: 
1.2       dwinter   114: 
                    115: class OSAS_add_Text(Folder):
                    116:     """Hinzufuegen eines Text-Dokumentes zum Storage"""
                    117:     security=ClassSecurityInfo()
                    118: 
                    119:     def __init__(self,id):
                    120:         """initialize a new instance"""
                    121:         self.id = id
                    122:         
                    123:         
                    124:     meta_type='OSAS_add_Text'    
                    125:     manage_options = Folder.manage_options+(
                    126:             {'label':'Main Config','action':'add_Text_config'},
                    127:             )
                    128:         
                    129:      
                    130:         
                    131:     security.declarePublic('add_Text_config')
                    132:     def add_Text_config(self):
                    133:         """Main configuration"""
1.17      dwinter   134:         pt=PageTemplateFile('Products/OSA_system/zpt/ChangeOSAS_addText.zpt').__of__(self)
1.2       dwinter   135:         return pt()
                    136:     
                    137:     security.declarePublic('change_OSAS_add_Text')
                    138:     def change_OSAS_add_Text(self,RESPONSE=None):
                    139:         """Change"""
                    140: #        self.RootFolderName=RootFolderName
                    141:         if RESPONSE is not None:
                    142:             RESPONSE.redirect('manage_main')
                    143: 
                    144:     def addText(self,path):
                    145:         """Add a fulltext"""
                    146:         return OSAS_add.addText(self,path)
                    147: 
                    148:     def addText2(self):
                    149:         """Read the file and store it"""
                    150:         return OSAS_add.addText2(self)
                    151: def manage_AddOSAS_add_TextForm(self):
                    152:     """interface for adding the OSAS_add_Metadata"""
1.17      dwinter   153:     pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_text.zpt').__of__(self)
1.2       dwinter   154:     return pt()
                    155: 
                    156: def manage_AddOSAS_add_Text(self,id,RESPONSE=None):
                    157:     """add the OSAS_root"""
                    158:     newObj=OSAS_add_Text(id)
                    159:     self.Destination()._setObject(id,newObj)
                    160:     if RESPONSE is not None:
                    161:         RESPONSE.redirect('manage_main')
                    162: 
                    163:             
                    164: InitializeClass(OSAS_add_Text)
1.3       dwinter   165: 
1.21      dwinter   166: 
1.3       dwinter   167: class OSAS_add_contextData(Folder):
                    168:     """Einfuegen eines Documentes in eine Collection"""
                    169:     
                    170:     security=ClassSecurityInfo()
                    171: 
                    172:             
                    173:     meta_type='OSAS_add_contextData'    
                    174: 
1.21      dwinter   175:     def XmlToTree(self,URL):
                    176:         """Collection XML to Tree"""
                    177:         
                    178:         
                    179:         class requestHandler(ContentHandler):
                    180:             def __init__(self):
                    181:                 self.depth=0
                    182:                 self.retStr=""
                    183:                 
                    184:             def startElement(self,name,attrs):
                    185:                 if name=="element":
                    186:                     self.depth+=1
1.24      dwinter   187:                     begin=""
                    188:                     end=""
                    189:                     if self.depth==1:
                    190:                         print "hi"
                    191:                         begin="<b>"
                    192:                         end="</b>"
                    193:                         
                    194:                     self.retStr+=spaces(self.depth)+"<input type='radio' name='collection' value='%s'>%s</input>"%(attrs.get('url'),begin+attrs.get('name')+end)+"<br>\n"
1.21      dwinter   195: 
                    196: 
                    197:                     
                    198:             def endElement(self,name):
                    199:                 if name=="element":
                    200:                     self.depth-=1
                    201: 
                    202: 
1.23      dwinter   203:         try:
                    204:             URL+="/getCollectionTreeXML"
                    205:             parser=make_parser()
                    206:             curHandler=requestHandler()
                    207:             parser.setContentHandler(curHandler)
                    208: 
                    209:             parser.parse(urllib.urlopen(URL))
                    210:             return curHandler.retStr
                    211:         except:
                    212:             return urllib.urlopen(URL).read()
                    213:         
1.3       dwinter   214:     def __init__(self,id,collection):
                    215:         self.id=id
                    216:         self.collection=collection
                    217: 
1.23      dwinter   218:   
                    219:    
                    220: 
                    221:     def getPartners(self,URL):
                    222:         """Zeige Partnerliste"""
                    223:         class requestHandler(ContentHandler):
                    224:             def __init__(self):
                    225:                 self.ret=[]
1.5       dwinter   226:                 
1.23      dwinter   227:             def startElement(self,name,attrs):
                    228:                 if name=="partner":
                    229:                     self.ret.append((attrs.get('id'),attrs.get('title')))
                    230: 
                    231: 
                    232:         URL+="/getPartnersXML"
                    233:             
                    234:         try:
                    235:             
                    236:             parser=make_parser()
                    237:             curHandler=requestHandler()
                    238:             parser.setContentHandler(curHandler)
                    239:             
                    240:             parser.parse(urllib.urlopen(URL))
                    241:             return curHandler.ret
                    242:         except:
                    243:             return [("",urllib.urlopen(URL).read())]
1.21      dwinter   244: 
                    245:     
1.5       dwinter   246:     def addContextData(self,path):
                    247:         """Hinzufügen zu einer Sammlung"""
1.6       dwinter   248:         try:
                    249:             urllib.urlopen(self.REQUEST['SERVER_URL']+path+"/index.meta")
1.9       dwinter   250:             
1.6       dwinter   251:         except:
1.14      dwinter   252:             return self.REQUEST['SERVER_URL']+path+"/index.meta file has to exist!"
1.3       dwinter   253: 
1.10      dwinter   254:         links=[(path,'standard storage')]
                    255:         
                    256:         links+=OSAS_show.readContexts(path) # auslesen von contexten für den link
                    257:         #print "LINK",links
1.15      dwinter   258:         #return links
1.9       dwinter   259:         self.REQUEST.SESSION['links']=links
1.17      dwinter   260:         pt=PageTemplateFile('Products/OSA_system/zpt/contextDataMain.zpt').__of__(self)
1.9       dwinter   261:         return pt()
1.5       dwinter   262:     
                    263:     
1.25    ! dwinter   264:     def addContextData2(self,path,collection,link,label,description,content_type,responsible,weight,credits=None):
1.5       dwinter   265:         """Hinzufuegen der Resource"""
                    266:         splitted=path.split("/")
1.13      dwinter   267:         #print "BLU"
1.5       dwinter   268:         id=splitted[len(splitted)-1]
                    269:         title=splitted[len(splitted)-1]
1.6       dwinter   270:         metalink=self.REQUEST['SERVER_URL']+path+"/index.meta"
1.13      dwinter   271:         
1.9       dwinter   272:         #link=TOBEDONE"
1.5       dwinter   273:         """Hinzufügen der Ressource"""
1.3       dwinter   274: 
1.21      dwinter   275:         params=urllib.urlencode({'id':id,'title':title,'link':link,'label':label,'description':description,'content_type':content_type,'responsible':responsible,'weight':weight,'credits':credits,'metalink':metalink})
                    276: 
                    277:         retStr=urllib.urlopen(collection+"/addResource",params).read()
                    278:         if not retStr:
                    279:             return "An Error occured adding the resource\n"
                    280: 
                    281:         urllib.urlopen(collection+"/"+id+"/ECHO_getResourceMD").read()
                    282:         #exec("collection_object=self"+collection)
1.5       dwinter   283:         
1.21      dwinter   284: 
1.5       dwinter   285:         
1.21      dwinter   286:         #ECHO_collection.manage_addECHO_resource(collection_object,id,title,label,description,content_type,responsible,link,metalink,credits,weight,RESPONSE=None)
1.13      dwinter   287:         #print "HI5"
1.21      dwinter   288:         #try:
                    289:         #    getattr(collection_object,id).ECHO_getResourceMD()
                    290:         #except:
                    291:         #    """nothing"""
1.12      dwinter   292:         #return "BLUByy"
1.5       dwinter   293:         return self.REQUEST.RESPONSE.redirect(self.REQUEST['URL2']+'?path='+path)
                    294:         
1.3       dwinter   295:     manage_options = Folder.manage_options+(
                    296:             {'label':'Main Config','action':'add_contextData_config'},
                    297:             )
                    298:         
                    299:      
                    300:     def add_contextData_config(self):
                    301:         """Main configuration"""
1.17      dwinter   302:         pt=PageTemplateFile('Products/OSA_system/zpt/ChangeOSAS_add_contextData.zpt').__of__(self)
1.3       dwinter   303:         return pt()
                    304:     
                    305:     
                    306:     def change_OSAS_add_contextData(self,collection,RESPONSE=None):
                    307:         """Change"""
                    308:         self.collection=collection
                    309:         if RESPONSE is not None:
                    310:             RESPONSE.redirect('manage_main')
                    311:             
                    312: def manage_AddOSAS_add_contextDataForm(self):
                    313:     """interface for adding the OSAS_add_Metadata"""
1.17      dwinter   314:     pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_contextData.zpt').__of__(self)
1.3       dwinter   315:     return pt()
                    316: 
                    317: def manage_AddOSAS_add_contextData(self,id,collection,RESPONSE=None):
                    318:     """add the OSAS_root"""
                    319:     newObj=OSAS_add_contextData(id,collection)
                    320:     self.Destination()._setObject(id,newObj)
                    321:     if RESPONSE is not None:
                    322:         RESPONSE.redirect('manage_main')
                    323: 
                    324:             
                    325: InitializeClass(OSAS_add_contextData)
1.2       dwinter   326: 
                    327: class OSAS_add_Presentation(Folder):
                    328:     """Hinzufügen der Presentationsinformationen"""
                    329:     security=ClassSecurityInfo()
                    330: 
                    331:     def __init__(self,id):
                    332:         """initialize a new instance"""
                    333:         self.id = id
                    334:         
                    335:         
                    336:     meta_type='OSAS_add_Presentation'    
                    337:     manage_options = Folder.manage_options+(
                    338:             {'label':'Main Config','action':'add_Presentation_config'},
                    339:             )
                    340:         
                    341:      
                    342:         
                    343:     security.declarePublic('add_Presentation_config')
                    344:     def add_Presentation_config(self):
                    345:         """Main configuration"""
1.17      dwinter   346:         pt=PageTemplateFile('Products/OSA_system/zpt/ChangeOSAS_addPresentation.zpt').__of__(self)
1.2       dwinter   347:         return pt()
                    348:     
                    349:     security.declarePublic('change_OSAS_add_Presentation')
                    350:     def change_OSAS_add_Presentation(self,RESPONSE=None):
                    351:         """Change"""
                    352: #        self.RootFolderName=RootFolderName
                    353:         if RESPONSE is not None:
                    354:             RESPONSE.redirect('manage_main')
                    355: 
                    356:     def addPresentation(self,path):
                    357:         """Hinzufügen der Presenationsinformation"""
                    358:         return OSAS_add.addPresentation(self,path)
                    359: 
                    360:     def addPresentation2(self):
                    361:         """Eingabe von Metadateninformationen"""
                    362:         return OSAS_add.addPresentation2(self)
                    363:     
                    364: def manage_AddOSAS_add_PresentationForm(self):
                    365:     """interface for adding the OSAS_add_Metadata"""
1.17      dwinter   366:     pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_presentation.zpt').__of__(self)
1.2       dwinter   367:     return pt()
                    368: 
                    369: def manage_AddOSAS_add_Presentation(self,id,RESPONSE=None):
                    370:     """add the OSAS_root"""
                    371:     newObj=OSAS_add_Presentation(id)
                    372:     self.Destination()._setObject(id,newObj)
                    373:     if RESPONSE is not None:
                    374:         RESPONSE.redirect('manage_main')
                    375: 
                    376:             
                    377: InitializeClass(OSAS_add_Presentation)
                    378: 
                    379: class OSAS_combineTextImage(Folder):
1.18      dwinter   380:     """Hinzufüge der Combine Text und Image"""
1.2       dwinter   381:     security=ClassSecurityInfo()
                    382: 
                    383:     def __init__(self,id):
                    384:         """initialize a new instance"""
                    385:         self.id = id
                    386:         
                    387:         
                    388:     meta_type='OSAS_combineTextImage'    
                    389:     manage_options = Folder.manage_options+(
                    390:             {'label':'Main Config','action':'combineTextImage_config'},
                    391:             )
                    392:         
1.15      dwinter   393: 
1.2       dwinter   394:         
                    395:     security.declarePublic('combineTextImage_config')
                    396:     def combineTextImage_config(self):
                    397:         """Main configuration"""
1.17      dwinter   398:         pt=PageTemplateFile('Products/OSA_system/zpt/ChangeOSAS_combineTextImage.zpt').__of__(self)
1.2       dwinter   399:         return pt()
                    400:     
                    401:     security.declarePublic('change_OSAS_combineTextImage')
                    402:     def change_OSAS_combineTextImage(self,RESPONSE=None):
                    403:         """Change"""
                    404: #        self.RootFolderName=RootFolderName
                    405:         if RESPONSE is not None:
                    406:             RESPONSE.redirect('manage_main')
                    407:             
                    408:     security.declarePublic('combineTextImage')
                    409:     def combineTextImage(self,path):
                    410:         """Hinzufügen der Presenationsinformation"""
1.18      dwinter   411:         """gibt input formular zur erstellung des texttools meta tag aus"""
                    412:    files = os.listdir(path)
                    413:    
                    414:    texts=[]
                    415:    imagefolders=[]
                    416:    presentationfolders=[]
                    417: 
                    418:    splitted=path.split("/")
                    419:    externxml=readArchimedesXML(splitted[len(splitted)-1])
                    420:    
                    421:    for filename in files:
                    422:        #print "FN",filename
                    423:        if archive.isdigilib2(path+"/"+filename):
                    424:            imagefolders.append(filename)
                    425:            
                    426:        if archive.isFullText(path,filename):
                    427:            #print "HI"
                    428:            texts.append(filename)
                    429:        if archive.isPresentation(path,filename):
                    430:            presentationfolders.append(filename)
                    431:    
                    432:    dom=xml.dom.minidom.parse(path+"/index.meta")
                    433:    try:
                    434:        filelanguage=archive.getText(dom.getElementsByTagName('lang')[0].childNodes)
                    435:    except:
                    436:        filelanguage=""
                    437:                 
                    438:    self.REQUEST.SESSION['isolist']=getISO()
                    439:    tmp=getISO().keys()
                    440:    tmp.sort()
                    441:    self.REQUEST.SESSION['isolistsort']=tmp
                    442:    self.REQUEST.SESSION['path']=path
                    443:    self.REQUEST.SESSION['texts']=texts
                    444:    self.REQUEST.SESSION['imagefolders']=imagefolders
                    445:    self.REQUEST.SESSION['presentationfolders']=presentationfolders
                    446:    self.REQUEST.SESSION['filelanguage']=filelanguage
                    447:    self.REQUEST.SESSION['externxml']=externxml
                    448: 
                    449:    newtemplate=PageTemplateFile('Products/OSA_system/zpt/ImageandText').__of__(self)
                    450:    return newtemplate()
                    451:    
                    452:     def getProjects(self,obj_ids=None):
                    453:         """Get the Project title for configuration"""
                    454:         ret=[]
1.19      dwinter   455:         print "HI"
1.18      dwinter   456:         try:
1.19      dwinter   457:             projects=self.ZopeFind(self.projects,obj_metatypes=['OSAS_project'],obj_ids=obj_ids)#assumes projects folder somewhere in the hierarchie.
                    458:             print "pr"
1.18      dwinter   459:             for project in projects:
                    460:                 ret.append((project[1].title,project[0],project[1]))
                    461:             print ret
                    462:             return ret
                    463:         
                    464:         except:
                    465:             return [('no Projectfolders','')]
                    466: 
                    467:     def getTextToolsField(self,path,name,default=''):
                    468:         """Lese Textoolsfelder aus index.meta im path aus"""
                    469:         
                    470:         try:
                    471:             dom=xml.dom.minidom.parse(path+"/index.meta")
                    472:             node=dom.getElementsByTagName('texttool')[0] #getNode
                    473:             subnode=node.getElementsByTagName(name)[0]
                    474:             print getText(subnode.childNodes)
                    475:             return getText(subnode.childNodes)
                    476:         except:
                    477:             return default
                    478:             
1.2       dwinter   479: 
                    480:     def combineTextImage2(self,path):
                    481:         """Eingabe von Metadateninformationen"""
1.7       dwinter   482:         OSAS_add.combineTextImage2(self,path) # Add images
                    483:         splitted=path.split("/")
                    484:         linkPath=splitted[len(splitted)-1]
1.22      dwinter   485:         linkViewerEnvironment="http://nausikaa2.mpiwg-berlin.mpg.de/cgi-bin/toc/toc.x.cgi?dir=%s&step=thumb" % linkPath
1.7       dwinter   486:         self.REQUEST.SESSION['linkViewerEnvironment']=linkViewerEnvironment
                    487: 
1.24      dwinter   488:         writeToContext(path,linkViewerEnvironment,"ECHO standard environment",unique="yes")
1.7       dwinter   489:         
1.17      dwinter   490:         pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_combineTextImageFinal.zpt').__of__(self)
1.7       dwinter   491: 
                    492:         return pt()
                    493: 
1.18      dwinter   494:     def isSelectedProject(self,obj,id):
                    495:         """is ausgewählt"""
                    496:         
                    497:         if self.REQUEST['project']==id:
                    498:             return 1
                    499:         else:
                    500:             return None
1.7       dwinter   501: 
1.24      dwinter   502: def writeToContext(path,link,description,unique="no"):
1.7       dwinter   503:     """Created an additional entry to the index.meta file of path"""
                    504:     dom=xml.dom.minidom.parse(path+"/index.meta")
                    505:     node=dom.getElementsByTagName('resource')[0]
                    506: 
1.24      dwinter   507:     if unique=="yes":
                    508:         
                    509:         contexts=node.getElementsByTagName('context')
                    510:         for context in contexts:
                    511:             nameTag=getText(context.getElementsByTagName('name')[0].childNodes)
                    512:             linkTag=getText(context.getElementsByTagName('link')[0].childNodes)
                    513:             
                    514:             print "unique",context,nameTag,description,linkTag,link
                    515:             linkTag=re.sub("\:86","",linkTag) # alter port 86 gleich ohne port nummer (hack)
                    516:             if (nameTag==description) and (linkTag==link):
                    517:                 node.removeChild(context).unlink()
                    518:                     
1.7       dwinter   519:     subnode=dom.createElement('context')
                    520: 
                    521:     linknode=dom.createElement('link')
                    522:     namelinknode=dom.createTextNode(link)
                    523:     linknode.appendChild(namelinknode)
                    524:     subnode.appendChild(linknode)
                    525: 
                    526:     linknode=dom.createElement('name')
                    527:     namelinknode=dom.createTextNode(description)
                    528:     linknode.appendChild(namelinknode)
                    529:     subnode.appendChild(linknode)
                    530: 
                    531:     node.appendChild(subnode)
                    532:     
                    533:     writefile=file(path+"/index.meta","w")
                    534:     #print path+"/index.meta"
                    535:     writefile.write(dom.toxml().encode('utf-8'))
                    536:     writefile.close()
1.2       dwinter   537: 
                    538: def manage_AddOSAS_combineTextImageForm(self):
                    539:     """interface for adding the OSAS_add_Metadata"""
1.17      dwinter   540:     pt=PageTemplateFile('Products/OSA_system/zpt/AddOSAS_combineTextImage.zpt').__of__(self)
1.2       dwinter   541:     return pt()
                    542: 
                    543: def manage_AddOSAS_combineTextImage(self,id,RESPONSE=None):
                    544:     """add the OSAS_root"""
                    545:     newObj=OSAS_combineTextImage(id)
                    546:     self.Destination()._setObject(id,newObj)
                    547:     if RESPONSE is not None:
                    548:         RESPONSE.redirect('manage_main')
                    549: 
                    550:             
                    551: InitializeClass(OSAS_combineTextImage)

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>