Conflicts: starmath/inc/applicat.hxx starmath/inc/dialog.hxx starmath/inc/document.hxx starmath/inc/edit.hxx starmath/inc/format.hxx starmath/inc/node.hxx starmath/inc/parse.hxx starmath/inc/rect.hxx starmath/inc/smdll.hxx starmath/inc/smmod.hxx starmath/inc/starmath.hrc starmath/inc/symbol.hxx starmath/inc/toolbox.hxx starmath/inc/utility.hxx starmath/inc/view.hxx starmath/prj/build.lst starmath/qa/cppunit/version.map starmath/sdi/smslots.sdi starmath/source/accessibility.cxx starmath/source/cfgitem.cxx starmath/source/cfgitem.hxx starmath/source/config.cxx starmath/source/dialog.cxx starmath/source/document.cxx starmath/source/edit.cxx starmath/source/format.cxx starmath/source/makefile.mk starmath/source/math_pch.cxx starmath/source/mathmlexport.cxx starmath/source/mathmlimport.cxx starmath/source/mathtype.cxx starmath/source/node.cxx starmath/source/parse.cxx starmath/source/rect.cxx starmath/source/register.cxx starmath/source/smdetect.cxx starmath/source/smdll.cxx starmath/source/smmod.cxx starmath/source/smres.src starmath/source/symbol.cxx starmath/source/toolbox.cxx starmath/source/unomodel.cxx starmath/source/utility.cxx starmath/source/view.cxx sw/JunitTest_sw_unoapi.mk sw/Library_swd.mk sw/Makefile sw/inc/IDocumentFieldsAccess.hxx sw/inc/IDocumentSettingAccess.hxx sw/inc/IDocumentUndoRedo.hxx sw/inc/IShellCursorSupplier.hxx sw/inc/SwUndoField.hxx sw/inc/acmplwrd.hxx sw/inc/authfld.hxx sw/inc/bparr.hxx sw/inc/calbck.hxx sw/inc/calc.hxx sw/inc/ccoll.hxx sw/inc/cellatr.hxx sw/inc/cellfml.hxx sw/inc/chpfld.hxx sw/inc/cmdid.h sw/inc/crsrsh.hxx sw/inc/crstate.hxx sw/inc/dbfld.hxx sw/inc/dbmgr.hxx sw/inc/dcontact.hxx sw/inc/ddefld.hxx sw/inc/doc.hxx sw/inc/docary.hxx sw/inc/docsh.hxx sw/inc/docstat.hxx sw/inc/docstyle.hxx sw/inc/docufld.hxx sw/inc/editsh.hxx sw/inc/errhdl.hxx sw/inc/expfld.hxx sw/inc/fchrfmt.hxx sw/inc/fesh.hxx sw/inc/fldbas.hxx sw/inc/flddat.hxx sw/inc/flddropdown.hxx sw/inc/flypos.hxx sw/inc/fmtanchr.hxx sw/inc/fmtautofmt.hxx sw/inc/fmtclds.hxx sw/inc/fmtcnct.hxx sw/inc/fmtcol.hxx sw/inc/fmtfsize.hxx sw/inc/fmtftn.hxx sw/inc/fmtftntx.hxx sw/inc/fmthdft.hxx sw/inc/fmtinfmt.hxx sw/inc/fmtline.hxx sw/inc/fmtornt.hxx sw/inc/fmtpdsc.hxx sw/inc/fmtruby.hxx sw/inc/fmtsrnd.hxx sw/inc/fmturl.hxx sw/inc/fmtwrapinfluenceonobjpos.hxx sw/inc/format.hxx sw/inc/frmatr.hxx sw/inc/frmfmt.hxx sw/inc/grfatr.hxx sw/inc/helpid.h sw/inc/hintids.hxx sw/inc/hints.hxx sw/inc/htmltbl.hxx sw/inc/inetfld.hxx sw/inc/io.hxx sw/inc/iodetect.hxx sw/inc/itabenum.hxx sw/inc/ndarr.hxx sw/inc/ndgrf.hxx sw/inc/ndindex.hxx sw/inc/ndnotxt.hxx sw/inc/ndole.hxx sw/inc/ndtxt.hxx sw/inc/ndtyp.hxx sw/inc/node.hxx sw/inc/numrule.hxx sw/inc/pagedesc.hxx sw/inc/pagepreviewlayout.hxx sw/inc/pam.hxx sw/inc/paratr.hxx sw/inc/poolfmt.awk sw/inc/poolfmt.hxx sw/inc/printdata.hxx sw/inc/reffld.hxx sw/inc/shellio.hxx sw/inc/shellres.hxx sw/inc/swabstdlg.hxx sw/inc/swatrset.hxx sw/inc/swerror.h sw/inc/swprtopt.hxx sw/inc/swtable.hxx sw/inc/swtypes.hxx sw/inc/tblafmt.hxx sw/inc/tgrditem.hxx sw/inc/tox.hxx sw/inc/undobj.hxx sw/inc/unocoll.hxx sw/inc/unoframe.hxx sw/inc/unoprnms.hxx sw/inc/usrfld.hxx sw/inc/viewopt.hxx sw/inc/viewsh.hxx sw/inc/viscrs.hxx sw/prj/build.lst sw/qa/complex/accessibility/makefile.mk sw/qa/core/Test-BigPtrArray.cxx sw/qa/core/makefile.mk sw/sdi/makefile.mk sw/source/core/access/makefile.mk sw/source/core/access/textmarkuphelper.cxx sw/source/core/attr/calbck.cxx sw/source/core/attr/cellatr.cxx sw/source/core/attr/fmtwrapinfluenceonobjpos.cxx sw/source/core/attr/format.cxx sw/source/core/attr/hints.cxx sw/source/core/bastyp/calc.cxx sw/source/core/bastyp/init.cxx sw/source/core/bastyp/makefile.mk sw/source/core/bastyp/swcache.cxx sw/source/core/crsr/bookmrk.cxx sw/source/core/crsr/callnk.cxx sw/source/core/crsr/crsrsh.cxx sw/source/core/crsr/crstrvl.cxx sw/source/core/crsr/findattr.cxx sw/source/core/crsr/findcoll.cxx sw/source/core/crsr/makefile.mk sw/source/core/crsr/pam.cxx sw/source/core/crsr/swcrsr.cxx sw/source/core/crsr/trvltbl.cxx sw/source/core/crsr/unocrsr.cxx sw/source/core/crsr/viscrs.cxx sw/source/core/doc/acmplwrd.cxx sw/source/core/doc/doc.cxx sw/source/core/doc/docbm.cxx sw/source/core/doc/doccomp.cxx sw/source/core/doc/docdesc.cxx sw/source/core/doc/docdraw.cxx sw/source/core/doc/docedt.cxx sw/source/core/doc/docfld.cxx sw/source/core/doc/docfly.cxx sw/source/core/doc/docfmt.cxx sw/source/core/doc/docftn.cxx sw/source/core/doc/docglbl.cxx sw/source/core/doc/docglos.cxx sw/source/core/doc/doclay.cxx sw/source/core/doc/docnew.cxx sw/source/core/doc/docnum.cxx sw/source/core/doc/docredln.cxx sw/source/core/doc/docruby.cxx sw/source/core/doc/docsort.cxx sw/source/core/doc/docstat.cxx sw/source/core/doc/doctxm.cxx sw/source/core/doc/fmtcol.cxx sw/source/core/doc/gctable.cxx sw/source/core/doc/htmltbl.cxx sw/source/core/doc/makefile.mk sw/source/core/doc/number.cxx sw/source/core/doc/poolfmt.cxx sw/source/core/doc/tblafmt.cxx sw/source/core/doc/tblcpy.cxx sw/source/core/doc/tblrwcl.cxx sw/source/core/docnode/makefile.mk sw/source/core/docnode/ndcopy.cxx sw/source/core/docnode/ndnum.cxx sw/source/core/docnode/ndsect.cxx sw/source/core/docnode/ndtbl.cxx sw/source/core/docnode/ndtbl1.cxx sw/source/core/docnode/node.cxx sw/source/core/docnode/node2lay.cxx sw/source/core/docnode/nodes.cxx sw/source/core/docnode/section.cxx sw/source/core/docnode/swbaslnk.cxx sw/source/core/draw/dcontact.cxx sw/source/core/draw/dflyobj.cxx sw/source/core/draw/drawdoc.cxx sw/source/core/draw/dview.cxx sw/source/core/draw/makefile.mk sw/source/core/edit/autofmt.cxx sw/source/core/edit/edattr.cxx sw/source/core/edit/eddel.cxx sw/source/core/edit/edfcol.cxx sw/source/core/edit/edfld.cxx sw/source/core/edit/edfldexp.cxx sw/source/core/edit/edfmt.cxx sw/source/core/edit/edglss.cxx sw/source/core/edit/editsh.cxx sw/source/core/edit/edlingu.cxx sw/source/core/edit/ednumber.cxx sw/source/core/edit/edsect.cxx sw/source/core/edit/edtab.cxx sw/source/core/edit/edtox.cxx sw/source/core/edit/edundo.cxx sw/source/core/edit/makefile.mk sw/source/core/except/dbgloop.cxx sw/source/core/except/errhdl.cxx sw/source/core/fields/authfld.cxx sw/source/core/fields/cellfml.cxx sw/source/core/fields/chpfld.cxx sw/source/core/fields/dbfld.cxx sw/source/core/fields/ddefld.cxx sw/source/core/fields/ddetbl.cxx sw/source/core/fields/docufld.cxx sw/source/core/fields/expfld.cxx sw/source/core/fields/fldbas.cxx sw/source/core/fields/flddat.cxx sw/source/core/fields/flddropdown.cxx sw/source/core/fields/macrofld.cxx sw/source/core/fields/makefile.mk sw/source/core/fields/reffld.cxx sw/source/core/fields/scrptfld.cxx sw/source/core/fields/tblcalc.cxx sw/source/core/fields/usrfld.cxx sw/source/core/frmedt/fecopy.cxx sw/source/core/frmedt/fedesc.cxx sw/source/core/frmedt/fefly1.cxx sw/source/core/frmedt/feshview.cxx sw/source/core/frmedt/fetab.cxx sw/source/core/frmedt/fews.cxx sw/source/core/frmedt/makefile.mk sw/source/core/frmedt/tblsel.cxx sw/source/core/graphic/grfatr.cxx sw/source/core/inc/SwUndoFmt.hxx sw/source/core/inc/SwUndoTOXChange.hxx sw/source/core/inc/anchoredobjectposition.hxx sw/source/core/inc/dbgloop.hxx sw/source/core/inc/drawfont.hxx sw/source/core/inc/flowfrm.hxx sw/source/core/inc/frame.hxx sw/source/core/inc/frmtool.hxx sw/source/core/inc/layact.hxx sw/source/core/inc/layfrm.hxx sw/source/core/inc/notxtfrm.hxx sw/source/core/inc/rolbck.hxx sw/source/core/inc/rootfrm.hxx sw/source/core/inc/scriptinfo.hxx sw/source/core/inc/swblocks.hxx sw/source/core/inc/swcache.hxx sw/source/core/inc/tabfrm.hxx sw/source/core/inc/txmsrt.hxx sw/source/core/inc/undoflystrattr.hxx sw/source/core/inc/viewimp.hxx sw/source/core/layout/atrfrm.cxx sw/source/core/layout/calcmove.cxx sw/source/core/layout/dbg_lay.cxx sw/source/core/layout/findfrm.cxx sw/source/core/layout/flowfrm.cxx sw/source/core/layout/fly.cxx sw/source/core/layout/flycnt.cxx sw/source/core/layout/flyincnt.cxx sw/source/core/layout/flylay.cxx sw/source/core/layout/frmtool.cxx sw/source/core/layout/ftnfrm.cxx sw/source/core/layout/layact.cxx sw/source/core/layout/laycache.cxx sw/source/core/layout/makefile.mk sw/source/core/layout/objectformatter.cxx sw/source/core/layout/pagechg.cxx sw/source/core/layout/pagedesc.cxx sw/source/core/layout/paintfrm.cxx sw/source/core/layout/sectfrm.cxx sw/source/core/layout/tabfrm.cxx sw/source/core/layout/trvlfrm.cxx sw/source/core/layout/unusedf.cxx sw/source/core/layout/wsfrm.cxx sw/source/core/makefile.mk sw/source/core/objectpositioning/anchoredobjectposition.cxx sw/source/core/objectpositioning/ascharanchoredobjectposition.cxx sw/source/core/objectpositioning/makefile.mk sw/source/core/objectpositioning/tocntntanchoredobjectposition.cxx sw/source/core/objectpositioning/tolayoutanchoredobjectposition.cxx sw/source/core/ole/ndole.cxx sw/source/core/para/makefile.mk sw/source/core/para/paratr.cxx sw/source/core/sw3io/makefile.mk sw/source/core/sw3io/sw3convert.cxx sw/source/core/swg/SwXMLTextBlocks.cxx sw/source/core/swg/makefile.mk sw/source/core/swg/swblocks.cxx sw/source/core/table/swnewtable.cxx sw/source/core/table/swtable.cxx sw/source/core/text/EnhancedPDFExportHelper.cxx sw/source/core/text/atrstck.cxx sw/source/core/text/frmcrsr.cxx sw/source/core/text/frmform.cxx sw/source/core/text/itrcrsr.cxx sw/source/core/text/itrform2.cxx sw/source/core/text/makefile.mk sw/source/core/text/porlay.cxx sw/source/core/text/pormulti.cxx sw/source/core/text/txtfld.cxx sw/source/core/text/txtfrm.cxx sw/source/core/text/txtio.cxx sw/source/core/tox/makefile.mk sw/source/core/tox/txmsrt.cxx sw/source/core/txtnode/fmtatr2.cxx sw/source/core/txtnode/fntcache.cxx sw/source/core/txtnode/fntcap.cxx sw/source/core/txtnode/makefile.mk sw/source/core/txtnode/ndhints.cxx sw/source/core/txtnode/ndtxt.cxx sw/source/core/txtnode/swfont.cxx sw/source/core/txtnode/thints.cxx sw/source/core/txtnode/txtatr2.cxx sw/source/core/txtnode/txtedt.cxx sw/source/core/undo/SwUndoField.cxx sw/source/core/undo/SwUndoPageDesc.cxx sw/source/core/undo/SwUndoTOXChange.cxx sw/source/core/undo/docundo.cxx sw/source/core/undo/makefile.mk sw/source/core/undo/rolbck.cxx sw/source/core/undo/unbkmk.cxx sw/source/core/undo/undel.cxx sw/source/core/undo/undobj.cxx sw/source/core/undo/undobj1.cxx sw/source/core/undo/unfmco.cxx sw/source/core/undo/unins.cxx sw/source/core/undo/unnum.cxx sw/source/core/undo/unoutl.cxx sw/source/core/undo/unredln.cxx sw/source/core/undo/unsect.cxx sw/source/core/undo/unsort.cxx sw/source/core/undo/unspnd.cxx sw/source/core/undo/untbl.cxx sw/source/core/unocore/makefile.mk sw/source/core/unocore/swunohelper.cxx sw/source/core/unocore/unobkm.cxx sw/source/core/unocore/unocoll.cxx sw/source/core/unocore/unocrsrhelper.cxx sw/source/core/unocore/unodraw.cxx sw/source/core/unocore/unofield.cxx sw/source/core/unocore/unoframe.cxx sw/source/core/unocore/unomap.cxx sw/source/core/unocore/unoprnms.cxx sw/source/core/unocore/unoredlines.cxx sw/source/core/unocore/unosett.cxx sw/source/core/unocore/unosrch.cxx sw/source/core/unocore/unostyle.cxx sw/source/core/unocore/unotbl.cxx sw/source/core/view/vdraw.cxx sw/source/core/view/viewimp.cxx sw/source/core/view/viewpg.cxx sw/source/core/view/viewsh.cxx sw/source/core/view/vnew.cxx sw/source/core/view/vprint.cxx sw/source/filter/ascii/ascatr.cxx sw/source/filter/ascii/makefile.mk sw/source/filter/ascii/wrtasc.cxx sw/source/filter/basflt/fltini.cxx sw/source/filter/basflt/iodetect.cxx sw/source/filter/basflt/makefile.mk sw/source/filter/html/SwAppletImpl.cxx sw/source/filter/html/css1atr.cxx sw/source/filter/html/htmlatr.cxx sw/source/filter/html/htmlbas.cxx sw/source/filter/html/htmlcss1.cxx sw/source/filter/html/htmlfly.cxx sw/source/filter/html/htmlftn.cxx sw/source/filter/html/htmlgrin.cxx sw/source/filter/html/htmlnum.cxx sw/source/filter/html/htmlplug.cxx sw/source/filter/html/htmltab.cxx sw/source/filter/html/makefile.mk sw/source/filter/html/parcss1.cxx sw/source/filter/html/svxcss1.cxx sw/source/filter/html/swhtml.cxx sw/source/filter/inc/msfilter.hxx sw/source/filter/inc/wrtswtbl.hxx sw/source/filter/rtf/makefile.mk sw/source/filter/rtf/rtffly.cxx sw/source/filter/rtf/rtfnum.cxx sw/source/filter/rtf/rtftbl.cxx sw/source/filter/rtf/swparrtf.cxx sw/source/filter/rtf/swparrtf.hxx sw/source/filter/writer/makefile.mk sw/source/filter/writer/writer.cxx sw/source/filter/writer/wrt_fn.cxx sw/source/filter/writer/wrtswtbl.cxx sw/source/filter/ww1/fltshell.cxx sw/source/filter/ww1/makefile.mk sw/source/filter/ww1/w1class.cxx sw/source/filter/ww1/w1class.hxx sw/source/filter/ww1/w1filter.cxx sw/source/filter/ww1/w1par.cxx sw/source/filter/ww1/w1sprm.cxx sw/source/filter/ww1/w1struct.hxx sw/source/filter/ww8/README-rtf.txt sw/source/filter/ww8/attributeoutputbase.hxx sw/source/filter/ww8/docxattributeoutput.cxx sw/source/filter/ww8/docxattributeoutput.hxx sw/source/filter/ww8/docxexport.cxx sw/source/filter/ww8/docxexport.hxx sw/source/filter/ww8/docxexportfilter.cxx sw/source/filter/ww8/dump/dump8.cxx sw/source/filter/ww8/dump/dump8a.cxx sw/source/filter/ww8/dump/msvbasic.cxx sw/source/filter/ww8/dump/msvbasic.hxx sw/source/filter/ww8/dump/ww8darr.cxx sw/source/filter/ww8/dump/ww8darr.hxx sw/source/filter/ww8/dump/ww8dout.cxx sw/source/filter/ww8/dump/ww8dout.hxx sw/source/filter/ww8/dump/ww8scan.cxx sw/source/filter/ww8/dump/ww8scan.hxx sw/source/filter/ww8/dump/ww8struc.hxx sw/source/filter/ww8/makefile.mk sw/source/filter/ww8/rtfattributeoutput.cxx sw/source/filter/ww8/rtfattributeoutput.hxx sw/source/filter/ww8/rtfexport.cxx sw/source/filter/ww8/rtfexport.hxx sw/source/filter/ww8/rtfexportfilter.cxx sw/source/filter/ww8/rtfexportfilter.hxx sw/source/filter/ww8/rtfimportfilter.cxx sw/source/filter/ww8/rtfimportfilter.hxx sw/source/filter/ww8/rtfsdrexport.cxx sw/source/filter/ww8/rtfsdrexport.hxx sw/source/filter/ww8/writerhelper.cxx sw/source/filter/ww8/writerwordglue.cxx sw/source/filter/ww8/wrtw8esh.cxx sw/source/filter/ww8/wrtw8nds.cxx sw/source/filter/ww8/wrtw8num.cxx sw/source/filter/ww8/wrtw8sty.cxx sw/source/filter/ww8/wrtww8.cxx sw/source/filter/ww8/wrtww8.hxx sw/source/filter/ww8/wrtww8gr.cxx sw/source/filter/ww8/ww8atr.cxx sw/source/filter/ww8/ww8attributeoutput.hxx sw/source/filter/ww8/ww8graf.cxx sw/source/filter/ww8/ww8graf.hxx sw/source/filter/ww8/ww8graf2.cxx sw/source/filter/ww8/ww8par.cxx sw/source/filter/ww8/ww8par.hxx sw/source/filter/ww8/ww8par2.cxx sw/source/filter/ww8/ww8par3.cxx sw/source/filter/ww8/ww8par5.cxx sw/source/filter/ww8/ww8par6.cxx sw/source/filter/ww8/ww8scan.cxx sw/source/filter/ww8/ww8scan.hxx sw/source/filter/ww8/ww8struc.hxx sw/source/filter/xml/makefile.mk sw/source/filter/xml/xmlimpit.cxx sw/source/filter/xml/xmltble.cxx sw/source/filter/xml/xmltbli.cxx sw/source/ui/app/appenv.cxx sw/source/ui/app/apphdl.cxx sw/source/ui/app/applab.cxx sw/source/ui/app/appopt.cxx sw/source/ui/app/docsh.cxx sw/source/ui/app/docsh2.cxx sw/source/ui/app/docshini.cxx sw/source/ui/app/docst.cxx sw/source/ui/app/docstyle.cxx sw/source/ui/app/makefile.mk sw/source/ui/app/mn.src sw/source/ui/app/swmodul1.cxx sw/source/ui/cctrl/makefile.mk sw/source/ui/cctrl/swlbox.cxx sw/source/ui/chrdlg/break.cxx sw/source/ui/chrdlg/ccoll.cxx sw/source/ui/chrdlg/chardlg.cxx sw/source/ui/chrdlg/drpcps.cxx sw/source/ui/chrdlg/makefile.mk sw/source/ui/chrdlg/numpara.cxx sw/source/ui/chrdlg/pardlg.cxx sw/source/ui/chrdlg/swuiccoll.cxx sw/source/ui/config/barcfg.cxx sw/source/ui/config/caption.cxx sw/source/ui/config/cfgitems.cxx sw/source/ui/config/fontcfg.cxx sw/source/ui/config/mailconfigpage.cxx sw/source/ui/config/makefile.mk sw/source/ui/config/modcfg.cxx sw/source/ui/config/optcomp.cxx sw/source/ui/config/optload.cxx sw/source/ui/config/optpage.cxx sw/source/ui/config/prtopt.cxx sw/source/ui/config/uinums.cxx sw/source/ui/config/usrpref.cxx sw/source/ui/config/viewopt.cxx sw/source/ui/dbui/dbinsdlg.cxx sw/source/ui/dbui/dbmgr.cxx sw/source/ui/dbui/dbtree.cxx sw/source/ui/dbui/makefile.mk sw/source/ui/dbui/mmaddressblockpage.cxx sw/source/ui/dbui/mmdocselectpage.cxx sw/source/ui/dbui/mmoutputpage.cxx sw/source/ui/dbui/swdbtoolsclient.cxx sw/source/ui/dialog/abstract.cxx sw/source/ui/dialog/ascfldlg.cxx sw/source/ui/dialog/macassgn.cxx sw/source/ui/dialog/makefile.mk sw/source/ui/dialog/regionsw.cxx sw/source/ui/dialog/swdlgfact.cxx sw/source/ui/dialog/swdlgfact.hxx sw/source/ui/dialog/uiregionsw.cxx sw/source/ui/dochdl/gloshdl.cxx sw/source/ui/dochdl/makefile.mk sw/source/ui/dochdl/swdtflvr.cxx sw/source/ui/docvw/PostItMgr.cxx sw/source/ui/docvw/SidebarWin.cxx sw/source/ui/docvw/edtdd.cxx sw/source/ui/docvw/edtwin.cxx sw/source/ui/docvw/edtwin2.cxx sw/source/ui/docvw/edtwin3.cxx sw/source/ui/docvw/makefile.mk sw/source/ui/docvw/romenu.cxx sw/source/ui/docvw/romenu.hxx sw/source/ui/docvw/srcedtw.cxx sw/source/ui/envelp/envfmt.cxx sw/source/ui/envelp/envimg.cxx sw/source/ui/envelp/envlop1.cxx sw/source/ui/envelp/envprt.cxx sw/source/ui/envelp/label1.cxx sw/source/ui/envelp/labfmt.cxx sw/source/ui/envelp/labprt.cxx sw/source/ui/envelp/mailmrge.cxx sw/source/ui/envelp/makefile.mk sw/source/ui/fldui/flddb.cxx sw/source/ui/fldui/flddinf.cxx sw/source/ui/fldui/flddok.cxx sw/source/ui/fldui/fldedt.cxx sw/source/ui/fldui/fldfunc.cxx sw/source/ui/fldui/fldmgr.cxx sw/source/ui/fldui/fldpage.cxx sw/source/ui/fldui/fldref.cxx sw/source/ui/fldui/fldtdlg.cxx sw/source/ui/fldui/fldvar.cxx sw/source/ui/fldui/fldwrap.cxx sw/source/ui/fldui/inpdlg.cxx sw/source/ui/fldui/makefile.mk sw/source/ui/fmtui/makefile.mk sw/source/ui/fmtui/tmpdlg.cxx sw/source/ui/frmdlg/colmgr.cxx sw/source/ui/frmdlg/column.cxx sw/source/ui/frmdlg/cption.cxx sw/source/ui/frmdlg/frmdlg.cxx sw/source/ui/frmdlg/frmmgr.cxx sw/source/ui/frmdlg/frmpage.cxx sw/source/ui/frmdlg/makefile.mk sw/source/ui/frmdlg/wrap.cxx sw/source/ui/globdoc/makefile.mk sw/source/ui/inc/bmpwin.hxx sw/source/ui/inc/colmgr.hxx sw/source/ui/inc/column.hxx sw/source/ui/inc/envimg.hxx sw/source/ui/inc/envlop.hxx sw/source/ui/inc/frmpage.hxx sw/source/ui/inc/inputwin.hxx sw/source/ui/inc/javaedit.hxx sw/source/ui/inc/num.hxx sw/source/ui/inc/optpage.hxx sw/source/ui/inc/regionsw.hxx sw/source/ui/inc/split.hxx sw/source/ui/inc/swlbox.hxx sw/source/ui/inc/swmn_tmpl.hrc sw/source/ui/inc/swuiidxmrk.hxx sw/source/ui/inc/tabsh.hxx sw/source/ui/inc/toxmgr.hxx sw/source/ui/inc/uiitems.hxx sw/source/ui/inc/view.hxx sw/source/ui/inc/workctrl.hxx sw/source/ui/inc/wrap.hxx sw/source/ui/inc/wrtsh.hxx sw/source/ui/index/cnttab.cxx sw/source/ui/index/makefile.mk sw/source/ui/index/toxmgr.cxx sw/source/ui/lingu/hhcwrp.cxx sw/source/ui/lingu/makefile.mk sw/source/ui/lingu/olmenu.cxx sw/source/ui/misc/bookmark.cxx sw/source/ui/misc/docfnote.cxx sw/source/ui/misc/glosbib.cxx sw/source/ui/misc/glosdoc.cxx sw/source/ui/misc/glshell.cxx sw/source/ui/misc/insfnote.cxx sw/source/ui/misc/linenum.cxx sw/source/ui/misc/makefile.mk sw/source/ui/misc/num.cxx sw/source/ui/misc/numberingtypelistbox.cxx sw/source/ui/misc/outline.cxx sw/source/ui/misc/pgfnote.cxx sw/source/ui/misc/pggrid.cxx sw/source/ui/misc/redlndlg.cxx sw/source/ui/misc/srtdlg.cxx sw/source/ui/misc/swmodalredlineacceptdlg.cxx sw/source/ui/ribbar/conarc.cxx sw/source/ui/ribbar/drawbase.cxx sw/source/ui/ribbar/inputwin.cxx sw/source/ui/ribbar/inputwin.src sw/source/ui/ribbar/makefile.mk sw/source/ui/ribbar/tbxanchr.cxx sw/source/ui/ribbar/workctrl.cxx sw/source/ui/ribbar/workctrl.src sw/source/ui/shells/annotsh.cxx sw/source/ui/shells/basesh.cxx sw/source/ui/shells/beziersh.cxx sw/source/ui/shells/drawdlg.cxx sw/source/ui/shells/drwbassh.cxx sw/source/ui/shells/drwtxtex.cxx sw/source/ui/shells/drwtxtsh.cxx sw/source/ui/shells/frmsh.cxx sw/source/ui/shells/grfsh.cxx sw/source/ui/shells/grfshex.cxx sw/source/ui/shells/makefile.mk sw/source/ui/shells/tabsh.cxx sw/source/ui/shells/textfld.cxx sw/source/ui/shells/textglos.cxx sw/source/ui/shells/textsh.cxx sw/source/ui/shells/textsh1.cxx sw/source/ui/shells/txtattr.cxx sw/source/ui/shells/txtcrsr.cxx sw/source/ui/shells/txtnum.cxx sw/source/ui/table/convert.cxx sw/source/ui/table/instable.cxx sw/source/ui/table/makefile.mk sw/source/ui/table/swtablerep.cxx sw/source/ui/table/tabledlg.cxx sw/source/ui/table/tablemgr.cxx sw/source/ui/table/tablepg.hxx sw/source/ui/table/tautofmt.cxx sw/source/ui/uiview/formatclipboard.cxx sw/source/ui/uiview/makefile.mk sw/source/ui/uiview/pview.cxx sw/source/ui/uiview/pview.src sw/source/ui/uiview/scroll.cxx sw/source/ui/uiview/srcview.cxx sw/source/ui/uiview/swcli.cxx sw/source/ui/uiview/uivwimp.cxx sw/source/ui/uiview/view.cxx sw/source/ui/uiview/view1.cxx sw/source/ui/uiview/view2.cxx sw/source/ui/uiview/viewcoll.cxx sw/source/ui/uiview/viewdlg2.cxx sw/source/ui/uiview/viewling.cxx sw/source/ui/uiview/viewmdi.cxx sw/source/ui/uiview/viewport.cxx sw/source/ui/uiview/viewprt.cxx sw/source/ui/uiview/viewsrch.cxx sw/source/ui/uiview/viewtab.cxx sw/source/ui/uno/SwXDocumentSettings.cxx sw/source/ui/uno/SwXPrintPreviewSettings.cxx sw/source/ui/uno/SwXPrintPreviewSettings.hxx sw/source/ui/uno/unoatxt.cxx sw/source/ui/uno/unomod.cxx sw/source/ui/uno/unotxdoc.cxx sw/source/ui/uno/unotxvw.cxx sw/source/ui/utlui/attrdesc.cxx sw/source/ui/utlui/content.cxx sw/source/ui/utlui/glbltree.cxx sw/source/ui/utlui/initui.cxx sw/source/ui/utlui/makefile.mk sw/source/ui/utlui/navipi.cxx sw/source/ui/utlui/navipi.src sw/source/ui/utlui/numfmtlb.cxx sw/source/ui/utlui/prcntfld.cxx sw/source/ui/utlui/uiitems.cxx sw/source/ui/utlui/uitool.cxx sw/source/ui/utlui/unotools.cxx sw/source/ui/utlui/viewlayoutctrl.cxx sw/source/ui/utlui/zoomctrl.cxx sw/source/ui/vba/makefile.mk sw/source/ui/vba/service.cxx sw/source/ui/web/makefile.mk sw/source/ui/wrtsh/makefile.mk sw/source/ui/wrtsh/wrtsh1.cxx sw/source/ui/wrtsh/wrtsh2.cxx sw/source/ui/wrtsh/wrtsh4.cxx sw/source/ui/wrtsh/wrtundo.cxx sw/util/hidother.src sw/util/makefile.mk sw/util/msword.map
2612 lines
84 KiB
C++
2612 lines
84 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*************************************************************************
|
|
*
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* Copyright 2000, 2010 Oracle and/or its affiliates.
|
|
*
|
|
* OpenOffice.org - a multi-platform office productivity suite
|
|
*
|
|
* This file is part of OpenOffice.org.
|
|
*
|
|
* OpenOffice.org is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License version 3
|
|
* only, as published by the Free Software Foundation.
|
|
*
|
|
* OpenOffice.org 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 Lesser General Public License version 3 for more details
|
|
* (a copy is included in the LICENSE file that accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* version 3 along with OpenOffice.org. If not, see
|
|
* <http://www.openoffice.org/license.html>
|
|
* for a copy of the LGPLv3 License.
|
|
*
|
|
************************************************************************/
|
|
|
|
// MARKER(update_precomp.py): autogen include statement, do not remove
|
|
#include "precompiled_sw.hxx"
|
|
|
|
#include <txtfrm.hxx>
|
|
#include <flyfrm.hxx>
|
|
#include <ndtxt.hxx>
|
|
#include <pam.hxx>
|
|
#include <unotextrange.hxx>
|
|
#include <unocrsrhelper.hxx>
|
|
#include <crstate.hxx>
|
|
#include <accmap.hxx>
|
|
#include <fesh.hxx>
|
|
#include <viewopt.hxx>
|
|
#include <osl/mutex.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <vcl/window.hxx>
|
|
#include <rtl/ustrbuf.hxx>
|
|
#include <com/sun/star/accessibility/AccessibleRole.hpp>
|
|
#include <com/sun/star/accessibility/AccessibleStateType.hpp>
|
|
#include <com/sun/star/accessibility/AccessibleTextType.hpp>
|
|
#include <com/sun/star/accessibility/AccessibleEventId.hpp>
|
|
#include <unotools/accessiblestatesethelper.hxx>
|
|
#include <com/sun/star/i18n/CharacterIteratorMode.hpp>
|
|
#include <com/sun/star/i18n/WordType.hpp>
|
|
#include <com/sun/star/i18n/XBreakIterator.hpp>
|
|
#include <com/sun/star/beans/UnknownPropertyException.hpp>
|
|
#include <breakit.hxx>
|
|
#include <accpara.hxx>
|
|
#include <access.hrc>
|
|
#include <accportions.hxx>
|
|
#include <sfx2/viewsh.hxx> // for ExecuteAtViewShell(...)
|
|
#include <sfx2/viewfrm.hxx> // for ExecuteAtViewShell(...)
|
|
#include <sfx2/dispatch.hxx> // for ExecuteAtViewShell(...)
|
|
#include <unotools/charclass.hxx> // for GetWordBoundary
|
|
// for get/setCharacterAttribute(...)
|
|
#include <unocrsr.hxx>
|
|
#include <unoport.hxx>
|
|
#include <doc.hxx>
|
|
#include <crsskip.hxx>
|
|
#include <txtatr.hxx>
|
|
#include <acchyperlink.hxx>
|
|
#include <acchypertextdata.hxx>
|
|
#include <unotools/accessiblerelationsethelper.hxx>
|
|
#include <com/sun/star/accessibility/AccessibleRelationType.hpp>
|
|
#include <comphelper/accessibletexthelper.hxx>
|
|
#include <unomap.hxx>
|
|
#include <unoprnms.hxx>
|
|
#include <com/sun/star/text/WritingMode2.hpp>
|
|
#include <editeng/brshitem.hxx>
|
|
#include <viewimp.hxx>
|
|
#include <boost/scoped_ptr.hpp>
|
|
#include <textmarkuphelper.hxx>
|
|
// #i10825#
|
|
#include <parachangetrackinginfo.hxx>
|
|
#include <com/sun/star/text/TextMarkupType.hpp>
|
|
// <--
|
|
#include <comphelper/stlunosequence.hxx> // #i92233#
|
|
|
|
#include <algorithm>
|
|
|
|
using namespace ::com::sun::star;
|
|
using namespace ::com::sun::star::accessibility;
|
|
|
|
using beans::PropertyValue;
|
|
using beans::XMultiPropertySet;
|
|
using beans::UnknownPropertyException;
|
|
using beans::PropertyState_DIRECT_VALUE;
|
|
|
|
using std::max;
|
|
using std::min;
|
|
using std::sort;
|
|
|
|
namespace com { namespace sun { namespace star {
|
|
namespace text {
|
|
class XText;
|
|
}
|
|
} } }
|
|
|
|
|
|
const sal_Char sServiceName[] = "com.sun.star.text.AccessibleParagraphView";
|
|
const sal_Char sImplementationName[] = "com.sun.star.comp.Writer.SwAccessibleParagraphView";
|
|
const xub_StrLen MAX_DESC_TEXT_LEN = 40;
|
|
const SwTxtNode* SwAccessibleParagraph::GetTxtNode() const
|
|
{
|
|
const SwFrm* pFrm = GetFrm();
|
|
DBG_ASSERT( pFrm->IsTxtFrm(), "The text frame has mutated!" );
|
|
|
|
const SwTxtNode* pNode = static_cast<const SwTxtFrm*>(pFrm)->GetTxtNode();
|
|
DBG_ASSERT( pNode != NULL, "A text frame without a text node." );
|
|
|
|
return pNode;
|
|
}
|
|
|
|
::rtl::OUString SwAccessibleParagraph::GetString()
|
|
{
|
|
return GetPortionData().GetAccessibleString();
|
|
}
|
|
|
|
::rtl::OUString SwAccessibleParagraph::GetDescription()
|
|
{
|
|
return ::rtl::OUString(); // provide empty description for paragraphs
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::GetCaretPos()
|
|
{
|
|
sal_Int32 nRet = -1;
|
|
|
|
// get the selection's point, and test whether it's in our node
|
|
// #i27301# - consider adjusted method signature
|
|
SwPaM* pCaret = GetCursor( false ); // caret is first PaM in PaM-ring
|
|
|
|
if( pCaret != NULL )
|
|
{
|
|
const SwTxtNode* pNode = GetTxtNode();
|
|
|
|
// check whether the point points into 'our' node
|
|
SwPosition* pPoint = pCaret->GetPoint();
|
|
if( pNode->GetIndex() == pPoint->nNode.GetIndex() )
|
|
{
|
|
// same node? Then check whether it's also within 'our' part
|
|
// of the paragraph
|
|
sal_uInt16 nIndex = pPoint->nContent.GetIndex();
|
|
if( GetPortionData().IsValidCorePosition( nIndex ) )
|
|
{
|
|
// Yes, it's us!
|
|
// consider that cursor/caret is in front of the list label
|
|
if ( pCaret->IsInFrontOfLabel() )
|
|
{
|
|
nRet = 0;
|
|
}
|
|
else
|
|
{
|
|
nRet = GetPortionData().GetAccessiblePosition( nIndex );
|
|
}
|
|
|
|
DBG_ASSERT( nRet >= 0, "invalid cursor?" );
|
|
DBG_ASSERT( nRet <= GetPortionData().GetAccessibleString().
|
|
getLength(), "invalid cursor?" );
|
|
}
|
|
// else: in this paragraph, but in different frame
|
|
}
|
|
// else: not in this paragraph
|
|
}
|
|
// else: no cursor -> no caret
|
|
|
|
return nRet;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetSelection(
|
|
sal_Int32& nStart, sal_Int32& nEnd)
|
|
{
|
|
sal_Bool bRet = sal_False;
|
|
nStart = -1;
|
|
nEnd = -1;
|
|
|
|
// get the selection, and test whether it affects our text node
|
|
SwPaM* pCrsr = GetCursor( true ); // #i27301# - consider adjusted method signature
|
|
if( pCrsr != NULL )
|
|
{
|
|
// get SwPosition for my node
|
|
const SwTxtNode* pNode = GetTxtNode();
|
|
sal_uLong nHere = pNode->GetIndex();
|
|
|
|
// iterate over ring
|
|
SwPaM* pRingStart = pCrsr;
|
|
do
|
|
{
|
|
// ignore, if no mark
|
|
if( pCrsr->HasMark() )
|
|
{
|
|
// check whether nHere is 'inside' pCrsr
|
|
SwPosition* pStart = pCrsr->Start();
|
|
sal_uLong nStartIndex = pStart->nNode.GetIndex();
|
|
SwPosition* pEnd = pCrsr->End();
|
|
sal_uLong nEndIndex = pEnd->nNode.GetIndex();
|
|
if( ( nHere >= nStartIndex ) &&
|
|
( nHere <= nEndIndex ) )
|
|
{
|
|
// translate start and end positions
|
|
|
|
// start position
|
|
sal_Int32 nLocalStart = -1;
|
|
if( nHere > nStartIndex )
|
|
{
|
|
// selection starts in previous node:
|
|
// then our local selection starts with the paragraph
|
|
nLocalStart = 0;
|
|
}
|
|
else
|
|
{
|
|
DBG_ASSERT( nHere == nStartIndex,
|
|
"miscalculated index" );
|
|
|
|
// selection starts in this node:
|
|
// then check whether it's before or inside our part of
|
|
// the paragraph, and if so, get the proper position
|
|
sal_uInt16 nCoreStart = pStart->nContent.GetIndex();
|
|
if( nCoreStart <
|
|
GetPortionData().GetFirstValidCorePosition() )
|
|
{
|
|
nLocalStart = 0;
|
|
}
|
|
else if( nCoreStart <=
|
|
GetPortionData().GetLastValidCorePosition() )
|
|
{
|
|
DBG_ASSERT(
|
|
GetPortionData().IsValidCorePosition(
|
|
nCoreStart ),
|
|
"problem determining valid core position" );
|
|
|
|
nLocalStart =
|
|
GetPortionData().GetAccessiblePosition(
|
|
nCoreStart );
|
|
}
|
|
}
|
|
|
|
// end position
|
|
sal_Int32 nLocalEnd = -1;
|
|
if( nHere < nEndIndex )
|
|
{
|
|
// selection ends in following node:
|
|
// then our local selection extends to the end
|
|
nLocalEnd = GetPortionData().GetAccessibleString().
|
|
getLength();
|
|
}
|
|
else
|
|
{
|
|
DBG_ASSERT( nHere == nEndIndex,
|
|
"miscalculated index" );
|
|
|
|
// selection ends in this node: then select everything
|
|
// before our part of the node
|
|
sal_uInt16 nCoreEnd = pEnd->nContent.GetIndex();
|
|
if( nCoreEnd >
|
|
GetPortionData().GetLastValidCorePosition() )
|
|
{
|
|
// selection extends beyond out part of this para
|
|
nLocalEnd = GetPortionData().GetAccessibleString().
|
|
getLength();
|
|
}
|
|
else if( nCoreEnd >=
|
|
GetPortionData().GetFirstValidCorePosition() )
|
|
{
|
|
// selection is inside our part of this para
|
|
DBG_ASSERT(
|
|
GetPortionData().IsValidCorePosition(
|
|
nCoreEnd ),
|
|
"problem determining valid core position" );
|
|
|
|
nLocalEnd = GetPortionData().GetAccessiblePosition(
|
|
nCoreEnd );
|
|
}
|
|
}
|
|
|
|
if( ( nLocalStart != -1 ) && ( nLocalEnd != -1 ) )
|
|
{
|
|
nStart = nLocalStart;
|
|
nEnd = nLocalEnd;
|
|
bRet = sal_True;
|
|
}
|
|
}
|
|
// else: this PaM doesn't point to this paragraph
|
|
}
|
|
// else: this PaM is collapsed and doesn't select anything
|
|
|
|
// next PaM in ring
|
|
pCrsr = static_cast<SwPaM*>( pCrsr->GetNext() );
|
|
}
|
|
while( !bRet && (pCrsr != pRingStart) );
|
|
}
|
|
// else: nocursor -> no selection
|
|
|
|
return bRet;
|
|
}
|
|
|
|
// #i27301# - new parameter <_bForSelection>
|
|
SwPaM* SwAccessibleParagraph::GetCursor( const bool _bForSelection )
|
|
{
|
|
// get the cursor shell; if we don't have any, we don't have a
|
|
// cursor/selection either
|
|
SwPaM* pCrsr = NULL;
|
|
SwCrsrShell* pCrsrShell = SwAccessibleParagraph::GetCrsrShell();
|
|
// #i27301# - if cursor is retrieved for selection, the cursors for
|
|
// a table selection has to be returned.
|
|
if ( pCrsrShell != NULL &&
|
|
( _bForSelection || !pCrsrShell->IsTableMode() ) )
|
|
// <--
|
|
{
|
|
SwFEShell *pFESh = pCrsrShell->ISA( SwFEShell )
|
|
? static_cast< SwFEShell * >( pCrsrShell ) : 0;
|
|
if( !pFESh ||
|
|
!(pFESh->IsFrmSelected() || pFESh->IsObjSelected() > 0) )
|
|
{
|
|
// get the selection, and test whether it affects our text node
|
|
pCrsr = pCrsrShell->GetCrsr( sal_False /* ??? */ );
|
|
}
|
|
}
|
|
|
|
return pCrsr;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::IsHeading() const
|
|
{
|
|
const SwTxtNode *pTxtNd = GetTxtNode();
|
|
return pTxtNd->IsOutline();
|
|
}
|
|
|
|
void SwAccessibleParagraph::GetStates(
|
|
::utl::AccessibleStateSetHelper& rStateSet )
|
|
{
|
|
SwAccessibleContext::GetStates( rStateSet );
|
|
|
|
// MULTILINE
|
|
rStateSet.AddState( AccessibleStateType::MULTI_LINE );
|
|
|
|
// MULTISELECTABLE
|
|
SwCrsrShell *pCrsrSh = GetCrsrShell();
|
|
if( pCrsrSh )
|
|
rStateSet.AddState( AccessibleStateType::MULTI_SELECTABLE );
|
|
|
|
// FOCUSABLE
|
|
if( pCrsrSh )
|
|
rStateSet.AddState( AccessibleStateType::FOCUSABLE );
|
|
|
|
// FOCUSED (simulates node index of cursor)
|
|
SwPaM* pCaret = GetCursor( false ); // #i27301# - consider adjusted method signature
|
|
const SwTxtNode* pTxtNd = GetTxtNode();
|
|
if( pCaret != 0 && pTxtNd != 0 &&
|
|
pTxtNd->GetIndex() == pCaret->GetPoint()->nNode.GetIndex() &&
|
|
nOldCaretPos != -1)
|
|
{
|
|
Window *pWin = GetWindow();
|
|
if( pWin && pWin->HasFocus() )
|
|
rStateSet.AddState( AccessibleStateType::FOCUSED );
|
|
::rtl::Reference < SwAccessibleContext > xThis( this );
|
|
GetMap()->SetCursorContext( xThis );
|
|
}
|
|
}
|
|
|
|
void SwAccessibleParagraph::_InvalidateContent( sal_Bool bVisibleDataFired )
|
|
{
|
|
::rtl::OUString sOldText( GetString() );
|
|
|
|
ClearPortionData();
|
|
|
|
const ::rtl::OUString& rText = GetString();
|
|
|
|
if( rText != sOldText )
|
|
{
|
|
// The text is changed
|
|
AccessibleEventObject aEvent;
|
|
aEvent.EventId = AccessibleEventId::TEXT_CHANGED;
|
|
|
|
// determine exact changes between sOldText and rText
|
|
comphelper::OCommonAccessibleText::implInitTextChangedEvent(
|
|
sOldText, rText,
|
|
aEvent.OldValue, aEvent.NewValue );
|
|
|
|
FireAccessibleEvent( aEvent );
|
|
}
|
|
else if( !bVisibleDataFired )
|
|
{
|
|
FireVisibleDataEvent();
|
|
}
|
|
|
|
sal_Bool bNewIsHeading = IsHeading();
|
|
sal_Bool bOldIsHeading;
|
|
{
|
|
osl::MutexGuard aGuard( aMutex );
|
|
bOldIsHeading = bIsHeading;
|
|
if( bIsHeading != bNewIsHeading )
|
|
bIsHeading = bNewIsHeading;
|
|
}
|
|
|
|
|
|
if( bNewIsHeading != bOldIsHeading || rText != sOldText )
|
|
{
|
|
::rtl::OUString sNewDesc( GetDescription() );
|
|
::rtl::OUString sOldDesc;
|
|
{
|
|
osl::MutexGuard aGuard( aMutex );
|
|
sOldDesc = sDesc;
|
|
if( sDesc != sNewDesc )
|
|
sDesc = sNewDesc;
|
|
}
|
|
|
|
if( sNewDesc != sOldDesc )
|
|
{
|
|
// The text is changed
|
|
AccessibleEventObject aEvent;
|
|
aEvent.EventId = AccessibleEventId::DESCRIPTION_CHANGED;
|
|
aEvent.OldValue <<= sOldDesc;
|
|
aEvent.NewValue <<= sNewDesc;
|
|
|
|
FireAccessibleEvent( aEvent );
|
|
}
|
|
}
|
|
}
|
|
|
|
void SwAccessibleParagraph::_InvalidateCursorPos()
|
|
{
|
|
// The text is changed
|
|
sal_Int32 nNew = GetCaretPos();
|
|
sal_Int32 nOld;
|
|
{
|
|
osl::MutexGuard aGuard( aMutex );
|
|
nOld = nOldCaretPos;
|
|
nOldCaretPos = nNew;
|
|
}
|
|
if( -1 != nNew )
|
|
{
|
|
// remember that object as the one that has the caret. This is
|
|
// neccessary to notify that object if the cursor leaves it.
|
|
::rtl::Reference < SwAccessibleContext > xThis( this );
|
|
GetMap()->SetCursorContext( xThis );
|
|
}
|
|
|
|
Window *pWin = GetWindow();
|
|
if( nOld != nNew )
|
|
{
|
|
// The cursor's node position is sumilated by the focus!
|
|
if( pWin && pWin->HasFocus() && -1 == nOld )
|
|
FireStateChangedEvent( AccessibleStateType::FOCUSED, sal_True );
|
|
|
|
|
|
AccessibleEventObject aEvent;
|
|
aEvent.EventId = AccessibleEventId::CARET_CHANGED;
|
|
aEvent.OldValue <<= nOld;
|
|
aEvent.NewValue <<= nNew;
|
|
|
|
FireAccessibleEvent( aEvent );
|
|
|
|
if( pWin && pWin->HasFocus() && -1 == nNew )
|
|
FireStateChangedEvent( AccessibleStateType::FOCUSED, sal_False );
|
|
}
|
|
}
|
|
|
|
void SwAccessibleParagraph::_InvalidateFocus()
|
|
{
|
|
Window *pWin = GetWindow();
|
|
if( pWin )
|
|
{
|
|
sal_Int32 nPos;
|
|
{
|
|
osl::MutexGuard aGuard( aMutex );
|
|
nPos = nOldCaretPos;
|
|
}
|
|
OSL_ENSURE( nPos != -1, "focus object should be selected" );
|
|
|
|
FireStateChangedEvent( AccessibleStateType::FOCUSED,
|
|
pWin->HasFocus() && nPos != -1 );
|
|
}
|
|
}
|
|
|
|
SwAccessibleParagraph::SwAccessibleParagraph(
|
|
SwAccessibleMap& rInitMap,
|
|
const SwTxtFrm& rTxtFrm )
|
|
: SwClient( const_cast<SwTxtNode*>(rTxtFrm.GetTxtNode()) ) // #i108125#
|
|
, SwAccessibleContext( &rInitMap, AccessibleRole::PARAGRAPH, &rTxtFrm )
|
|
, sDesc()
|
|
, pPortionData( NULL )
|
|
, pHyperTextData( NULL )
|
|
, nOldCaretPos( -1 )
|
|
, bIsHeading( sal_False )
|
|
, aSelectionHelper( *this )
|
|
, mpParaChangeTrackInfo( new SwParaChangeTrackingInfo( rTxtFrm ) ) // #i108125#
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
bIsHeading = IsHeading();
|
|
SetName( ::rtl::OUString() ); // set an empty accessibility name for paragraphs
|
|
|
|
// If this object has the focus, then it is remembered by the map itself.
|
|
nOldCaretPos = GetCaretPos();
|
|
}
|
|
|
|
SwAccessibleParagraph::~SwAccessibleParagraph()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
delete pPortionData;
|
|
delete pHyperTextData;
|
|
delete mpParaChangeTrackInfo; // #i108125#
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::HasCursor()
|
|
{
|
|
osl::MutexGuard aGuard( aMutex );
|
|
return nOldCaretPos != -1;
|
|
}
|
|
|
|
void SwAccessibleParagraph::UpdatePortionData()
|
|
throw( uno::RuntimeException )
|
|
{
|
|
// obtain the text frame
|
|
DBG_ASSERT( GetFrm() != NULL, "The text frame has vanished!" );
|
|
DBG_ASSERT( GetFrm()->IsTxtFrm(), "The text frame has mutated!" );
|
|
const SwTxtFrm* pFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
|
|
// build new portion data
|
|
delete pPortionData;
|
|
pPortionData = new SwAccessiblePortionData(
|
|
pFrm->GetTxtNode(), GetMap()->GetShell()->GetViewOptions() );
|
|
pFrm->VisitPortions( *pPortionData );
|
|
|
|
DBG_ASSERT( pPortionData != NULL, "UpdatePortionData() failed" );
|
|
}
|
|
|
|
void SwAccessibleParagraph::ClearPortionData()
|
|
{
|
|
delete pPortionData;
|
|
pPortionData = NULL;
|
|
|
|
delete pHyperTextData;
|
|
pHyperTextData = 0;
|
|
}
|
|
|
|
|
|
void SwAccessibleParagraph::ExecuteAtViewShell( sal_uInt16 nSlot )
|
|
{
|
|
DBG_ASSERT( GetMap() != NULL, "no map?" );
|
|
ViewShell* pViewShell = GetMap()->GetShell();
|
|
|
|
DBG_ASSERT( pViewShell != NULL, "View shell exptected!" );
|
|
SfxViewShell* pSfxShell = pViewShell->GetSfxViewShell();
|
|
|
|
DBG_ASSERT( pSfxShell != NULL, "SfxViewShell shell exptected!" );
|
|
if( !pSfxShell )
|
|
return;
|
|
|
|
SfxViewFrame *pFrame = pSfxShell->GetViewFrame();
|
|
DBG_ASSERT( pFrame != NULL, "View frame exptected!" );
|
|
if( !pFrame )
|
|
return;
|
|
|
|
SfxDispatcher *pDispatcher = pFrame->GetDispatcher();
|
|
DBG_ASSERT( pDispatcher != NULL, "Dispatcher exptected!" );
|
|
if( !pDispatcher )
|
|
return;
|
|
|
|
pDispatcher->Execute( nSlot );
|
|
}
|
|
|
|
SwXTextPortion* SwAccessibleParagraph::CreateUnoPortion(
|
|
sal_Int32 nStartIndex,
|
|
sal_Int32 nEndIndex )
|
|
{
|
|
DBG_ASSERT( (IsValidChar(nStartIndex, GetString().getLength()) &&
|
|
(nEndIndex == -1)) ||
|
|
IsValidRange(nStartIndex, nEndIndex, GetString().getLength()),
|
|
"please check parameters before calling this method" );
|
|
|
|
sal_uInt16 nStart = GetPortionData().GetModelPosition( nStartIndex );
|
|
sal_uInt16 nEnd = (nEndIndex == -1) ? (nStart + 1) :
|
|
GetPortionData().GetModelPosition( nEndIndex );
|
|
|
|
// create UNO cursor
|
|
SwTxtNode* pTxtNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
SwIndex aIndex( pTxtNode, nStart );
|
|
SwPosition aStartPos( *pTxtNode, aIndex );
|
|
SwUnoCrsr* pUnoCursor = pTxtNode->GetDoc()->CreateUnoCrsr( aStartPos );
|
|
pUnoCursor->SetMark();
|
|
pUnoCursor->GetMark()->nContent = nEnd;
|
|
|
|
// create a (dummy) text portion to be returned
|
|
uno::Reference<text::XText> aEmpty;
|
|
SwXTextPortion* pPortion =
|
|
new SwXTextPortion ( pUnoCursor, aEmpty, PORTION_TEXT);
|
|
delete pUnoCursor;
|
|
|
|
return pPortion;
|
|
}
|
|
|
|
|
|
//
|
|
// range checking for parameter
|
|
//
|
|
|
|
sal_Bool SwAccessibleParagraph::IsValidChar(
|
|
sal_Int32 nPos, sal_Int32 nLength)
|
|
{
|
|
return (nPos >= 0) && (nPos < nLength);
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::IsValidPosition(
|
|
sal_Int32 nPos, sal_Int32 nLength)
|
|
{
|
|
return (nPos >= 0) && (nPos <= nLength);
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::IsValidRange(
|
|
sal_Int32 nBegin, sal_Int32 nEnd, sal_Int32 nLength)
|
|
{
|
|
return IsValidPosition(nBegin, nLength) && IsValidPosition(nEnd, nLength);
|
|
}
|
|
|
|
|
|
//
|
|
// text boundaries
|
|
//
|
|
|
|
|
|
sal_Bool SwAccessibleParagraph::GetCharBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString&,
|
|
sal_Int32 nPos )
|
|
{
|
|
rBound.startPos = nPos;
|
|
rBound.endPos = nPos+1;
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetWordBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString& rText,
|
|
sal_Int32 nPos )
|
|
{
|
|
sal_Bool bRet = sal_False;
|
|
|
|
// now ask the Break-Iterator for the word
|
|
DBG_ASSERT( pBreakIt != NULL, "We always need a break." );
|
|
DBG_ASSERT( pBreakIt->GetBreakIter().is(), "No break-iterator." );
|
|
if( pBreakIt->GetBreakIter().is() )
|
|
{
|
|
// get locale for this position
|
|
sal_uInt16 nModelPos = GetPortionData().GetModelPosition( nPos );
|
|
lang::Locale aLocale = pBreakIt->GetLocale(
|
|
GetTxtNode()->GetLang( nModelPos ) );
|
|
|
|
// which type of word are we interested in?
|
|
// (DICTIONARY_WORD includes punctuation, ANY_WORD doesn't.)
|
|
const sal_uInt16 nWordType = i18n::WordType::ANY_WORD;
|
|
|
|
// get word boundary, as the Break-Iterator sees fit.
|
|
rBound = pBreakIt->GetBreakIter()->getWordBoundary(
|
|
rText, nPos, aLocale, nWordType, sal_True );
|
|
|
|
// It's a word if the first character is an alpha-numeric character.
|
|
bRet = GetAppCharClass().isLetterNumeric(
|
|
rText.getStr()[ rBound.startPos ] );
|
|
}
|
|
else
|
|
{
|
|
// no break Iterator -> no word
|
|
rBound.startPos = nPos;
|
|
rBound.endPos = nPos;
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetSentenceBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString&,
|
|
sal_Int32 nPos )
|
|
{
|
|
GetPortionData().GetSentenceBoundary( rBound, nPos );
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetLineBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString& rText,
|
|
sal_Int32 nPos )
|
|
{
|
|
if( rText.getLength() == nPos )
|
|
GetPortionData().GetLastLineBoundary( rBound );
|
|
else
|
|
GetPortionData().GetLineBoundary( rBound, nPos );
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetParagraphBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString& rText,
|
|
sal_Int32 )
|
|
{
|
|
rBound.startPos = 0;
|
|
rBound.endPos = rText.getLength();
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetAttributeBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString&,
|
|
sal_Int32 nPos )
|
|
{
|
|
GetPortionData().GetAttributeBoundary( rBound, nPos );
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::GetGlyphBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString& rText,
|
|
sal_Int32 nPos )
|
|
{
|
|
sal_Bool bRet = sal_False;
|
|
|
|
// ask the Break-Iterator for the glyph by moving one cell
|
|
// forward, and then one cell back
|
|
DBG_ASSERT( pBreakIt != NULL, "We always need a break." );
|
|
DBG_ASSERT( pBreakIt->GetBreakIter().is(), "No break-iterator." );
|
|
if( pBreakIt->GetBreakIter().is() )
|
|
{
|
|
// get locale for this position
|
|
sal_uInt16 nModelPos = GetPortionData().GetModelPosition( nPos );
|
|
lang::Locale aLocale = pBreakIt->GetLocale(
|
|
GetTxtNode()->GetLang( nModelPos ) );
|
|
|
|
// get word boundary, as the Break-Iterator sees fit.
|
|
const sal_uInt16 nIterMode = i18n::CharacterIteratorMode::SKIPCELL;
|
|
sal_Int32 nDone = 0;
|
|
rBound.endPos = pBreakIt->GetBreakIter()->nextCharacters(
|
|
rText, nPos, aLocale, nIterMode, 1, nDone );
|
|
rBound.startPos = pBreakIt->GetBreakIter()->previousCharacters(
|
|
rText, rBound.endPos, aLocale, nIterMode, 1, nDone );
|
|
|
|
DBG_ASSERT( rBound.startPos <= nPos, "start pos too high" );
|
|
DBG_ASSERT( rBound.endPos >= nPos, "end pos too low" );
|
|
}
|
|
else
|
|
{
|
|
// no break Iterator -> no glyph
|
|
rBound.startPos = nPos;
|
|
rBound.endPos = nPos;
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
|
|
sal_Bool SwAccessibleParagraph::GetTextBoundary(
|
|
i18n::Boundary& rBound,
|
|
const ::rtl::OUString& rText,
|
|
sal_Int32 nPos,
|
|
sal_Int16 nTextType )
|
|
throw (
|
|
lang::IndexOutOfBoundsException,
|
|
lang::IllegalArgumentException,
|
|
uno::RuntimeException)
|
|
{
|
|
// error checking
|
|
if( !( AccessibleTextType::LINE == nTextType
|
|
? IsValidPosition( nPos, rText.getLength() )
|
|
: IsValidChar( nPos, rText.getLength() ) ) )
|
|
throw lang::IndexOutOfBoundsException();
|
|
|
|
sal_Bool bRet;
|
|
|
|
switch( nTextType )
|
|
{
|
|
case AccessibleTextType::WORD:
|
|
bRet = GetWordBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::SENTENCE:
|
|
bRet = GetSentenceBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::PARAGRAPH:
|
|
bRet = GetParagraphBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::CHARACTER:
|
|
bRet = GetCharBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::LINE:
|
|
bRet = GetLineBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::ATTRIBUTE_RUN:
|
|
bRet = GetAttributeBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
case AccessibleTextType::GLYPH:
|
|
bRet = GetGlyphBoundary( rBound, rText, nPos );
|
|
break;
|
|
|
|
default:
|
|
throw lang::IllegalArgumentException( );
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
::rtl::OUString SAL_CALL SwAccessibleParagraph::getAccessibleDescription (void)
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC( XAccessibleContext );
|
|
|
|
osl::MutexGuard aGuard2( aMutex );
|
|
if( !sDesc.getLength() )
|
|
sDesc = GetDescription();
|
|
|
|
return sDesc;
|
|
}
|
|
|
|
lang::Locale SAL_CALL SwAccessibleParagraph::getLocale (void)
|
|
throw (IllegalAccessibleComponentStateException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
SwTxtFrm *pTxtFrm = PTR_CAST( SwTxtFrm, GetFrm() );
|
|
if( !pTxtFrm )
|
|
{
|
|
THROW_RUNTIME_EXCEPTION( XAccessibleContext, "internal error (no text frame)" );
|
|
}
|
|
|
|
const SwTxtNode *pTxtNd = pTxtFrm->GetTxtNode();
|
|
lang::Locale aLoc( pBreakIt->GetLocale( pTxtNd->GetLang( 0 ) ) );
|
|
|
|
return aLoc;
|
|
}
|
|
|
|
/** #i27138# - paragraphs are in relation CONTENT_FLOWS_FROM and/or CONTENT_FLOWS_TO */
|
|
uno::Reference<XAccessibleRelationSet> SAL_CALL SwAccessibleParagraph::getAccessibleRelationSet()
|
|
throw ( uno::RuntimeException )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC( XAccessibleContext );
|
|
|
|
utl::AccessibleRelationSetHelper* pHelper = new utl::AccessibleRelationSetHelper();
|
|
|
|
const SwTxtFrm* pTxtFrm = dynamic_cast<const SwTxtFrm*>(GetFrm());
|
|
OSL_ENSURE( pTxtFrm,
|
|
"<SwAccessibleParagraph::getAccessibleRelationSet()> - missing text frame");
|
|
if ( pTxtFrm )
|
|
{
|
|
const SwCntntFrm* pPrevCntFrm( pTxtFrm->FindPrevCnt( true ) );
|
|
if ( pPrevCntFrm )
|
|
{
|
|
uno::Sequence< uno::Reference<XInterface> > aSequence(1);
|
|
aSequence[0] = GetMap()->GetContext( pPrevCntFrm );
|
|
AccessibleRelation aAccRel( AccessibleRelationType::CONTENT_FLOWS_FROM,
|
|
aSequence );
|
|
pHelper->AddRelation( aAccRel );
|
|
}
|
|
|
|
const SwCntntFrm* pNextCntFrm( pTxtFrm->FindNextCnt( true ) );
|
|
if ( pNextCntFrm )
|
|
{
|
|
uno::Sequence< uno::Reference<XInterface> > aSequence(1);
|
|
aSequence[0] = GetMap()->GetContext( pNextCntFrm );
|
|
AccessibleRelation aAccRel( AccessibleRelationType::CONTENT_FLOWS_TO,
|
|
aSequence );
|
|
pHelper->AddRelation( aAccRel );
|
|
}
|
|
}
|
|
|
|
return pHelper;
|
|
}
|
|
|
|
void SAL_CALL SwAccessibleParagraph::grabFocus()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC( XAccessibleContext );
|
|
|
|
// get cursor shell
|
|
SwCrsrShell *pCrsrSh = GetCrsrShell();
|
|
SwPaM *pCrsr = GetCursor( false ); // #i27301# - consider new method signature
|
|
const SwTxtFrm *pTxtFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
const SwTxtNode* pTxtNd = pTxtFrm->GetTxtNode();
|
|
|
|
if( pCrsrSh != 0 && pTxtNd != 0 &&
|
|
( pCrsr == 0 ||
|
|
pCrsr->GetPoint()->nNode.GetIndex() != pTxtNd->GetIndex() ||
|
|
!pTxtFrm->IsInside( pCrsr->GetPoint()->nContent.GetIndex()) ) )
|
|
{
|
|
// create pam for selection
|
|
SwIndex aIndex( const_cast< SwTxtNode * >( pTxtNd ),
|
|
pTxtFrm->GetOfst() );
|
|
SwPosition aStartPos( *pTxtNd, aIndex );
|
|
SwPaM aPaM( aStartPos );
|
|
|
|
// set PaM at cursor shell
|
|
Select( aPaM );
|
|
|
|
|
|
}
|
|
|
|
/* ->#i13955# */
|
|
Window * pWindow = GetWindow();
|
|
|
|
if (pWindow != NULL)
|
|
pWindow->GrabFocus();
|
|
/* <-#i13955# */
|
|
}
|
|
|
|
// #i71385#
|
|
bool lcl_GetBackgroundColor( Color & rColor,
|
|
const SwFrm* pFrm,
|
|
SwCrsrShell* pCrsrSh )
|
|
{
|
|
const SvxBrushItem* pBackgrdBrush = 0;
|
|
const Color* pSectionTOXColor = 0;
|
|
SwRect aDummyRect;
|
|
if ( pFrm &&
|
|
pFrm->GetBackgroundBrush( pBackgrdBrush, pSectionTOXColor, aDummyRect, false ) )
|
|
{
|
|
if ( pSectionTOXColor )
|
|
{
|
|
rColor = *pSectionTOXColor;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
rColor = pBackgrdBrush->GetColor();
|
|
return true;
|
|
}
|
|
}
|
|
else if ( pCrsrSh )
|
|
{
|
|
rColor = pCrsrSh->Imp()->GetRetoucheColor();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getForeground()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
Color aBackgroundCol;
|
|
|
|
if ( lcl_GetBackgroundColor( aBackgroundCol, GetFrm(), GetCrsrShell() ) )
|
|
{
|
|
if ( aBackgroundCol.IsDark() )
|
|
{
|
|
return COL_WHITE;
|
|
}
|
|
else
|
|
{
|
|
return COL_BLACK;
|
|
}
|
|
}
|
|
|
|
return SwAccessibleContext::getForeground();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getBackground()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
Color aBackgroundCol;
|
|
|
|
if ( lcl_GetBackgroundColor( aBackgroundCol, GetFrm(), GetCrsrShell() ) )
|
|
{
|
|
return aBackgroundCol.GetColor();
|
|
}
|
|
|
|
return SwAccessibleContext::getBackground();
|
|
}
|
|
// <--
|
|
|
|
::rtl::OUString SAL_CALL SwAccessibleParagraph::getImplementationName()
|
|
throw( uno::RuntimeException )
|
|
{
|
|
return ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(sImplementationName));
|
|
}
|
|
|
|
sal_Bool SAL_CALL SwAccessibleParagraph::supportsService(
|
|
const ::rtl::OUString& sTestServiceName)
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return sTestServiceName.equalsAsciiL( sServiceName,
|
|
sizeof(sServiceName)-1 ) ||
|
|
sTestServiceName.equalsAsciiL( sAccessibleServiceName,
|
|
sizeof(sAccessibleServiceName)-1 );
|
|
}
|
|
|
|
uno::Sequence< ::rtl::OUString > SAL_CALL SwAccessibleParagraph::getSupportedServiceNames()
|
|
throw( uno::RuntimeException )
|
|
{
|
|
uno::Sequence< ::rtl::OUString > aRet(2);
|
|
::rtl::OUString* pArray = aRet.getArray();
|
|
pArray[0] = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(sServiceName) );
|
|
pArray[1] = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(sAccessibleServiceName) );
|
|
return aRet;
|
|
}
|
|
|
|
//
|
|
//===== XInterface =======================================================
|
|
//
|
|
|
|
uno::Any SwAccessibleParagraph::queryInterface( const uno::Type& rType )
|
|
throw (uno::RuntimeException)
|
|
{
|
|
uno::Any aRet;
|
|
if ( rType == ::getCppuType((uno::Reference<XAccessibleText> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleText> aAccText = (XAccessibleText *) *this; // resolve ambiguity
|
|
aRet <<= aAccText;
|
|
}
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleEditableText> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleEditableText> aAccEditText = this;
|
|
aRet <<= aAccEditText;
|
|
}
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleSelection> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleSelection> aAccSel = this;
|
|
aRet <<= aAccSel;
|
|
}
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleHypertext> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleHypertext> aAccHyp = this;
|
|
aRet <<= aAccHyp;
|
|
}
|
|
// #i63870#
|
|
// add interface com::sun:star:accessibility::XAccessibleTextAttributes
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleTextAttributes> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleTextAttributes> aAccTextAttr = this;
|
|
aRet <<= aAccTextAttr;
|
|
}
|
|
// <--
|
|
// #i89175#
|
|
// add interface com::sun:star:accessibility::XAccessibleTextMarkup
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleTextMarkup> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleTextMarkup> aAccTextMarkup = this;
|
|
aRet <<= aAccTextMarkup;
|
|
}
|
|
// add interface com::sun:star:accessibility::XAccessibleMultiLineText
|
|
else if ( rType == ::getCppuType((uno::Reference<XAccessibleMultiLineText> *)0) )
|
|
{
|
|
uno::Reference<XAccessibleMultiLineText> aAccMultiLineText = this;
|
|
aRet <<= aAccMultiLineText;
|
|
}
|
|
// <--
|
|
else
|
|
{
|
|
aRet = SwAccessibleContext::queryInterface(rType);
|
|
}
|
|
|
|
return aRet;
|
|
}
|
|
|
|
//====== XTypeProvider ====================================================
|
|
uno::Sequence< uno::Type > SAL_CALL SwAccessibleParagraph::getTypes() throw(uno::RuntimeException)
|
|
{
|
|
uno::Sequence< uno::Type > aTypes( SwAccessibleContext::getTypes() );
|
|
|
|
sal_Int32 nIndex = aTypes.getLength();
|
|
// #i63870# - add type accessibility::XAccessibleTextAttributes
|
|
// #i89175# - add type accessibility::XAccessibleTextMarkup and
|
|
// accessibility::XAccessibleMultiLineText
|
|
aTypes.realloc( nIndex + 6 );
|
|
|
|
uno::Type* pTypes = aTypes.getArray();
|
|
pTypes[nIndex++] = ::getCppuType( static_cast< uno::Reference< XAccessibleEditableText > * >( 0 ) );
|
|
pTypes[nIndex++] = ::getCppuType( static_cast< uno::Reference< XAccessibleTextAttributes > * >( 0 ) );
|
|
pTypes[nIndex++] = ::getCppuType( static_cast< uno::Reference< XAccessibleSelection > * >( 0 ) );
|
|
pTypes[nIndex++] = ::getCppuType( static_cast< uno::Reference< XAccessibleTextMarkup > * >( 0 ) );
|
|
pTypes[nIndex++] = ::getCppuType( static_cast< uno::Reference< XAccessibleMultiLineText > * >( 0 ) );
|
|
pTypes[nIndex] = ::getCppuType( static_cast< uno::Reference< XAccessibleHypertext > * >( 0 ) );
|
|
// <--
|
|
|
|
return aTypes;
|
|
}
|
|
|
|
uno::Sequence< sal_Int8 > SAL_CALL SwAccessibleParagraph::getImplementationId()
|
|
throw(uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
static uno::Sequence< sal_Int8 > aId( 16 );
|
|
static sal_Bool bInit = sal_False;
|
|
if(!bInit)
|
|
{
|
|
rtl_createUuid( (sal_uInt8 *)(aId.getArray() ), 0, sal_True );
|
|
bInit = sal_True;
|
|
}
|
|
return aId;
|
|
}
|
|
|
|
|
|
//
|
|
//===== XAccesibleText ===================================================
|
|
//
|
|
|
|
sal_Int32 SwAccessibleParagraph::getCaretPosition()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
sal_Int32 nRet = GetCaretPos();
|
|
{
|
|
osl::MutexGuard aOldCaretPosGuard( aMutex );
|
|
OSL_ENSURE( nRet == nOldCaretPos, "caret pos out of sync" );
|
|
nOldCaretPos = nRet;
|
|
}
|
|
if( -1 != nRet )
|
|
{
|
|
::rtl::Reference < SwAccessibleContext > xThis( this );
|
|
GetMap()->SetCursorContext( xThis );
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
sal_Bool SAL_CALL SwAccessibleParagraph::setCaretPosition( sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
// parameter checking
|
|
sal_Int32 nLength = GetString().getLength();
|
|
if ( ! IsValidPosition( nIndex, nLength ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
sal_Bool bRet = sal_False;
|
|
|
|
// get cursor shell
|
|
SwCrsrShell* pCrsrShell = GetCrsrShell();
|
|
if( pCrsrShell != NULL )
|
|
{
|
|
// create pam for selection
|
|
SwTxtNode* pNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
SwIndex aIndex( pNode, GetPortionData().GetModelPosition(nIndex));
|
|
SwPosition aStartPos( *pNode, aIndex );
|
|
SwPaM aPaM( aStartPos );
|
|
|
|
// set PaM at cursor shell
|
|
bRet = Select( aPaM );
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
sal_Unicode SwAccessibleParagraph::getCharacter( sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
::rtl::OUString sText( GetString() );
|
|
|
|
// return character (if valid)
|
|
if( IsValidChar(nIndex, sText.getLength() ) )
|
|
{
|
|
return sText.getStr()[nIndex];
|
|
}
|
|
else
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
// #i63870# - re-implement method on behalf of methods
|
|
// <_getDefaultAttributesImpl(..)> and <_getRunAttributesImpl(..)>
|
|
uno::Sequence<PropertyValue> SwAccessibleParagraph::getCharacterAttributes(
|
|
sal_Int32 nIndex,
|
|
const uno::Sequence< ::rtl::OUString >& aRequestedAttributes )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
const ::rtl::OUString& rText = GetString();
|
|
|
|
if( ! IsValidChar( nIndex, rText.getLength() ) )
|
|
throw lang::IndexOutOfBoundsException();
|
|
|
|
// retrieve default character attributes
|
|
tAccParaPropValMap aDefAttrSeq;
|
|
_getDefaultAttributesImpl( aRequestedAttributes, aDefAttrSeq, true );
|
|
|
|
// retrieved run character attributes
|
|
tAccParaPropValMap aRunAttrSeq;
|
|
_getRunAttributesImpl( nIndex, aRequestedAttributes, aRunAttrSeq );
|
|
|
|
// merge default and run attributes
|
|
uno::Sequence< PropertyValue > aValues( aDefAttrSeq.size() );
|
|
PropertyValue* pValues = aValues.getArray();
|
|
sal_Int32 i = 0;
|
|
for ( tAccParaPropValMap::const_iterator aDefIter = aDefAttrSeq.begin();
|
|
aDefIter != aDefAttrSeq.end();
|
|
++aDefIter )
|
|
{
|
|
tAccParaPropValMap::const_iterator aRunIter =
|
|
aRunAttrSeq.find( aDefIter->first );
|
|
if ( aRunIter != aRunAttrSeq.end() )
|
|
{
|
|
pValues[i] = aRunIter->second;
|
|
}
|
|
else
|
|
{
|
|
pValues[i] = aDefIter->second;
|
|
}
|
|
++i;
|
|
}
|
|
|
|
return aValues;
|
|
}
|
|
|
|
// #i63870#
|
|
void SwAccessibleParagraph::_getDefaultAttributesImpl(
|
|
const uno::Sequence< ::rtl::OUString >& aRequestedAttributes,
|
|
tAccParaPropValMap& rDefAttrSeq,
|
|
const bool bOnlyCharAttrs )
|
|
{
|
|
// retrieve default attributes
|
|
const SwTxtNode* pTxtNode( GetTxtNode() );
|
|
::boost::scoped_ptr<SfxItemSet> pSet;
|
|
if ( !bOnlyCharAttrs )
|
|
{
|
|
pSet.reset( new SfxItemSet( const_cast<SwAttrPool&>(pTxtNode->GetDoc()->GetAttrPool()),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END - 1,
|
|
RES_PARATR_BEGIN, RES_PARATR_END - 1,
|
|
RES_FRMATR_BEGIN, RES_FRMATR_END - 1,
|
|
0 ) );
|
|
}
|
|
else
|
|
{
|
|
pSet.reset( new SfxItemSet( const_cast<SwAttrPool&>(pTxtNode->GetDoc()->GetAttrPool()),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END - 1,
|
|
0 ) );
|
|
}
|
|
// #i82637# - From the perspective of the a11y API the default character
|
|
// attributes are the character attributes, which are set at the paragraph style
|
|
// of the paragraph. The character attributes set at the automatic paragraph
|
|
// style of the paragraph are treated as run attributes.
|
|
// pTxtNode->SwCntntNode::GetAttr( *pSet );
|
|
// get default paragraph attributes, if needed, and merge these into <pSet>
|
|
if ( !bOnlyCharAttrs )
|
|
{
|
|
SfxItemSet aParaSet( const_cast<SwAttrPool&>(pTxtNode->GetDoc()->GetAttrPool()),
|
|
RES_PARATR_BEGIN, RES_PARATR_END - 1,
|
|
RES_FRMATR_BEGIN, RES_FRMATR_END - 1,
|
|
0 );
|
|
pTxtNode->SwCntntNode::GetAttr( aParaSet );
|
|
pSet->Put( aParaSet );
|
|
}
|
|
// get default character attributes and merge these into <pSet>
|
|
OSL_ENSURE( pTxtNode->GetTxtColl(),
|
|
"<SwAccessibleParagraph::_getDefaultAttributesImpl(..)> - missing paragraph style. Serious defect, please inform OD!" );
|
|
if ( pTxtNode->GetTxtColl() )
|
|
{
|
|
SfxItemSet aCharSet( const_cast<SwAttrPool&>(pTxtNode->GetDoc()->GetAttrPool()),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END - 1,
|
|
0 );
|
|
aCharSet.Put( pTxtNode->GetTxtColl()->GetAttrSet() );
|
|
pSet->Put( aCharSet );
|
|
}
|
|
// <--
|
|
|
|
// build-up sequence containing the run attributes <rDefAttrSeq>
|
|
tAccParaPropValMap aDefAttrSeq;
|
|
{
|
|
const SfxItemPropertyMap* pPropMap =
|
|
aSwMapProvider.GetPropertySet( PROPERTY_MAP_TEXT_CURSOR )->getPropertyMap();
|
|
PropertyEntryVector_t aPropertyEntries = pPropMap->getPropertyEntries();
|
|
PropertyEntryVector_t::const_iterator aPropIt = aPropertyEntries.begin();
|
|
while ( aPropIt != aPropertyEntries.end() )
|
|
{
|
|
const SfxPoolItem* pItem = pSet->GetItem( aPropIt->nWID );
|
|
if ( pItem )
|
|
{
|
|
uno::Any aVal;
|
|
pItem->QueryValue( aVal, aPropIt->nMemberId );
|
|
|
|
PropertyValue rPropVal;
|
|
rPropVal.Name = aPropIt->sName;
|
|
rPropVal.Value = aVal;
|
|
rPropVal.Handle = -1;
|
|
rPropVal.State = beans::PropertyState_DEFAULT_VALUE;
|
|
|
|
aDefAttrSeq[rPropVal.Name] = rPropVal;
|
|
}
|
|
++aPropIt;
|
|
}
|
|
|
|
// #i72800#
|
|
// add property value entry for the paragraph style
|
|
if ( !bOnlyCharAttrs && pTxtNode->GetTxtColl() )
|
|
{
|
|
const ::rtl::OUString sParaStyleName =
|
|
::rtl::OUString::createFromAscii(
|
|
GetPropName( UNO_NAME_PARA_STYLE_NAME ).pName );
|
|
if ( aDefAttrSeq.find( sParaStyleName ) == aDefAttrSeq.end() )
|
|
{
|
|
PropertyValue rPropVal;
|
|
rPropVal.Name = sParaStyleName;
|
|
uno::Any aVal( uno::makeAny( ::rtl::OUString( pTxtNode->GetTxtColl()->GetName() ) ) );
|
|
rPropVal.Value = aVal;
|
|
rPropVal.Handle = -1;
|
|
rPropVal.State = beans::PropertyState_DEFAULT_VALUE;
|
|
|
|
aDefAttrSeq[rPropVal.Name] = rPropVal;
|
|
}
|
|
}
|
|
// <--
|
|
|
|
// #i73371#
|
|
// resolve value text::WritingMode2::PAGE of property value entry WritingMode
|
|
if ( !bOnlyCharAttrs && GetFrm() )
|
|
{
|
|
const ::rtl::OUString sWritingMode =
|
|
::rtl::OUString::createFromAscii(
|
|
GetPropName( UNO_NAME_WRITING_MODE ).pName );
|
|
tAccParaPropValMap::iterator aIter = aDefAttrSeq.find( sWritingMode );
|
|
if ( aIter != aDefAttrSeq.end() )
|
|
{
|
|
PropertyValue rPropVal( aIter->second );
|
|
sal_Int16 nVal = rPropVal.Value.get<sal_Int16>();
|
|
if ( nVal == text::WritingMode2::PAGE )
|
|
{
|
|
const SwFrm* pUpperFrm( GetFrm()->GetUpper() );
|
|
while ( pUpperFrm )
|
|
{
|
|
if ( pUpperFrm->GetType() &
|
|
( FRM_PAGE | FRM_FLY | FRM_SECTION | FRM_TAB | FRM_CELL ) )
|
|
{
|
|
if ( pUpperFrm->IsVertical() )
|
|
{
|
|
nVal = text::WritingMode2::TB_RL;
|
|
}
|
|
else if ( pUpperFrm->IsRightToLeft() )
|
|
{
|
|
nVal = text::WritingMode2::RL_TB;
|
|
}
|
|
else
|
|
{
|
|
nVal = text::WritingMode2::LR_TB;
|
|
}
|
|
rPropVal.Value <<= nVal;
|
|
aDefAttrSeq[rPropVal.Name] = rPropVal;
|
|
break;
|
|
}
|
|
|
|
if ( dynamic_cast<const SwFlyFrm*>(pUpperFrm) )
|
|
{
|
|
pUpperFrm = dynamic_cast<const SwFlyFrm*>(pUpperFrm)->GetAnchorFrm();
|
|
}
|
|
else
|
|
{
|
|
pUpperFrm = pUpperFrm->GetUpper();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// <--
|
|
}
|
|
|
|
if ( aRequestedAttributes.getLength() == 0 )
|
|
{
|
|
rDefAttrSeq = aDefAttrSeq;
|
|
}
|
|
else
|
|
{
|
|
const ::rtl::OUString* pReqAttrs = aRequestedAttributes.getConstArray();
|
|
const sal_Int32 nLength = aRequestedAttributes.getLength();
|
|
for( sal_Int32 i = 0; i < nLength; ++i )
|
|
{
|
|
tAccParaPropValMap::const_iterator const aIter = aDefAttrSeq.find( pReqAttrs[i] );
|
|
if ( aIter != aDefAttrSeq.end() )
|
|
{
|
|
rDefAttrSeq[ aIter->first ] = aIter->second;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
uno::Sequence< PropertyValue > SwAccessibleParagraph::getDefaultAttributes(
|
|
const uno::Sequence< ::rtl::OUString >& aRequestedAttributes )
|
|
throw ( uno::RuntimeException )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
tAccParaPropValMap aDefAttrSeq;
|
|
_getDefaultAttributesImpl( aRequestedAttributes, aDefAttrSeq );
|
|
|
|
// #i92233#
|
|
static rtl::OUString sMMToPixelRatio(RTL_CONSTASCII_USTRINGPARAM("MMToPixelRatio"));
|
|
bool bProvideMMToPixelRatio( false );
|
|
{
|
|
if ( aRequestedAttributes.getLength() == 0 )
|
|
{
|
|
bProvideMMToPixelRatio = true;
|
|
}
|
|
else
|
|
{
|
|
const rtl::OUString* aRequestedAttrIter =
|
|
::std::find( ::comphelper::stl_begin( aRequestedAttributes ),
|
|
::comphelper::stl_end( aRequestedAttributes ),
|
|
sMMToPixelRatio );
|
|
if ( aRequestedAttrIter != ::comphelper::stl_end( aRequestedAttributes ) )
|
|
{
|
|
bProvideMMToPixelRatio = true;
|
|
}
|
|
}
|
|
}
|
|
// <--
|
|
|
|
uno::Sequence< PropertyValue > aValues( aDefAttrSeq.size() +
|
|
( bProvideMMToPixelRatio ? 1 : 0 ) );
|
|
PropertyValue* pValues = aValues.getArray();
|
|
sal_Int32 i = 0;
|
|
for ( tAccParaPropValMap::const_iterator aIter = aDefAttrSeq.begin();
|
|
aIter != aDefAttrSeq.end();
|
|
++aIter )
|
|
{
|
|
pValues[i] = aIter->second;
|
|
++i;
|
|
}
|
|
|
|
// #i92233#
|
|
if ( bProvideMMToPixelRatio )
|
|
{
|
|
PropertyValue rPropVal;
|
|
rPropVal.Name = sMMToPixelRatio;
|
|
const Size a100thMMSize( 1000, 1000 );
|
|
const Size aPixelSize = GetMap()->LogicToPixel( a100thMMSize );
|
|
const float fRatio = ((float)a100thMMSize.Width()/100)/aPixelSize.Width();
|
|
rPropVal.Value = uno::makeAny( fRatio );
|
|
rPropVal.Handle = -1;
|
|
rPropVal.State = beans::PropertyState_DEFAULT_VALUE;
|
|
pValues[ aValues.getLength() - 1 ] = rPropVal;
|
|
}
|
|
// <--
|
|
|
|
return aValues;
|
|
}
|
|
|
|
void SwAccessibleParagraph::_getRunAttributesImpl(
|
|
const sal_Int32 nIndex,
|
|
const uno::Sequence< ::rtl::OUString >& aRequestedAttributes,
|
|
tAccParaPropValMap& rRunAttrSeq )
|
|
{
|
|
// create PaM for character at position <nIndex>
|
|
SwPaM* pPaM( 0 );
|
|
{
|
|
const SwTxtNode* pTxtNode( GetTxtNode() );
|
|
SwPosition* pStartPos = new SwPosition( *pTxtNode );
|
|
pStartPos->nContent.Assign( const_cast<SwTxtNode*>(pTxtNode), static_cast<sal_uInt16>(nIndex) );
|
|
SwPosition* pEndPos = new SwPosition( *pTxtNode );
|
|
pEndPos->nContent.Assign( const_cast<SwTxtNode*>(pTxtNode), static_cast<sal_uInt16>(nIndex+1) );
|
|
|
|
pPaM = new SwPaM( *pStartPos, *pEndPos );
|
|
|
|
delete pStartPos;
|
|
delete pEndPos;
|
|
}
|
|
|
|
// retrieve character attributes for the created PaM <pPaM>
|
|
SfxItemSet aSet( pPaM->GetDoc()->GetAttrPool(),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END -1,
|
|
0 );
|
|
// #i82637#
|
|
// From the perspective of the a11y API the character attributes, which
|
|
// are set at the automatic paragraph style of the paragraph are treated
|
|
// as run attributes.
|
|
// SwXTextCursor::GetCrsrAttr( *pPaM, aSet, sal_True, sal_True );
|
|
// get character attributes from automatic paragraph style and merge these into <aSet>
|
|
{
|
|
const SwTxtNode* pTxtNode( GetTxtNode() );
|
|
if ( pTxtNode->HasSwAttrSet() )
|
|
{
|
|
SfxItemSet aAutomaticParaStyleCharAttrs( pPaM->GetDoc()->GetAttrPool(),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END -1,
|
|
0 );
|
|
aAutomaticParaStyleCharAttrs.Put( *(pTxtNode->GetpSwAttrSet()), sal_False );
|
|
aSet.Put( aAutomaticParaStyleCharAttrs );
|
|
}
|
|
}
|
|
// get character attributes at <pPaM> and merge these into <aSet>
|
|
{
|
|
SfxItemSet aCharAttrsAtPaM( pPaM->GetDoc()->GetAttrPool(),
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END -1,
|
|
0 );
|
|
SwUnoCursorHelper::GetCrsrAttr(*pPaM, aCharAttrsAtPaM, sal_True, sal_True);
|
|
aSet.Put( aCharAttrsAtPaM );
|
|
}
|
|
// <--
|
|
|
|
// build-up sequence containing the run attributes <rRunAttrSeq>
|
|
{
|
|
tAccParaPropValMap aRunAttrSeq;
|
|
{
|
|
tAccParaPropValMap aDefAttrSeq;
|
|
uno::Sequence< ::rtl::OUString > aDummy;
|
|
_getDefaultAttributesImpl( aDummy, aDefAttrSeq, true ); // #i82637#
|
|
|
|
const SfxItemPropertyMap* pPropMap =
|
|
aSwMapProvider.GetPropertySet( PROPERTY_MAP_TEXT_CURSOR )->getPropertyMap();
|
|
PropertyEntryVector_t aPropertyEntries = pPropMap->getPropertyEntries();
|
|
PropertyEntryVector_t::const_iterator aPropIt = aPropertyEntries.begin();
|
|
while ( aPropIt != aPropertyEntries.end() )
|
|
{
|
|
const SfxPoolItem* pItem( 0 );
|
|
// #i82637# - Found character attributes, whose value equals the value of
|
|
// the corresponding default character attributes, are excluded.
|
|
if ( aSet.GetItemState( aPropIt->nWID, sal_True, &pItem ) == SFX_ITEM_SET )
|
|
{
|
|
uno::Any aVal;
|
|
pItem->QueryValue( aVal, aPropIt->nMemberId );
|
|
|
|
PropertyValue rPropVal;
|
|
rPropVal.Name = aPropIt->sName;
|
|
rPropVal.Value = aVal;
|
|
rPropVal.Handle = -1;
|
|
rPropVal.State = PropertyState_DIRECT_VALUE;
|
|
|
|
tAccParaPropValMap::const_iterator aDefIter =
|
|
aDefAttrSeq.find( rPropVal.Name );
|
|
if ( aDefIter == aDefAttrSeq.end() ||
|
|
rPropVal.Value != aDefIter->second.Value )
|
|
{
|
|
aRunAttrSeq[rPropVal.Name] = rPropVal;
|
|
}
|
|
}
|
|
|
|
++aPropIt;
|
|
}
|
|
}
|
|
|
|
if ( aRequestedAttributes.getLength() == 0 )
|
|
{
|
|
rRunAttrSeq = aRunAttrSeq;
|
|
}
|
|
else
|
|
{
|
|
const ::rtl::OUString* pReqAttrs = aRequestedAttributes.getConstArray();
|
|
const sal_Int32 nLength = aRequestedAttributes.getLength();
|
|
for( sal_Int32 i = 0; i < nLength; ++i )
|
|
{
|
|
tAccParaPropValMap::iterator aIter = aRunAttrSeq.find( pReqAttrs[i] );
|
|
if ( aIter != aRunAttrSeq.end() )
|
|
{
|
|
rRunAttrSeq[ (*aIter).first ] = (*aIter).second;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
delete pPaM;
|
|
}
|
|
|
|
uno::Sequence< PropertyValue > SwAccessibleParagraph::getRunAttributes(
|
|
sal_Int32 nIndex,
|
|
const uno::Sequence< ::rtl::OUString >& aRequestedAttributes )
|
|
throw ( lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
{
|
|
const ::rtl::OUString& rText = GetString();
|
|
if ( !IsValidChar( nIndex, rText.getLength() ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
}
|
|
|
|
tAccParaPropValMap aRunAttrSeq;
|
|
_getRunAttributesImpl( nIndex, aRequestedAttributes, aRunAttrSeq );
|
|
|
|
uno::Sequence< PropertyValue > aValues( aRunAttrSeq.size() );
|
|
PropertyValue* pValues = aValues.getArray();
|
|
sal_Int32 i = 0;
|
|
for ( tAccParaPropValMap::const_iterator aIter = aRunAttrSeq.begin();
|
|
aIter != aRunAttrSeq.end();
|
|
++aIter )
|
|
{
|
|
pValues[i] = aIter->second;
|
|
++i;
|
|
}
|
|
|
|
return aValues;
|
|
}
|
|
// <--
|
|
|
|
awt::Rectangle SwAccessibleParagraph::getCharacterBounds(
|
|
sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
|
|
/* #i12332# The position after the string needs special treatment.
|
|
IsValidChar -> IsValidPosition
|
|
*/
|
|
if( ! (IsValidPosition( nIndex, GetString().getLength() ) ) )
|
|
throw lang::IndexOutOfBoundsException();
|
|
|
|
/* #i12332# */
|
|
sal_Bool bBehindText = sal_False;
|
|
if ( nIndex == GetString().getLength() )
|
|
bBehindText = sal_True;
|
|
|
|
// get model position & prepare GetCharRect() arguments
|
|
SwCrsrMoveState aMoveState;
|
|
aMoveState.bRealHeight = sal_True;
|
|
aMoveState.bRealWidth = sal_True;
|
|
SwSpecialPos aSpecialPos;
|
|
SwTxtNode* pNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
|
|
sal_uInt16 nPos = 0;
|
|
|
|
/* #i12332# FillSpecialPos does not accept nIndex ==
|
|
GetString().getLength(). In that case nPos is set to the
|
|
length of the string in the core. This way GetCharRect
|
|
returns the rectangle for a cursor at the end of the
|
|
paragraph. */
|
|
if (bBehindText)
|
|
{
|
|
nPos = pNode->GetTxt().Len();
|
|
}
|
|
else
|
|
nPos = GetPortionData().FillSpecialPos
|
|
(nIndex, aSpecialPos, aMoveState.pSpecialPos );
|
|
|
|
// call GetCharRect
|
|
SwRect aCoreRect;
|
|
SwIndex aIndex( pNode, nPos );
|
|
SwPosition aPosition( *pNode, aIndex );
|
|
GetFrm()->GetCharRect( aCoreRect, aPosition, &aMoveState );
|
|
|
|
// translate core coordinates into accessibility coordinates
|
|
Window *pWin = GetWindow();
|
|
CHECK_FOR_WINDOW( XAccessibleComponent, pWin );
|
|
|
|
Rectangle aScreenRect( GetMap()->CoreToPixel( aCoreRect.SVRect() ));
|
|
SwRect aFrmLogBounds( GetBounds( *(GetMap()) ) ); // twip rel to doc root
|
|
|
|
Point aFrmPixPos( GetMap()->CoreToPixel( aFrmLogBounds.SVRect() ).TopLeft() );
|
|
aScreenRect.Move( -aFrmPixPos.X(), -aFrmPixPos.Y() );
|
|
|
|
// convert into AWT Rectangle
|
|
return awt::Rectangle(
|
|
aScreenRect.Left(), aScreenRect.Top(),
|
|
aScreenRect.GetWidth(), aScreenRect.GetHeight() );
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::getCharacterCount()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
return GetString().getLength();
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::getIndexAtPoint( const awt::Point& rPoint )
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
// construct SwPosition (where GetCrsrOfst() will put the result into)
|
|
SwTxtNode* pNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
SwIndex aIndex( pNode, 0);
|
|
SwPosition aPos( *pNode, aIndex );
|
|
|
|
// construct Point (translate into layout coordinates)
|
|
Window *pWin = GetWindow();
|
|
CHECK_FOR_WINDOW( XAccessibleComponent, pWin );
|
|
Point aPoint( rPoint.X, rPoint.Y );
|
|
SwRect aLogBounds( GetBounds( *(GetMap()), GetFrm() ) ); // twip rel to doc root
|
|
Point aPixPos( GetMap()->CoreToPixel( aLogBounds.SVRect() ).TopLeft() );
|
|
aPoint.X() += aPixPos.X();
|
|
aPoint.Y() += aPixPos.Y();
|
|
MapMode aMapMode = pWin->GetMapMode();
|
|
Point aCorePoint( GetMap()->PixelToCore( aPoint ) );
|
|
if( !aLogBounds.IsInside( aCorePoint ) )
|
|
{
|
|
/* #i12332# rPoint is may also be in rectangle returned by
|
|
getCharacterBounds(getCharacterCount() */
|
|
|
|
awt::Rectangle aRectEndPos =
|
|
getCharacterBounds(getCharacterCount());
|
|
|
|
if (rPoint.X - aRectEndPos.X >= 0 &&
|
|
rPoint.X - aRectEndPos.X < aRectEndPos.Width &&
|
|
rPoint.Y - aRectEndPos.Y >= 0 &&
|
|
rPoint.Y - aRectEndPos.Y < aRectEndPos.Height)
|
|
return getCharacterCount();
|
|
|
|
return -1;
|
|
}
|
|
|
|
// ask core for position
|
|
DBG_ASSERT( GetFrm() != NULL, "The text frame has vanished!" );
|
|
DBG_ASSERT( GetFrm()->IsTxtFrm(), "The text frame has mutated!" );
|
|
const SwTxtFrm* pFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
SwCrsrMoveState aMoveState;
|
|
aMoveState.bPosMatchesBounds = sal_True;
|
|
sal_Bool bSuccess = pFrm->GetCrsrOfst( &aPos, aCorePoint, &aMoveState );
|
|
|
|
SwIndex aCntntIdx = aPos.nContent;
|
|
const xub_StrLen nIndex = aCntntIdx.GetIndex();
|
|
if ( nIndex > 0 )
|
|
{
|
|
SwRect aResultRect;
|
|
pFrm->GetCharRect( aResultRect, aPos );
|
|
bool bVert = pFrm->IsVertical();
|
|
bool bR2L = pFrm->IsRightToLeft();
|
|
|
|
if ( (!bVert && aResultRect.Pos().X() > aCorePoint.X()) ||
|
|
( bVert && aResultRect.Pos().Y() > aCorePoint.Y()) ||
|
|
( bR2L && aResultRect.Right() < aCorePoint.X()) )
|
|
{
|
|
SwIndex aIdxPrev( pNode, nIndex - 1);
|
|
SwPosition aPosPrev( *pNode, aIdxPrev );
|
|
SwRect aResultRectPrev;
|
|
pFrm->GetCharRect( aResultRectPrev, aPosPrev );
|
|
if ( (!bVert && aResultRectPrev.Pos().X() < aCorePoint.X() && aResultRect.Pos().Y() == aResultRectPrev.Pos().Y()) ||
|
|
( bVert && aResultRectPrev.Pos().Y() < aCorePoint.Y() && aResultRect.Pos().X() == aResultRectPrev.Pos().X()) ||
|
|
( bR2L && aResultRectPrev.Right() > aCorePoint.X() && aResultRect.Pos().Y() == aResultRectPrev.Pos().Y()) )
|
|
aPos = aPosPrev;
|
|
}
|
|
}
|
|
|
|
return bSuccess ?
|
|
GetPortionData().GetAccessiblePosition( aPos.nContent.GetIndex() )
|
|
: -1L;
|
|
}
|
|
|
|
::rtl::OUString SwAccessibleParagraph::getSelectedText()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
sal_Int32 nStart, nEnd;
|
|
sal_Bool bSelected = GetSelection( nStart, nEnd );
|
|
return bSelected
|
|
? GetString().copy( nStart, nEnd - nStart )
|
|
: ::rtl::OUString();
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::getSelectionStart()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
sal_Int32 nStart, nEnd;
|
|
GetSelection( nStart, nEnd );
|
|
return nStart;
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::getSelectionEnd()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
sal_Int32 nStart, nEnd;
|
|
GetSelection( nStart, nEnd );
|
|
return nEnd;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::setSelection( sal_Int32 nStartIndex, sal_Int32 nEndIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
// parameter checking
|
|
sal_Int32 nLength = GetString().getLength();
|
|
if ( ! IsValidRange( nStartIndex, nEndIndex, nLength ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
sal_Bool bRet = sal_False;
|
|
|
|
// get cursor shell
|
|
SwCrsrShell* pCrsrShell = GetCrsrShell();
|
|
if( pCrsrShell != NULL )
|
|
{
|
|
// create pam for selection
|
|
SwTxtNode* pNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
SwIndex aIndex( pNode, GetPortionData().GetModelPosition(nStartIndex));
|
|
SwPosition aStartPos( *pNode, aIndex );
|
|
SwPaM aPaM( aStartPos );
|
|
aPaM.SetMark();
|
|
aPaM.GetPoint()->nContent =
|
|
GetPortionData().GetModelPosition(nEndIndex);
|
|
|
|
// set PaM at cursor shell
|
|
bRet = Select( aPaM );
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
::rtl::OUString SwAccessibleParagraph::getText()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
return GetString();
|
|
}
|
|
|
|
::rtl::OUString SwAccessibleParagraph::getTextRange(
|
|
sal_Int32 nStartIndex, sal_Int32 nEndIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
::rtl::OUString sText( GetString() );
|
|
|
|
if ( IsValidRange( nStartIndex, nEndIndex, sText.getLength() ) )
|
|
{
|
|
OrderRange( nStartIndex, nEndIndex );
|
|
return sText.copy(nStartIndex, nEndIndex-nStartIndex );
|
|
}
|
|
else
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SwAccessibleParagraph::getTextAtIndex( sal_Int32 nIndex, sal_Int16 nTextType ) throw (lang::IndexOutOfBoundsException, lang::IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
/*accessibility::*/TextSegment aResult;
|
|
aResult.SegmentStart = -1;
|
|
aResult.SegmentEnd = -1;
|
|
|
|
const ::rtl::OUString rText = GetString();
|
|
// implement the silly specification that first position after
|
|
// text must return an empty string, rather than throwing an
|
|
// IndexOutOfBoundsException, except for LINE, where the last
|
|
// line is returned
|
|
if( nIndex == rText.getLength() && AccessibleTextType::LINE != nTextType )
|
|
return aResult;
|
|
|
|
// with error checking
|
|
i18n::Boundary aBound;
|
|
sal_Bool bWord = GetTextBoundary( aBound, rText, nIndex, nTextType );
|
|
|
|
DBG_ASSERT( aBound.startPos >= 0, "illegal boundary" );
|
|
DBG_ASSERT( aBound.startPos <= aBound.endPos, "illegal boundary" );
|
|
|
|
// return word (if present)
|
|
if ( bWord )
|
|
{
|
|
aResult.SegmentText = rText.copy( aBound.startPos, aBound.endPos - aBound.startPos );
|
|
aResult.SegmentStart = aBound.startPos;
|
|
aResult.SegmentEnd = aBound.endPos;
|
|
}
|
|
|
|
return aResult;
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SwAccessibleParagraph::getTextBeforeIndex( sal_Int32 nIndex, sal_Int16 nTextType ) throw (lang::IndexOutOfBoundsException, lang::IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
const ::rtl::OUString rText = GetString();
|
|
|
|
/*accessibility::*/TextSegment aResult;
|
|
aResult.SegmentStart = -1;
|
|
aResult.SegmentEnd = -1;
|
|
|
|
// get starting pos
|
|
i18n::Boundary aBound;
|
|
if (nIndex == rText.getLength())
|
|
aBound.startPos = aBound.endPos = nIndex;
|
|
else
|
|
{
|
|
sal_Bool bTmp = GetTextBoundary( aBound, rText, nIndex, nTextType );
|
|
|
|
if ( ! bTmp )
|
|
aBound.startPos = aBound.endPos = nIndex;
|
|
}
|
|
|
|
// now skip to previous word
|
|
sal_Bool bWord = sal_False;
|
|
while( !bWord )
|
|
{
|
|
nIndex = min( nIndex, aBound.startPos ) - 1;
|
|
if( nIndex >= 0 )
|
|
bWord = GetTextBoundary( aBound, rText, nIndex, nTextType );
|
|
else
|
|
break; // exit if beginning of string is reached
|
|
}
|
|
|
|
if ( bWord )
|
|
{
|
|
aResult.SegmentText = rText.copy( aBound.startPos, aBound.endPos - aBound.startPos );
|
|
aResult.SegmentStart = aBound.startPos;
|
|
aResult.SegmentEnd = aBound.endPos;
|
|
};
|
|
return aResult;
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SwAccessibleParagraph::getTextBehindIndex( sal_Int32 nIndex, sal_Int16 nTextType ) throw (lang::IndexOutOfBoundsException, lang::IllegalArgumentException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
|
|
/*accessibility::*/TextSegment aResult;
|
|
aResult.SegmentStart = -1;
|
|
aResult.SegmentEnd = -1;
|
|
const ::rtl::OUString rText = GetString();
|
|
|
|
// implement the silly specification that first position after
|
|
// text must return an empty string, rather than throwing an
|
|
// IndexOutOfBoundsException
|
|
if( nIndex == rText.getLength() )
|
|
return aResult;
|
|
|
|
|
|
// get first word, then skip to next word
|
|
i18n::Boundary aBound;
|
|
GetTextBoundary( aBound, rText, nIndex, nTextType );
|
|
sal_Bool bWord = sal_False;
|
|
while( !bWord )
|
|
{
|
|
nIndex = max( sal_Int32(nIndex+1), aBound.endPos );
|
|
if( nIndex < rText.getLength() )
|
|
bWord = GetTextBoundary( aBound, rText, nIndex, nTextType );
|
|
else
|
|
break; // exit if end of string is reached
|
|
}
|
|
|
|
if ( bWord )
|
|
{
|
|
aResult.SegmentText = rText.copy( aBound.startPos, aBound.endPos - aBound.startPos );
|
|
aResult.SegmentStart = aBound.startPos;
|
|
aResult.SegmentEnd = aBound.endPos;
|
|
}
|
|
return aResult;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::copyText( sal_Int32 nStartIndex, sal_Int32 nEndIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
CHECK_FOR_DEFUNC_THIS( XAccessibleText, *this );
|
|
SolarMutexGuard aGuard;
|
|
|
|
// select and copy (through dispatch mechanism)
|
|
setSelection( nStartIndex, nEndIndex );
|
|
ExecuteAtViewShell( SID_COPY );
|
|
return sal_True;
|
|
}
|
|
|
|
|
|
//
|
|
//===== XAccesibleEditableText ==========================================
|
|
//
|
|
|
|
sal_Bool SwAccessibleParagraph::cutText( sal_Int32 nStartIndex, sal_Int32 nEndIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleEditableText );
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( !IsEditableState() )
|
|
return sal_False;
|
|
|
|
// select and cut (through dispatch mechanism)
|
|
setSelection( nStartIndex, nEndIndex );
|
|
ExecuteAtViewShell( SID_CUT );
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::pasteText( sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleEditableText );
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( !IsEditableState() )
|
|
return sal_False;
|
|
|
|
// select and paste (through dispatch mechanism)
|
|
setSelection( nIndex, nIndex );
|
|
ExecuteAtViewShell( SID_PASTE );
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::deleteText( sal_Int32 nStartIndex, sal_Int32 nEndIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
return replaceText( nStartIndex, nEndIndex, ::rtl::OUString() );
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::insertText( const ::rtl::OUString& sText, sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
return replaceText( nIndex, nIndex, sText );
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::replaceText(
|
|
sal_Int32 nStartIndex, sal_Int32 nEndIndex,
|
|
const ::rtl::OUString& sReplacement )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC( XAccessibleEditableText );
|
|
|
|
const ::rtl::OUString& rText = GetString();
|
|
|
|
if( IsValidRange( nStartIndex, nEndIndex, rText.getLength() ) )
|
|
{
|
|
if( !IsEditableState() )
|
|
return sal_False;
|
|
|
|
SwTxtNode* pNode = const_cast<SwTxtNode*>( GetTxtNode() );
|
|
|
|
// translate positions
|
|
sal_uInt16 nStart, nEnd;
|
|
sal_Bool bSuccess = GetPortionData().GetEditableRange(
|
|
nStartIndex, nEndIndex, nStart, nEnd );
|
|
|
|
// edit only if the range is editable
|
|
if( bSuccess )
|
|
{
|
|
// create SwPosition for nStartIndex
|
|
SwIndex aIndex( pNode, nStart );
|
|
SwPosition aStartPos( *pNode, aIndex );
|
|
|
|
// create SwPosition for nEndIndex
|
|
SwPosition aEndPos( aStartPos );
|
|
aEndPos.nContent = nEnd;
|
|
|
|
// now create XTextRange as helper and set string
|
|
const uno::Reference<text::XTextRange> xRange(
|
|
SwXTextRange::CreateXTextRange(
|
|
*pNode->GetDoc(), aStartPos, &aEndPos));
|
|
xRange->setString(sReplacement);
|
|
|
|
// delete portion data
|
|
ClearPortionData();
|
|
}
|
|
|
|
return bSuccess;
|
|
}
|
|
else
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
struct IndexCompare
|
|
{
|
|
const PropertyValue* pValues;
|
|
IndexCompare( const PropertyValue* pVals ) : pValues(pVals) {}
|
|
bool operator() ( const sal_Int32& a, const sal_Int32& b ) const
|
|
{
|
|
return (pValues[a].Name < pValues[b].Name) ? true : false;
|
|
}
|
|
};
|
|
|
|
|
|
sal_Bool SwAccessibleParagraph::setAttributes(
|
|
sal_Int32 nStartIndex,
|
|
sal_Int32 nEndIndex,
|
|
const uno::Sequence<PropertyValue>& rAttributeSet )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC( XAccessibleEditableText );
|
|
|
|
const ::rtl::OUString& rText = GetString();
|
|
|
|
if( ! IsValidRange( nStartIndex, nEndIndex, rText.getLength() ) )
|
|
throw lang::IndexOutOfBoundsException();
|
|
|
|
if( !IsEditableState() )
|
|
return sal_False;
|
|
|
|
|
|
// create a (dummy) text portion for the sole purpose of calling
|
|
// setPropertyValue on it
|
|
uno::Reference<XMultiPropertySet> xPortion = CreateUnoPortion( nStartIndex,
|
|
nEndIndex );
|
|
|
|
// build sorted index array
|
|
sal_Int32 nLength = rAttributeSet.getLength();
|
|
const PropertyValue* pPairs = rAttributeSet.getConstArray();
|
|
sal_Int32* pIndices = new sal_Int32[nLength];
|
|
sal_Int32 i;
|
|
for( i = 0; i < nLength; i++ )
|
|
pIndices[i] = i;
|
|
sort( &pIndices[0], &pIndices[nLength], IndexCompare(pPairs) );
|
|
|
|
// create sorted sequences accoring to index array
|
|
uno::Sequence< ::rtl::OUString > aNames( nLength );
|
|
::rtl::OUString* pNames = aNames.getArray();
|
|
uno::Sequence< uno::Any > aValues( nLength );
|
|
uno::Any* pValues = aValues.getArray();
|
|
for( i = 0; i < nLength; i++ )
|
|
{
|
|
const PropertyValue& rVal = pPairs[pIndices[i]];
|
|
pNames[i] = rVal.Name;
|
|
pValues[i] = rVal.Value;
|
|
}
|
|
delete[] pIndices;
|
|
|
|
// now set the values
|
|
sal_Bool bRet = sal_True;
|
|
try
|
|
{
|
|
xPortion->setPropertyValues( aNames, aValues );
|
|
}
|
|
catch( UnknownPropertyException &e )
|
|
{
|
|
// error handling through return code!
|
|
bRet = sal_False;
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::setText( const ::rtl::OUString& sText )
|
|
throw (uno::RuntimeException)
|
|
{
|
|
return replaceText(0, GetString().getLength(), sText);
|
|
}
|
|
|
|
//===== XAccessibleSelection ============================================
|
|
|
|
void SwAccessibleParagraph::selectAccessibleChild(
|
|
sal_Int32 nChildIndex )
|
|
throw ( lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
aSelectionHelper.selectAccessibleChild(nChildIndex);
|
|
}
|
|
|
|
sal_Bool SwAccessibleParagraph::isAccessibleChildSelected(
|
|
sal_Int32 nChildIndex )
|
|
throw ( lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
return aSelectionHelper.isAccessibleChildSelected(nChildIndex);
|
|
}
|
|
|
|
void SwAccessibleParagraph::clearAccessibleSelection( )
|
|
throw ( uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
aSelectionHelper.clearAccessibleSelection();
|
|
}
|
|
|
|
void SwAccessibleParagraph::selectAllAccessibleChildren( )
|
|
throw ( uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
aSelectionHelper.selectAllAccessibleChildren();
|
|
}
|
|
|
|
sal_Int32 SwAccessibleParagraph::getSelectedAccessibleChildCount( )
|
|
throw ( uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
return aSelectionHelper.getSelectedAccessibleChildCount();
|
|
}
|
|
|
|
uno::Reference<XAccessible> SwAccessibleParagraph::getSelectedAccessibleChild(
|
|
sal_Int32 nSelectedChildIndex )
|
|
throw ( lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException)
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
return aSelectionHelper.getSelectedAccessibleChild(nSelectedChildIndex);
|
|
}
|
|
|
|
// index has to be treated as global child index.
|
|
void SwAccessibleParagraph::deselectAccessibleChild(
|
|
sal_Int32 nChildIndex )
|
|
throw ( lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException )
|
|
{
|
|
CHECK_FOR_DEFUNC( XAccessibleSelection );
|
|
|
|
aSelectionHelper.deselectAccessibleChild( nChildIndex );
|
|
}
|
|
|
|
//===== XAccessibleHypertext ============================================
|
|
|
|
class SwHyperlinkIter_Impl
|
|
{
|
|
const SwpHints *pHints;
|
|
xub_StrLen nStt;
|
|
xub_StrLen nEnd;
|
|
sal_uInt16 nPos;
|
|
|
|
public:
|
|
SwHyperlinkIter_Impl( const SwTxtFrm *pTxtFrm );
|
|
const SwTxtAttr *next();
|
|
sal_uInt16 getCurrHintPos() const { return nPos-1; }
|
|
|
|
xub_StrLen startIdx() const { return nStt; }
|
|
xub_StrLen endIdx() const { return nEnd; }
|
|
};
|
|
|
|
SwHyperlinkIter_Impl::SwHyperlinkIter_Impl( const SwTxtFrm *pTxtFrm ) :
|
|
pHints( pTxtFrm->GetTxtNode()->GetpSwpHints() ),
|
|
nStt( pTxtFrm->GetOfst() ),
|
|
nPos( 0 )
|
|
{
|
|
const SwTxtFrm *pFollFrm = pTxtFrm->GetFollow();
|
|
nEnd = pFollFrm ? pFollFrm->GetOfst() : pTxtFrm->GetTxtNode()->Len();
|
|
}
|
|
|
|
const SwTxtAttr *SwHyperlinkIter_Impl::next()
|
|
{
|
|
const SwTxtAttr *pAttr = 0;
|
|
if( pHints )
|
|
{
|
|
while( !pAttr && nPos < pHints->Count() )
|
|
{
|
|
const SwTxtAttr *pHt = (*pHints)[nPos];
|
|
if( RES_TXTATR_INETFMT == pHt->Which() )
|
|
{
|
|
xub_StrLen nHtStt = *pHt->GetStart();
|
|
xub_StrLen nHtEnd = *pHt->GetAnyEnd();
|
|
if( nHtEnd > nHtStt &&
|
|
( (nHtStt >= nStt && nHtStt < nEnd) ||
|
|
(nHtEnd > nStt && nHtEnd <= nEnd) ) )
|
|
{
|
|
pAttr = pHt;
|
|
}
|
|
}
|
|
++nPos;
|
|
}
|
|
}
|
|
|
|
return pAttr;
|
|
};
|
|
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getHyperLinkCount()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
CHECK_FOR_DEFUNC( XAccessibleHypertext );
|
|
|
|
sal_Int32 nCount = 0;
|
|
// #i77108# - provide hyperlinks also in editable documents.
|
|
// if( !IsEditableState() )
|
|
// <--
|
|
{
|
|
const SwTxtFrm *pTxtFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
SwHyperlinkIter_Impl aIter( pTxtFrm );
|
|
while( aIter.next() )
|
|
nCount++;
|
|
}
|
|
|
|
return nCount;
|
|
}
|
|
|
|
uno::Reference< XAccessibleHyperlink > SAL_CALL
|
|
SwAccessibleParagraph::getHyperLink( sal_Int32 nLinkIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC( XAccessibleHypertext );
|
|
|
|
uno::Reference< XAccessibleHyperlink > xRet;
|
|
|
|
// #i77108#
|
|
{
|
|
const SwTxtFrm *pTxtFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
SwHyperlinkIter_Impl aHIter( pTxtFrm );
|
|
while( nLinkIndex-- )
|
|
aHIter.next();
|
|
|
|
const SwTxtAttr *pHt = aHIter.next();
|
|
if( pHt )
|
|
{
|
|
if( !pHyperTextData )
|
|
pHyperTextData = new SwAccessibleHyperTextData;
|
|
SwAccessibleHyperTextData::iterator aIter =
|
|
pHyperTextData ->find( pHt );
|
|
if( aIter != pHyperTextData->end() )
|
|
{
|
|
xRet = (*aIter).second;
|
|
}
|
|
if( !xRet.is() )
|
|
{
|
|
sal_Int32 nHStt= GetPortionData().GetAccessiblePosition(
|
|
max( aHIter.startIdx(), *pHt->GetStart() ) );
|
|
sal_Int32 nHEnd= GetPortionData().GetAccessiblePosition(
|
|
min( aHIter.endIdx(), *pHt->GetAnyEnd() ) );
|
|
xRet = new SwAccessibleHyperlink( aHIter.getCurrHintPos(),
|
|
this, nHStt, nHEnd );
|
|
if( aIter != pHyperTextData->end() )
|
|
{
|
|
(*aIter).second = xRet;
|
|
}
|
|
else
|
|
{
|
|
SwAccessibleHyperTextData::value_type aEntry( pHt, xRet );
|
|
pHyperTextData->insert( aEntry );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if( !xRet.is() )
|
|
throw lang::IndexOutOfBoundsException();
|
|
|
|
return xRet;
|
|
}
|
|
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getHyperLinkIndex( sal_Int32 nCharIndex )
|
|
throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
CHECK_FOR_DEFUNC( XAccessibleHypertext );
|
|
|
|
// parameter checking
|
|
sal_Int32 nLength = GetString().getLength();
|
|
if ( ! IsValidPosition( nCharIndex, nLength ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
sal_Int32 nRet = -1;
|
|
// #i77108#
|
|
{
|
|
const SwTxtFrm *pTxtFrm = static_cast<const SwTxtFrm*>( GetFrm() );
|
|
SwHyperlinkIter_Impl aHIter( pTxtFrm );
|
|
|
|
xub_StrLen nIdx = GetPortionData().GetModelPosition( nCharIndex );
|
|
sal_Int32 nPos = 0;
|
|
const SwTxtAttr *pHt = aHIter.next();
|
|
while( pHt && !(nIdx >= *pHt->GetStart() && nIdx < *pHt->GetAnyEnd()) )
|
|
{
|
|
pHt = aHIter.next();
|
|
nPos++;
|
|
}
|
|
|
|
if( pHt )
|
|
nRet = nPos;
|
|
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
// #i71360#, #i108125# - adjustments for change tracking text markup
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getTextMarkupCount( sal_Int32 nTextMarkupType )
|
|
throw (lang::IllegalArgumentException,
|
|
uno::RuntimeException)
|
|
{
|
|
std::auto_ptr<SwTextMarkupHelper> pTextMarkupHelper;
|
|
switch ( nTextMarkupType )
|
|
{
|
|
case text::TextMarkupType::TRACK_CHANGE_INSERTION:
|
|
case text::TextMarkupType::TRACK_CHANGE_DELETION:
|
|
case text::TextMarkupType::TRACK_CHANGE_FORMATCHANGE:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper(
|
|
GetPortionData(),
|
|
*(mpParaChangeTrackInfo->getChangeTrackingTextMarkupList( nTextMarkupType ) )) );
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper( GetPortionData(), *GetTxtNode() ) );
|
|
}
|
|
}
|
|
|
|
return pTextMarkupHelper->getTextMarkupCount( nTextMarkupType );
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SAL_CALL
|
|
SwAccessibleParagraph::getTextMarkup( sal_Int32 nTextMarkupIndex,
|
|
sal_Int32 nTextMarkupType )
|
|
throw (lang::IndexOutOfBoundsException,
|
|
lang::IllegalArgumentException,
|
|
uno::RuntimeException)
|
|
{
|
|
std::auto_ptr<SwTextMarkupHelper> pTextMarkupHelper;
|
|
switch ( nTextMarkupType )
|
|
{
|
|
case text::TextMarkupType::TRACK_CHANGE_INSERTION:
|
|
case text::TextMarkupType::TRACK_CHANGE_DELETION:
|
|
case text::TextMarkupType::TRACK_CHANGE_FORMATCHANGE:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper(
|
|
GetPortionData(),
|
|
*(mpParaChangeTrackInfo->getChangeTrackingTextMarkupList( nTextMarkupType ) )) );
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper( GetPortionData(), *GetTxtNode() ) );
|
|
}
|
|
}
|
|
|
|
return pTextMarkupHelper->getTextMarkup( nTextMarkupIndex, nTextMarkupType );
|
|
}
|
|
|
|
uno::Sequence< /*accessibility::*/TextSegment > SAL_CALL
|
|
SwAccessibleParagraph::getTextMarkupAtIndex( sal_Int32 nCharIndex,
|
|
sal_Int32 nTextMarkupType )
|
|
throw (lang::IndexOutOfBoundsException,
|
|
lang::IllegalArgumentException,
|
|
uno::RuntimeException)
|
|
{
|
|
// parameter checking
|
|
const sal_Int32 nLength = GetString().getLength();
|
|
if ( ! IsValidPosition( nCharIndex, nLength ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
std::auto_ptr<SwTextMarkupHelper> pTextMarkupHelper;
|
|
switch ( nTextMarkupType )
|
|
{
|
|
case text::TextMarkupType::TRACK_CHANGE_INSERTION:
|
|
case text::TextMarkupType::TRACK_CHANGE_DELETION:
|
|
case text::TextMarkupType::TRACK_CHANGE_FORMATCHANGE:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper(
|
|
GetPortionData(),
|
|
*(mpParaChangeTrackInfo->getChangeTrackingTextMarkupList( nTextMarkupType ) )) );
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
pTextMarkupHelper.reset( new SwTextMarkupHelper( GetPortionData(), *GetTxtNode() ) );
|
|
}
|
|
}
|
|
|
|
return pTextMarkupHelper->getTextMarkupAtIndex( nCharIndex, nTextMarkupType );
|
|
}
|
|
// <--
|
|
|
|
// #i89175#
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getLineNumberAtIndex( sal_Int32 nIndex )
|
|
throw (lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException)
|
|
{
|
|
// parameter checking
|
|
const sal_Int32 nLength = GetString().getLength();
|
|
if ( ! IsValidPosition( nIndex, nLength ) )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
const sal_Int32 nLineNo = GetPortionData().GetLineNo( nIndex );
|
|
return nLineNo;
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SAL_CALL
|
|
SwAccessibleParagraph::getTextAtLineNumber( sal_Int32 nLineNo )
|
|
throw (lang::IndexOutOfBoundsException,
|
|
uno::RuntimeException)
|
|
{
|
|
// parameter checking
|
|
if ( nLineNo < 0 ||
|
|
nLineNo >= GetPortionData().GetLineCount() )
|
|
{
|
|
throw lang::IndexOutOfBoundsException();
|
|
}
|
|
|
|
i18n::Boundary aLineBound;
|
|
GetPortionData().GetBoundaryOfLine( nLineNo, aLineBound );
|
|
|
|
/*accessibility::*/TextSegment aTextAtLine;
|
|
const ::rtl::OUString rText = GetString();
|
|
aTextAtLine.SegmentText = rText.copy( aLineBound.startPos,
|
|
aLineBound.endPos - aLineBound.startPos );
|
|
aTextAtLine.SegmentStart = aLineBound.startPos;
|
|
aTextAtLine.SegmentEnd = aLineBound.endPos;
|
|
|
|
return aTextAtLine;
|
|
}
|
|
|
|
/*accessibility::*/TextSegment SAL_CALL SwAccessibleParagraph::getTextAtLineWithCaret()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
const sal_Int32 nLineNoOfCaret = getNumberOfLineWithCaret();
|
|
|
|
if ( nLineNoOfCaret >= 0 &&
|
|
nLineNoOfCaret < GetPortionData().GetLineCount() )
|
|
{
|
|
return getTextAtLineNumber( nLineNoOfCaret );
|
|
}
|
|
|
|
return /*accessibility::*/TextSegment();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL SwAccessibleParagraph::getNumberOfLineWithCaret()
|
|
throw (uno::RuntimeException)
|
|
{
|
|
const sal_Int32 nCaretPos = getCaretPosition();
|
|
const sal_Int32 nLength = GetString().getLength();
|
|
if ( !IsValidPosition( nCaretPos, nLength ) )
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
sal_Int32 nLineNo = GetPortionData().GetLineNo( nCaretPos );
|
|
|
|
// special handling for cursor positioned at end of text line via End key
|
|
if ( nCaretPos != 0 )
|
|
{
|
|
i18n::Boundary aLineBound;
|
|
GetPortionData().GetBoundaryOfLine( nLineNo, aLineBound );
|
|
if ( nCaretPos == aLineBound.startPos )
|
|
{
|
|
SwCrsrShell* pCrsrShell = SwAccessibleParagraph::GetCrsrShell();
|
|
if ( pCrsrShell != 0 )
|
|
{
|
|
const awt::Rectangle aCharRect = getCharacterBounds( nCaretPos );
|
|
|
|
const SwRect& aCursorCoreRect = pCrsrShell->GetCharRect();
|
|
// translate core coordinates into accessibility coordinates
|
|
Window *pWin = GetWindow();
|
|
CHECK_FOR_WINDOW( XAccessibleComponent, pWin );
|
|
|
|
Rectangle aScreenRect( GetMap()->CoreToPixel( aCursorCoreRect.SVRect() ));
|
|
|
|
SwRect aFrmLogBounds( GetBounds( *(GetMap()) ) ); // twip rel to doc root
|
|
Point aFrmPixPos( GetMap()->CoreToPixel( aFrmLogBounds.SVRect() ).TopLeft() );
|
|
aScreenRect.Move( -aFrmPixPos.X(), -aFrmPixPos.Y() );
|
|
|
|
// convert into AWT Rectangle
|
|
const awt::Rectangle aCursorRect( aScreenRect.Left(),
|
|
aScreenRect.Top(),
|
|
aScreenRect.GetWidth(),
|
|
aScreenRect.GetHeight() );
|
|
|
|
if ( aCharRect.X != aCursorRect.X ||
|
|
aCharRect.Y != aCursorRect.Y )
|
|
{
|
|
--nLineNo;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return nLineNo;
|
|
}
|
|
|
|
// #i108125#
|
|
void SwAccessibleParagraph::Modify( SfxPoolItem* pOld, SfxPoolItem* pNew )
|
|
{
|
|
mpParaChangeTrackInfo->reset();
|
|
|
|
SwClient::Modify( pOld, pNew );
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|